package com.planet_ink.coffee_mud.Abilities.Common; import com.planet_ink.coffee_mud.core.interfaces.*; import com.planet_ink.coffee_mud.core.*; import com.planet_ink.coffee_mud.core.collections.*; import com.planet_ink.coffee_mud.Abilities.Common.CraftingSkill.CraftParms; import com.planet_ink.coffee_mud.Abilities.Common.CraftingSkill.CraftingActivity; import com.planet_ink.coffee_mud.Abilities.interfaces.*; import com.planet_ink.coffee_mud.Areas.interfaces.*; import com.planet_ink.coffee_mud.Behaviors.interfaces.*; import com.planet_ink.coffee_mud.CharClasses.interfaces.*; import com.planet_ink.coffee_mud.Commands.interfaces.*; import com.planet_ink.coffee_mud.Common.interfaces.*; import com.planet_ink.coffee_mud.Common.interfaces.Session.InputCallback; import com.planet_ink.coffee_mud.Exits.interfaces.*; import com.planet_ink.coffee_mud.Items.interfaces.*; import com.planet_ink.coffee_mud.Libraries.interfaces.AchievementLibrary; import com.planet_ink.coffee_mud.Libraries.interfaces.ListingLibrary; import com.planet_ink.coffee_mud.Libraries.interfaces.MaterialLibrary; import com.planet_ink.coffee_mud.Locales.interfaces.*; import com.planet_ink.coffee_mud.MOBS.interfaces.*; import com.planet_ink.coffee_mud.Races.interfaces.*; import java.util.*; /* Copyright 2003-2019 Bo Zimmerman Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ public class ScrimShaw extends EnhancedCraftingSkill implements ItemCraftor, MendingSkill { @Override public String ID() { return "ScrimShaw"; } private final static String localizedName = CMLib.lang().L("Scrimshawing"); @Override public String name() { return localizedName; } private static final String[] triggerStrings = I(new String[] { "SCRIM", "SCRIMSHAWING" }); @Override public String[] triggerStrings() { return triggerStrings; } @Override public String supportedResourceString() { return "BONE|IVORY"; } @Override public String parametersFormat() { return "ITEM_NAME\tITEM_LEVEL\tBUILD_TIME_TICKS\tMATERIALS_REQUIRED\tITEM_BASE_VALUE\t" +"ITEM_CLASS_ID\tSTATUE||CONTAINER_TYPE_OR_LIDLOCK||WEAPON_CLASS||RIDE_BASIS||CODED_WEAR_LOCATION\t" +"BASE_ARMOR_AMOUNT||CONTAINER_CAPACITY||BASE_DAMAGE||LIGHT_DURATION||MAX_WAND_USES||DICE_SIDES\t" +"CODED_SPELL_LIST"; } //protected static final int RCP_FINALNAME=0; //protected static final int RCP_LEVEL=1; //protected static final int RCP_TICKS=2; protected static final int RCP_WOOD = 3; protected static final int RCP_VALUE = 4; protected static final int RCP_CLASSTYPE = 5; protected static final int RCP_MISCTYPE = 6; protected static final int RCP_CAPACITY = 7; protected static final int RCP_SPELL = 8; protected Item key=null; @Override public boolean tick(final Tickable ticking, final int tickID) { if((affected!=null)&&(affected instanceof MOB)&&(tickID==Tickable.TICKID_MOB)) { if(buildingI==null) unInvoke(); } return super.tick(ticking,tickID); } @Override public String parametersFile() { return "scrimshaw.txt"; } @Override protected List<List<String>> loadRecipes() { return super.loadRecipes(parametersFile()); } @Override public void unInvoke() { if(canBeUninvoked()) { if(affected instanceof MOB) { final MOB mob=(MOB)affected; if((buildingI!=null)&&(!aborted)) { if(messedUp) { if(activity == CraftingActivity.MENDING) messedUpCrafting(mob); else if(activity == CraftingActivity.LEARNING) { commonEmote(mob,L("<S-NAME> fail(s) to learn how to make @x1.",buildingI.name())); buildingI.destroy(); } else commonTell(mob,L("<S-NAME> mess(es) up scrimshawing @x1.",buildingI.name(mob))); } else { if(activity == CraftingActivity.MENDING) { buildingI.setUsesRemaining(100); CMLib.achievements().possiblyBumpAchievement(mob, AchievementLibrary.Event.MENDER, 1, this); } else if(activity==CraftingActivity.LEARNING) { deconstructRecipeInto(mob, buildingI, recipeHolder ); buildingI.destroy(); } else { dropAWinner(mob,buildingI); CMLib.achievements().possiblyBumpAchievement(mob, AchievementLibrary.Event.CRAFTING, 1, this); if(key!=null) { dropAWinner(mob,key); if(buildingI instanceof Container) key.setContainer((Container)buildingI); } } } } buildingI=null; key=null; activity = CraftingActivity.CRAFTING; } } super.unInvoke(); } @Override public boolean mayICraft(final Item I) { if(I==null) return false; if(!super.mayBeCrafted(I)) return false; if((I.material()!=RawMaterial.RESOURCE_BONE) &&(I.material()!=RawMaterial.RESOURCE_IVORY)) return false; if(CMLib.flags().isDeadlyOrMaliciousEffect(I)) return false; if(I instanceof Rideable) { final Rideable R=(Rideable)I; final int rideType=R.rideBasis(); switch(rideType) { case Rideable.RIDEABLE_SLEEP: case Rideable.RIDEABLE_SIT: case Rideable.RIDEABLE_TABLE: return true; default: return false; } } if(I instanceof Shield) return true; if(I instanceof Ammunition) return true; if(I instanceof Armor) return (I.rawProperLocationBitmap()&Wearable.WORN_FEET)==0; if(I instanceof Weapon) return !((I instanceof AmmunitionWeapon) && ((AmmunitionWeapon)I).requiresAmmunition()); if(I instanceof Container) return true; if((I instanceof Drink)&&(!(I instanceof Potion))) return false; if(I instanceof FalseLimb) return true; if((I instanceof Light)&&((I.rawProperLocationBitmap()&Wearable.WORN_MOUTH)>0)) return true; if(I.rawProperLocationBitmap()==Wearable.WORN_HELD) return true; return (isANativeItem(I.Name())); } @Override public boolean supportsMending(final Physical item) { return canMend(null, item, true); } @Override protected boolean canMend(final MOB mob, final Environmental E, final boolean quiet) { if(!super.canMend(mob,E,quiet)) return false; if((!(E instanceof Item))||(!mayICraft((Item)E))) { if(!quiet) commonTell(mob,L("That's not a scrimshawable item.")); return false; } return true; } @Override public String getDecodedComponentsDescription(final MOB mob, final List<String> recipe) { return super.getComponentDescription( mob, recipe, RCP_WOOD ); } @Override public boolean invoke(final MOB mob, final List<String> commands, final Physical givenTarget, final boolean auto, final int asLevel) { return autoGenInvoke(mob,commands,givenTarget,auto,asLevel,0,false,new Vector<Item>(0)); } @Override protected boolean autoGenInvoke(final MOB mob, final List<String> commands, final Physical givenTarget, final boolean auto, final int asLevel, final int autoGenerate, final boolean forceLevels, final List<Item> crafted) { final List<String> originalCommands = new XVector<String>(commands); if(super.checkStop(mob, commands)) return true; if(super.checkInfo(mob, commands)) return true; final PairVector<EnhancedExpertise,Integer> enhancedTypes=enhancedTypes(mob,commands); int recipeLevel = 1; randomRecipeFix(mob,addRecipes(mob,loadRecipes()),commands,autoGenerate); if(commands.size()==0) { commonTell(mob,L("Scrim what? Enter \"scrim list\" for a list, \"scrim info <item>\", \"scrim scan\"," + " \"scrim learn <item>\" to gain recipes, \"scrim mend <item>\", or \"scrim stop\" to cancel.")); return false; } if((!auto) &&(commands.size()>0) &&((commands.get(0)).equalsIgnoreCase("bundle"))) { bundling=true; if(super.invoke(mob,commands,givenTarget,auto,asLevel)) return super.bundle(mob,commands); return false; } final List<List<String>> recipes=addRecipes(mob,loadRecipes()); final String str=commands.get(0); String startStr=null; bundling=false; int duration=4; if(str.equalsIgnoreCase("list")) { String mask=CMParms.combine(commands,1); boolean allFlag=false; if(mask.equalsIgnoreCase("all")) { allFlag=true; mask=""; } final int[] cols={ CMLib.lister().fixColWidth(23,mob.session()), CMLib.lister().fixColWidth(3,mob.session()), CMLib.lister().fixColWidth(10,mob.session()) }; final StringBuffer buf=new StringBuffer(""); buf.append(L("@x1 @x2 @x3 ",CMStrings.padRight(L("Item"),cols[0]),CMStrings.padRight(L("Lvl"),cols[1]),CMStrings.padRight(L("Bone req"), cols[2]))); buf.append(L("@x1 @x2 @x3\n\r",CMStrings.padRight(L("Item"),cols[0]),CMStrings.padRight(L("Lvl"),cols[1]),L("Bone req"))); final List<List<String>> listRecipes=((mask.length()==0) || mask.equalsIgnoreCase("all")) ? recipes : super.matchingRecipeNames(recipes, mask, true); int col=0; final int numCols=2; for(int r=0;r<listRecipes.size();r++) { final List<String> V=listRecipes.get(r); if(V.size()>0) { final String item=replacePercent(V.get(RCP_FINALNAME),""); final int level=CMath.s_int(V.get(RCP_LEVEL)); final String wood=getComponentDescription(mob,V,RCP_WOOD); if((level<=xlevel(mob))||allFlag) { col++; if(col>=numCols) { buf.append(CMStrings.padRight(item,cols[0])+" "+CMStrings.padRight(""+level,cols[1])+" "+wood+"\n\r"); col=0; } else buf.append(CMStrings.padRight(item,cols[0])+" "+CMStrings.padRight(""+level,cols[1])+" "+CMStrings.padRight(wood,cols[2])+" "); } } } if(col>0) buf.append("\n\r"); commonTell(mob,buf.toString()); enhanceList(mob); return true; } else if(((commands.get(0))).equalsIgnoreCase("learn")) { return doLearnRecipe(mob, commands, givenTarget, auto, asLevel); } else if(str.equalsIgnoreCase("scan")) return publicScan(mob,commands); else if(str.equalsIgnoreCase("mend")) { buildingI=null; activity = CraftingActivity.CRAFTING; key=null; messedUp=false; final Vector<String> newCommands=CMParms.parse(CMParms.combine(commands,1)); buildingI=getTargetItemFavorMOB(mob,mob.location(),givenTarget,newCommands,Wearable.FILTER_UNWORNONLY); if(!canMend(mob,buildingI,false)) return false; activity = CraftingActivity.MENDING; if(!super.invoke(mob,commands,givenTarget,auto,asLevel)) return false; startStr=L("<S-NAME> start(s) mending @x1.",buildingI.name()); displayText=L("You are mending @x1",buildingI.name()); verb=L("mending @x1",buildingI.name()); } else { buildingI=null; activity = CraftingActivity.CRAFTING; key=null; messedUp=false; aborted=false; String statue=null; if((commands.size()>1)&&(commands.get(commands.size()-1)).startsWith("STATUE=")) { statue=((commands.get(commands.size()-1)).substring(7)).trim(); if(statue.length()==0) statue=null; else commands.remove(commands.size()-1); } int amount=-1; if((commands.size()>1)&&(CMath.isNumber(commands.get(commands.size()-1)))) { amount=CMath.s_int(commands.get(commands.size()-1)); commands.remove(commands.size()-1); } final String recipeName=CMParms.combine(commands,0); List<String> foundRecipe=null; final List<List<String>> matches=matchingRecipeNames(recipes,recipeName,true); for(int r=0;r<matches.size();r++) { final List<String> V=matches.get(r); if(V.size()>0) { final int level=CMath.s_int(V.get(RCP_LEVEL)); if((autoGenerate>0)||(level<=xlevel(mob))) { foundRecipe=V; recipeLevel=level; break; } } } if(foundRecipe==null) { commonTell(mob,L("You don't know how to scrim a '@x1'. Try \"scrim list\" for a list.",recipeName)); return false; } final String woodRequiredStr = foundRecipe.get(RCP_WOOD); final int[] compData = new int[CF_TOTAL]; final String realRecipeName=replacePercent(foundRecipe.get(RCP_FINALNAME),""); final List<Object> componentsFoundList=getAbilityComponents(mob, woodRequiredStr, "make "+CMLib.english().startWithAorAn(realRecipeName),autoGenerate,compData,1); if(componentsFoundList==null) return false; int woodRequired=CMath.s_int(woodRequiredStr); woodRequired=adjustWoodRequired(woodRequired,mob); final String misctype=foundRecipe.get(RCP_MISCTYPE); final int[] pm={RawMaterial.RESOURCE_BONE,RawMaterial.RESOURCE_IVORY}; bundling=misctype.equalsIgnoreCase("BUNDLE"); final int[][] data=fetchFoundResourceData(mob, woodRequired,"bone",pm, 0,null,null, bundling, autoGenerate, enhancedTypes); if(data==null) return false; fixDataForComponents(data,woodRequiredStr,(autoGenerate>0) && (woodRequired==0),componentsFoundList, 1); woodRequired=data[0][FOUND_AMT]; if(amount>woodRequired) woodRequired=amount; final Session session=mob.session(); if((misctype.equalsIgnoreCase("statue")) &&(session!=null) &&((statue==null)||(statue.trim().length()==0))) { final Ability me=this; final Physical target=givenTarget; if(autoGenerate>0) statue=mob.Name(); else session.prompt(new InputCallback(InputCallback.Type.PROMPT,"",0) { @Override public void showPrompt() { session.promptPrint(L("What is this a statue of?\n\r: ")); } @Override public void timedOut() { } @Override public void callBack() { final String of=this.input; if((of.trim().length()==0)||(of.indexOf('<')>=0)) return; final Vector<String> newCommands=new XVector<String>(originalCommands); newCommands.add("STATUE="+of); me.invoke(mob, newCommands, target, auto, asLevel); } }); return false; } if(!super.invoke(mob,commands,givenTarget,auto,asLevel)) return false; final MaterialLibrary.DeadResourceRecord deadMats; if((componentsFoundList.size() > 0)||(autoGenerate>0)) deadMats = new MaterialLibrary.DeadResourceRecord(); else { deadMats = CMLib.materials().destroyResources(mob.location(),woodRequired, data[0][FOUND_CODE],data[0][FOUND_SUB],data[1][FOUND_CODE],data[1][FOUND_SUB]); } final MaterialLibrary.DeadResourceRecord deadComps = CMLib.ableComponents().destroyAbilityComponents(componentsFoundList); final int lostValue=autoGenerate>0?0:(deadMats.lostValue + deadComps.lostValue); buildingI=CMClass.getItem(foundRecipe.get(RCP_CLASSTYPE)); final Item buildingI=this.buildingI; if(buildingI==null) { commonTell(mob,L("There's no such thing as a @x1!!!",foundRecipe.get(RCP_CLASSTYPE))); return false; } duration=getDuration(CMath.s_int(foundRecipe.get(RCP_TICKS)),mob,CMath.s_int(foundRecipe.get(RCP_LEVEL)),4); buildingI.setMaterial(super.getBuildingMaterial(woodRequired, data, compData)); String itemName=determineFinalName(foundRecipe.get(RCP_FINALNAME),buildingI.material(),deadMats,deadComps); if(bundling) itemName="a "+woodRequired+"# "+itemName; else itemName=CMLib.english().startWithAorAn(itemName); buildingI.setName(itemName); startStr=L("<S-NAME> start(s) scrimshawing @x1.",buildingI.name()); displayText=L("You are scrimshawing @x1",buildingI.name()); verb=L("scrimshawing @x1",buildingI.name()); buildingI.setDisplayText(L("@x1 lies here",itemName)); buildingI.setDescription(determineDescription(itemName, buildingI.material(), deadMats, deadComps)); buildingI.basePhyStats().setWeight(getStandardWeight(woodRequired+compData[CF_AMOUNT],data[1][FOUND_CODE], bundling)); buildingI.setBaseValue(CMath.s_int(foundRecipe.get(RCP_VALUE))+(woodRequired*(RawMaterial.CODES.VALUE(data[0][FOUND_CODE])))); buildingI.basePhyStats().setLevel(CMath.s_int(foundRecipe.get(RCP_LEVEL))); setBrand(mob, buildingI); final int capacity=CMath.s_int(foundRecipe.get(RCP_CAPACITY)); final String spell=(foundRecipe.size()>RCP_SPELL)?foundRecipe.get(RCP_SPELL).trim():""; if(bundling) buildingI.setBaseValue(lostValue); addSpells(buildingI,spell,deadMats.lostProps,deadComps.lostProps); key=null; if((misctype.equalsIgnoreCase("statue")) &&(statue!=null) &&(statue.trim().length()>0)) { buildingI.setName(L("@x1 of @x2",itemName,statue.trim())); buildingI.setDisplayText(L("@x1 of @x2 is here",itemName,statue.trim())); buildingI.setDescription(L("@x1 of @x2. ",itemName,statue.trim())); } else if((buildingI instanceof Container) &&(!(buildingI instanceof Armor))) { if(capacity>0) ((Container)buildingI).setCapacity(capacity+woodRequired); if(misctype.indexOf("LID")>=0) ((Container)buildingI).setDoorsNLocks(true,false,true,false,false,false); else if(misctype.indexOf("LOCK")>=0) { ((Container)buildingI).setDoorsNLocks(true,false,true,true,false,true); ((Container)buildingI).setKeyName(Double.toString(Math.random())); key=CMClass.getItem("GenKey"); ((DoorKey)key).setKey(((Container)buildingI).keyName()); key.setName(L("a key")); key.setDisplayText(L("a small key sits here")); key.setDescription(L("looks like a key to @x1",buildingI.name())); key.recoverPhyStats(); setBrand(mob, key); key.text(); } if(!(buildingI instanceof Armor)) ((Container)buildingI).setContainTypes(getContainerType(misctype)); } if(buildingI instanceof Weapon) { ((Weapon)buildingI).basePhyStats().setAttackAdjustment(baseYield()+abilityCode()-1); ((Weapon)buildingI).setWeaponClassification(Weapon.CLASS_FLAILED); setWeaponTypeClass((Weapon)buildingI,misctype,Weapon.TYPE_BASHING,Weapon.TYPE_PIERCING); buildingI.basePhyStats().setDamage(capacity); ((Weapon)buildingI).setRawProperLocationBitmap(Wearable.WORN_WIELD|Wearable.WORN_HELD); ((Weapon)buildingI).setRawLogicalAnd(false); } if((buildingI instanceof Armor)&&(!(buildingI instanceof FalseLimb))) { ((Armor)buildingI).basePhyStats().setArmor(0); if(capacity!=0) ((Armor)buildingI).basePhyStats().setArmor(capacity+(baseYield()+abilityCode()-1)); setWearLocation(buildingI,misctype,0); } if(buildingI instanceof Rideable) { setRideBasis((Rideable)buildingI,misctype); } if(buildingI.ID().endsWith("Dice")) { buildingI.basePhyStats().setAbility(capacity); } if(buildingI instanceof Light) { ((Light)buildingI).setDuration(capacity); if((buildingI instanceof Container) &&(((Container)buildingI).containTypes()!=Container.CONTAIN_SMOKEABLES)) ((Container)buildingI).setCapacity(0); } buildingI.recoverPhyStats(); buildingI.text(); buildingI.recoverPhyStats(); } messedUp=!proficiencyCheck(mob,0,auto); if(bundling) { messedUp=false; duration=1; verb=L("bundling @x1",RawMaterial.CODES.NAME(buildingI.material()).toLowerCase()); startStr=L("<S-NAME> start(s) @x1.",verb); displayText=L("You are @x1",verb); } if(autoGenerate>0) { crafted.add(buildingI); return true; } final CMMsg msg=CMClass.getMsg(mob,buildingI,this,getActivityMessageType(),startStr); if(mob.location().okMessage(mob,msg)) { mob.location().send(mob,msg); buildingI=(Item)msg.target(); beneficialAffect(mob,mob,asLevel,duration); enhanceItem(mob,buildingI,recipeLevel,enhancedTypes); } else if(bundling) { messedUp=false; aborted=false; unInvoke(); } return true; } }