package com.planet_ink.coffee_mud.Abilities.Misc; import com.planet_ink.coffee_mud.Abilities.StdAbility; 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.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.Exits.interfaces.*; import com.planet_ink.coffee_mud.Items.interfaces.*; import com.planet_ink.coffee_mud.Libraries.interfaces.*; 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 2004-2016 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. */ @SuppressWarnings({"unchecked","rawtypes"}) public class Injury extends StdAbility implements LimbDamage, HealthCondition { @Override public String ID() { return "Injury"; } private final static String localizedName = CMLib.lang().L("Injury"); @Override public String name() { return localizedName; } public PairVector<String,Integer>[] injuries=new PairVector[Race.BODY_PARTS]; protected CMMsg lastMsg = null; protected String lastLoc = null; public int lastHP = -1; //public final static String[] BODYPARTSTR={ // "ANTENEA","EYE","EAR","HEAD","NECK","ARM","HAND","TORSO","LEG","FOOT", // "NOSE","GILL","MOUTH","WAIST","TAIL","WING"}; public final static int[] INJURYCHANCE= { 3,3,3,11,3,12,5,35,13,5,3,0,0,3,3,3 }; @Override public String getHealthConditionDesc() { final StringBuffer buf=new StringBuffer(""); Pair<String,Integer> O=null; PairVector<String,Integer> V=null; try { if(injuries!=null) { for(int i=0;i<Race.BODY_PARTS;i++) { V=injuries[i]; if(V!=null) for(int i2=0;i2<V.size();i2++) { O=V.elementAt(i2); String wounds=""; final int dmg = O.second.intValue(); if (dmg<5) wounds=("a bruised "); else if (dmg<10) wounds=("a scratched "); else if (dmg<20) wounds=("a cut "); else if (dmg<30) wounds=("a sliced "); else if (dmg<40) wounds=("a gashed "); else if (dmg<60) wounds=("a bloody "); else if ((dmg<75)||(i==Race.BODY_TORSO)) wounds=("a mangled "); else if ((dmg<100)||(i==Race.BODY_HEAD)) wounds=("a dangling "); else wounds=("a shredded "); buf.append(", "+wounds+O.first.toLowerCase()+" ("+dmg+"%)"); } } } } catch (final Exception e) { } if(buf.length()==0) return ""; return buf.substring(1); } @Override public String displayText() { final String buf=getHealthConditionDesc(); if(buf.length()==0) return ""; return "(Injuries:"+buf+")"; } @Override protected int canAffectCode() { return CAN_MOBS; } @Override protected int canTargetCode() { return CAN_MOBS; } @Override public int abstractQuality() { return Ability.QUALITY_INDIFFERENT; } @Override public boolean putInCommandlist() { return false; } @Override public boolean canBeUninvoked() { return true; } @Override public int classificationCode() { return Ability.ACODE_PROPERTY; } @Override public int usageType() { return USAGE_MOVEMENT | USAGE_MANA; } @Override public void unInvoke() { final Environmental E=affected; super.unInvoke(); if((E instanceof MOB)&&(canBeUninvoked())&&(!((MOB)E).amDead())) ((MOB)E).tell(L("Your injuries are healed.")); } @Override public String text() { PairVector<String,Integer> V=null; Pair<String,Integer> O=null; final StringBuffer buf=new StringBuffer(""); if(injuries!=null) for(int i=0;i<Race.BODY_PARTS;i++) { V=injuries[i]; if(V!=null) for(int i2=0;i2<V.size();i2++) { O=V.elementAt(i2); buf.append(i+":"+O.first.toLowerCase()+":"+O.second.intValue()+";"); } } return buf.toString(); } @Override public void setMiscText(String txt) { if(txt.startsWith("+")) { if(affected instanceof MOB) { final MOB mob=(MOB)affected; txt=txt.substring(1); final int x=txt.indexOf('='); if(x<0) return; final String chosenName=txt.substring(0,x); final String amount=txt.substring(x+1); Amputation A=(Amputation)mob.fetchEffect("Amputation"); if(A==null) { A=new Amputation(); A.setAffectedOne(mob); } final List<String> remains=A.unaffectedLimbSet(); if(mob.charStats().getBodyPart(Race.BODY_HEAD)>0) remains.add("head"); if(mob.charStats().getBodyPart(Race.BODY_TORSO)>0) remains.add("torso"); final int chosenOne=remains.indexOf(chosenName); if(chosenOne<0) return; if(injuries==null) injuries=new PairVector[Race.BODY_PARTS]; Integer chosenBodyLoc=Race.BODYPARTHASH_RL_LOWER.get(remains.get(chosenOne).toLowerCase().trim()); if(chosenBodyLoc!=null) { PairVector<String,Integer> bodyVec=injuries[chosenBodyLoc.intValue()]; if(bodyVec==null) { injuries[chosenBodyLoc.intValue()]=new PairVector(); bodyVec=injuries[chosenBodyLoc.intValue()]; } int whichInjury=-1; for(int i=0;i<bodyVec.size();i++) { final Pair<String,Integer> O=bodyVec.get(i); if(O.first.equalsIgnoreCase(remains.get(chosenOne))) { whichInjury=i; break; } } Pair<String,Integer> O=null; if(whichInjury<0) { O=new Pair<String,Integer>( remains.get(chosenOne).toLowerCase(), Integer.valueOf(0) ); bodyVec.addElement(O); whichInjury=bodyVec.size()-1; } O=bodyVec.get(whichInjury); O.second=Integer.valueOf(O.second.intValue()+CMath.s_int(amount)); if(O.second.intValue()>100) O.second=Integer.valueOf(100); } } } else if(txt.indexOf('/')>0) super.setMiscText(txt); else { injuries=new PairVector[Race.BODY_PARTS]; final List<String> sets=CMParms.parseSemicolons(txt,true); for(int s=0;s<sets.size();s++) { final String set=sets.get(s); final List<String> V=CMParms.parseAny(set,':',false); if(V.size()==3) { final int part=CMath.s_int(V.get(0)); if((part>=0)&&(part<Race.BODY_PARTS)) { final String msg=V.get(1); final int hurt=CMath.s_int(V.get(V.size()-1)); if(injuries[part]==null) injuries[part] = new PairVector(); injuries[part].add(new Pair<String,Integer>(msg,Integer.valueOf(hurt))); } } } } if(affected instanceof MOB) { final MOB mob=(MOB)affected; if(lastHP<0) lastHP=mob.curState().getHitPoints(); } } @Override public List<String> unaffectedLimbSet() { Amputation ampuA=(Amputation)affected.fetchEffect("Amputation"); if(ampuA==null) { ampuA=new Amputation(); ampuA.setAffectedOne(affected); } final List<String> remains=ampuA.unaffectedLimbSet(); if(affected instanceof MOB) { final MOB mob=(MOB)affected; if(mob.charStats().getBodyPart(Race.BODY_HEAD)>0) remains.add("head"); if(mob.charStats().getBodyPart(Race.BODY_TORSO)>0) remains.add("torso"); } final List<String> affected=affectedLimbNameSet(); final List<String> checks=new ArrayList<String>(remains); for(String limb : checks) { if(affected.contains(limb)) remains.remove(limb); } return remains; } @Override public Item damageLimb(String limbName) { if(affected!=null) { if(affected instanceof MOB) { boolean success=((MOB)affected).location().show(((MOB)affected),this,CMMsg.MSG_OK_VISUAL,L("^G<S-YOUPOSS> @x1 is injured!^?",limbName)); if(!success) return null; } } for(int p=0;p<Race.BODY_PARTS;p++) { if(limbName.equalsIgnoreCase(Race.BODYPARTSTR[p]) ||limbName.equalsIgnoreCase("right "+Race.BODYPARTSTR[p]) ||limbName.equalsIgnoreCase("left "+Race.BODYPARTSTR[p])) { PairVector<String,Integer> parts=injuries[p]; if(parts == null) { parts=new PairVector<String,Integer>(); injuries[p]=parts; } for(Pair<String,Integer> part : parts) { if(part.first.equalsIgnoreCase(limbName)) { if(part.second.intValue()<90) part.second = Integer.valueOf(part.second.intValue()+10); return null; } } parts.add(new Pair<String,Integer>(limbName.toLowerCase(),Integer.valueOf(10))); return null; } } return null; } @Override public List<String> affectedLimbNameSet() { List<String> list=new Vector<String>(1); PairVector<String,Integer> V=null; Pair<String,Integer> O=null; if(injuries!=null) { for(int i=0;i<Race.BODY_PARTS;i++) { V=injuries[i]; if(V!=null) { for(int i2=0;i2<V.size();i2++) { O=V.elementAt(i2); list.add(O.first.toLowerCase()); } } } } return list; } @Override public void restoreLimb(String limbName) { for(int partNum=0;partNum<Race.BODY_PARTS;partNum++) { if(injuries[partNum]!=null) { for(Pair<String,Integer> part : injuries[partNum]) { if(part.first.equalsIgnoreCase(limbName)) { injuries[partNum].remove(part); if(injuries[partNum].size()==0) injuries[partNum]=null; if((affected != null)&&(affectedLimbNameSet().size()==0)) affected.delEffect(this); return; } } } } } @Override public boolean tick(Tickable ticking, int tickID) { if((affected instanceof MOB)&&(tickID==Tickable.TICKID_MOB)) { final MOB mob=(MOB)affected; if(mob.curState().getHitPoints()>=mob.maxState().getHitPoints()) { for(int i=0;i<injuries.length;i++) injuries[i]=null; unInvoke(); } else if((mob.curState().getHitPoints()>lastHP)&&(lastHP>=0)) { final Vector<int[]> choicesToHeal=new Vector<int[]>(); for(int i=0;i<injuries.length;i++) { if(injuries[i]!=null) { for(int x=0;x<injuries[i].size();x++) { final int[] choice=new int[2]; choice[0] = i; choice[1] = x; choicesToHeal.addElement(choice); } } } if(choicesToHeal.size()==0) { for(int i=0;i<injuries.length;i++) injuries[i]=null; unInvoke(); } else { int pct=(int)Math.round(CMath.div(mob.curState().getHitPoints()-lastHP,mob.maxState().getHitPoints())*100.0); if(pct<=0) pct=1; int tries=100; while((pct>0)&&((--tries)>0)&&(choicesToHeal.size()>0)) { final int which=CMLib.dice().roll(1,choicesToHeal.size(),-1); final int[] choice=choicesToHeal.elementAt(which); if(choice[0]<injuries.length) { final PairVector<String,Integer> V=injuries[choice[0]]; if((V!=null)&&(choice[1]<V.size())) { final Pair<String,Integer> O=V.get(choice[1]); if(pct>O.second.intValue()) { V.removeElement(O); if(V.size()==0) injuries[choice[0]]=null; pct-=O.second.intValue(); choicesToHeal.removeElementAt(which); } else { O.second=Integer.valueOf(O.second.intValue()-pct); pct=0; } } } } } } lastHP=mob.curState().getHitPoints(); } return super.tick(ticking,tickID); } public static String[][] TRANSLATE= { {"<T-HIM-HER>","<T-HIS-HER>"}, {"<T-NAME>","<T-YOUPOSS>"}, {"<T-NAMESELF>","<T-YOUPOSS>"} }; public String fixMessageString(String message, String loc) { if(message==null) return null; int x=message.indexOf("<DAMAGE>"); if(x<0) x=message.indexOf("<DAMAGES>"); if(x<0) return message; int y=Integer.MAX_VALUE; int which=-1; for(int i=0;i<TRANSLATE.length;i++) { final int y1=message.indexOf(TRANSLATE[i][0],x); if((y1>x)&&(y1<y)){ y=y1; which=i;} } if(which>=0) message=message.substring(0,y)+TRANSLATE[which][1]+" "+loc+message.substring(y+TRANSLATE[which][0].length()); return message; } @Override public boolean okMessage(Environmental host, CMMsg msg) { if((msg.target()==affected) &&(msg.targetMinor()==CMMsg.TYP_DAMAGE) &&(msg.value()>0) &&(msg.target() instanceof MOB) &&(msg.targetMessage()!=null) &&(msg.targetMessage().indexOf("<DAMAGE>")>=0) &&(super.miscText.startsWith(msg.source().Name()+"/") // special directed damage stuff ||((CMProps.getIntVar(CMProps.Int.INJPCTHP)>=(int)Math.round(CMath.div(((MOB)msg.target()).curState().getHitPoints(),((MOB)msg.target()).maxState().getHitPoints())*100.0)) &&(CMLib.dice().rollPercentage()<=CMProps.getIntVar(CMProps.Int.INJPCTCHANCE))))) { final MOB mob=(MOB)msg.target(); Amputation ampuA=(Amputation)mob.fetchEffect("Amputation"); if(ampuA==null) { ampuA=new Amputation(); ampuA.setAffectedOne(mob); } final List<String> remains=ampuA.unaffectedLimbSet(); if(mob.charStats().getBodyPart(Race.BODY_HEAD)>0) remains.add("head"); if(mob.charStats().getBodyPart(Race.BODY_TORSO)>0) remains.add("torso"); if(remains.size()>0) { if(super.miscText.startsWith(msg.source().Name()+"/")) { final String remainLoc=super.miscText.substring(super.miscText.indexOf('/')+1).toLowerCase(); if(remains.contains(remainLoc)) { remains.clear(); remains.add(remainLoc); } else return super.okMessage(host, msg); } final int[] chances=new int[remains.size()]; int total=0; for(int x=0;x<remains.size();x++) { final String remain=remains.get(x); // is lowercase if(Race.BODYPARTHASH_RL_LOWER.containsKey(remain)) { final int bodyPart=Race.BODYPARTHASH_RL_LOWER.get(remain).intValue(); final int amount=INJURYCHANCE[bodyPart]; chances[x]+=amount; total+=amount; } } if(total>0) { int randomRoll=CMLib.dice().roll(1,total,-1); int chosenOne=-1; if((lastMsg!=null) &&(lastLoc!=null) &&((msg==lastMsg)||((lastMsg.trailerMsgs()!=null)&&(lastMsg.trailerMsgs().contains(msg)))) &&(remains.contains(lastLoc))) chosenOne=remains.indexOf(lastLoc); else if((super.miscText.startsWith(msg.source().Name()+"/")) &&(remains.contains(super.miscText.substring(msg.source().Name().length()+1)))) { chosenOne=remains.indexOf(super.miscText.substring(msg.source().Name().length()+1)); super.miscText=""; } else for(int i=0;i<chances.length;i++) { if(chances[i]>0) { chosenOne=i; randomRoll-=chances[i]; if(randomRoll<=0) break; } } final int BodyPct=(int)Math.round(CMath.div(msg.value(),mob.maxState().getHitPoints())*100.0); int LimbPct=BodyPct*CMProps.getIntVar(CMProps.Int.INJMULTIPLIER); if(LimbPct<1) LimbPct=1; Integer chosenBodyLoc=Race.BODYPARTHASH_RL_LOWER.get(remains.get(chosenOne)); if(chosenBodyLoc!=null) { lastMsg=msg; lastLoc=remains.get(chosenOne); PairVector<String,Integer> bodyVec=injuries[chosenBodyLoc.intValue()]; if(bodyVec==null) { injuries[chosenBodyLoc.intValue()]=new PairVector(); bodyVec=injuries[chosenBodyLoc.intValue()]; } int whichInjury=-1; for(int i=0;i<bodyVec.size();i++) { final Pair<String,Integer> O=bodyVec.get(i); if(O.first.equalsIgnoreCase(remains.get(chosenOne))) { whichInjury=i; break; } } final String newTarg=fixMessageString(msg.targetMessage(),remains.get(chosenOne).toLowerCase()); if(!newTarg.equalsIgnoreCase(msg.targetMessage())) { msg.modify(msg.source(),msg.target(),msg.tool(), msg.sourceCode(),fixMessageString(msg.sourceMessage(),remains.get(chosenOne).toLowerCase()), msg.targetCode(),newTarg, msg.othersCode(),fixMessageString(msg.othersMessage(),remains.get(chosenOne).toLowerCase())); Pair<String,Integer> O=null; if(whichInjury<0) { O=new Pair<String,Integer>( remains.get(chosenOne).toLowerCase(), Integer.valueOf(0)); bodyVec.addElement(O); whichInjury=bodyVec.size()-1; } O=bodyVec.get(whichInjury); O.second=Integer.valueOf(O.second.intValue()+LimbPct); if(O.second.intValue()>100) O.second=Integer.valueOf(100); if((O.second.intValue()>=100) ||((BodyPct>5) &&((msg.tool() instanceof Electronics)||(BodyPct>=CMProps.getIntVar(CMProps.Int.INJPCTHPAMP))))) { boolean proceed=(CMLib.dice().rollPercentage()<=CMProps.getIntVar(CMProps.Int.INJPCTCHANCEAMP)) &&(mob.phyStats().level()>=CMProps.getIntVar(CMProps.Int.INJMINLEVEL)); if(msg.tool() instanceof Weapon) { switch(((Weapon)msg.tool()).weaponDamageType()) { case Weapon.TYPE_FROSTING: case Weapon.TYPE_GASSING: proceed=false; break; default: break; } } if(Amputation.validamputees[chosenBodyLoc.intValue()]&&proceed) { bodyVec.removeElement(O); if(bodyVec.size()==0) injuries[chosenBodyLoc.intValue()]=null; if(ampuA.damageLimb(O.first.toLowerCase())!=null) { if(mob.fetchEffect(ampuA.ID())==null) { ampuA.makeLongLasting(); mob.addEffect(ampuA); } } } } } } } } } return super.okMessage(host,msg); } @Override public boolean invoke(MOB mob, List<String> commands, Physical givenTarget, boolean auto, int asLevel) { if(givenTarget.fetchEffect(ID())!=null) return false; super.tickDown=2; Ability A=(Ability)copyOf(); A.startTickDown(mob,givenTarget,Ability.TICKS_ALMOST_FOREVER); return true; } }