package com.planet_ink.coffee_mud.Abilities.Properties; 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 2001-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 Prop_HaveResister extends Property implements TriggeredAffect { @Override public String ID() { return "Prop_HaveResister"; } @Override public String name() { return "Resistance due to ownership"; } @Override protected int canAffectCode() { return Ability.CAN_ITEMS; } @Override public boolean bubbleAffect() { return true; } protected CharStats adjCharStats = null; protected String maskString = ""; protected boolean ignoreCharStats = true; protected long lastProtection = 0; protected int remainingProtection = 0; protected boolean alwaysWeapProt = false; protected volatile short lastEffectCount = 0; protected boolean hasEffectDuration = false; protected final Map<String, Integer> prots = new TreeMap<String, Integer>(); @Override public long flags() { return Ability.FLAG_RESISTER; } @Override public int triggerMask() { return TriggeredAffect.TRIGGER_GET; } @Override public void setMiscText(final String newText) { super.setMiscText(newText); adjCharStats=(CharStats)CMClass.getCommon("DefaultCharStats"); ignoreCharStats=true; String parmString=newText; final int maskindex=newText.toUpperCase().indexOf("MASK="); if(maskindex>0) { maskString=newText.substring(maskindex+5).trim(); parmString=newText.substring(0,maskindex).trim(); } parmString = parmString.toUpperCase(); final List<String> parmParts = CMParms.parseSpaces(parmString.toUpperCase(), true); final List<String> previousSet = new LinkedList<String>(); this.prots.clear(); for(String parts : parmParts) { Integer newPct = null; if(parts.endsWith("%")) { parts=parts.substring(0,parts.length()-1).trim(); } if(CMath.isInteger(parts)) { newPct = Integer.valueOf(CMath.s_int(parts)); } else if(CMath.isNumber(parts)) { final double d = CMath.s_double(parts); if((d > 1.0) || (d < -1.0)) newPct = Integer.valueOf((int)Math.round(d)); else newPct = Integer.valueOf((int)Math.round(d * 100.0)); } else if(parts.equals("ALWAYS")) this.alwaysWeapProt=true; else { previousSet.add(parts); } if(newPct != null) { for(final String previousKey : previousSet) this.prots.put(previousKey, newPct); previousSet.clear(); } } hasEffectDuration = this.prots.containsKey("DEBUF-DURATION"); for(final int i : CharStats.CODES.SAVING_THROWS()) { if(parmString.toUpperCase().indexOf(CharStats.CODES.NAME(i))>=0) adjCharStats.setStat(i,getProtection(CharStats.CODES.NAME(i))); else adjCharStats.setStat(i,getProtection(CMStrings.limit(CharStats.CODES.NAME(i),4))); if(adjCharStats.getStat(i)!=0) ignoreCharStats=false; } } protected void ensureStarted() { if(adjCharStats==null) setMiscText(text()); } @Override public void affectCharStats(final MOB affectedMOB, final CharStats affectedStats) { ensureStarted(); if((!ignoreCharStats) &&(canResist(affectedMOB)) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,affectedMOB,false)))) { for(final int i : CharStats.CODES.SAVING_THROWS()) affectedStats.setStat(i,affectedStats.getStat(i)+adjCharStats.getStat(i)); } super.affectCharStats(affectedMOB,affectedStats); } public boolean checkProtection(final String protType) { return this.prots.containsKey(protType); } public int getProtection(String protType) { protType = protType.toUpperCase(); if(!this.prots.containsKey(protType)) return 0; return this.prots.get(protType).intValue(); } protected int weaponProtection(final String kind, final int damage, final int myLevel, final int hisLevel) { if(this.alwaysWeapProt) { final int protection = getProtection(kind); return (int)Math.round(CMath.mul(damage,1.0-CMath.div(protection,100.0))); } int protection=remainingProtection; if((System.currentTimeMillis()-lastProtection)>=CMProps.getTickMillis()) { protection = (getProtection(kind) + (myLevel - hisLevel)); lastProtection = System.currentTimeMillis(); } if(protection<=0) return damage; remainingProtection=protection-100; if (protection >= 100) { return 0; } return (int)Math.round(CMath.mul(damage,1.0-CMath.div(protection,100.0))); } public void resistAffect(final CMMsg msg, final MOB mob, final Ability me, final String maskString) { if(mob.location()==null) return; if(mob.amDead()) return; if(!msg.amITarget(mob)) return; if((msg.targetMinor()==CMMsg.TYP_DAMAGE) &&((msg.value())>0) &&(msg.tool() instanceof Weapon)) { if(checkProtection("NON-MAGICAL-WEAPONS") && (!CMLib.flags().isABonusItems((Weapon)msg.tool()))) { if((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false))) msg.setValue(weaponProtection("NON-MAGICAL-WEAPONS",msg.value(),mob.phyStats().level(),msg.source().phyStats().level())); } if(checkProtection("NON-SILVER-WEAPONS") && (((Weapon)msg.tool()).material() != RawMaterial.RESOURCE_SILVER)) { if((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false))) msg.setValue(weaponProtection("NON-SILVER-WEAPONS",msg.value(),mob.phyStats().level(),msg.source().phyStats().level())); } if(checkProtection("WEAPONS")) { if((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false))) msg.setValue(weaponProtection("WEAPONS",msg.value(),mob.phyStats().level(),msg.source().phyStats().level())); } else { final Weapon W=(Weapon)msg.tool(); if((W.weaponDamageType()==Weapon.TYPE_BASHING) &&(checkProtection("BLUNT")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) msg.setValue(weaponProtection("BLUNT",msg.value(),mob.phyStats().level(),msg.source().phyStats().level())); if((W.weaponDamageType()==Weapon.TYPE_PIERCING) &&(checkProtection("PIERCE")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) msg.setValue(weaponProtection("PIERCE",msg.value(),mob.phyStats().level(),msg.source().phyStats().level())); if((W.weaponDamageType()==Weapon.TYPE_SLASHING) &&(checkProtection("SLASH")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) msg.setValue(weaponProtection("SLASH",msg.value(),mob.phyStats().level(),msg.source().phyStats().level())); } return; } } @Override public String accountForYourself() { return "The owner gains resistances: " + describeResistance(text()); } @Override public boolean tick(final Tickable ticking, final int tickID) { if(!super.tick(ticking, tickID)) return false; if(hasEffectDuration) { final Physical affected=this.affected; if((affected instanceof MOB) &&(((MOB)affected).numEffects() != lastEffectCount)) { lastEffectCount = (short)((MOB)affected).numEffects(); int maxTicks = getProtection("DEBUF-DURATION"); if(maxTicks <= 0) maxTicks = 1; for(int i=((MOB)affected).numEffects()-1;i>=0;i--) { final Ability A=((MOB)affected).fetchEffect(i); if((A.invoker() != affected) &&(A.abstractQuality() == Ability.QUALITY_MALICIOUS) &&((CMath.s_int(A.getStat("TICKDOWN"))>maxTicks))) A.setStat("TICKDOWN",""+maxTicks); } } } return true; } public boolean isOk(final CMMsg msg, final Ability me, final MOB mob, final String maskString) { if(CMath.bset(msg.targetMajor(),CMMsg.MASK_MAGIC)) { if((msg.tool() instanceof Ability) &&(msg.sourceMinor()!=CMMsg.TYP_TEACH)) { final Ability A=(Ability)msg.tool(); if(CMath.bset(A.flags(),Ability.FLAG_TRANSPORTING)) { if((checkProtection("TELEPORT")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) { msg.source().tell(L("You can't seem to fixate on '@x1'.",mob.name())); return false; } } else if(!CMath.bset(msg.targetMajor(),CMMsg.MASK_MALICIOUS)) return true; else { if((A.classificationCode()&Ability.ALL_ACODES)==Ability.ACODE_PRAYER) { final boolean holySet = CMath.bset(A.flags(),Ability.FLAG_HOLY); final boolean unholySet = CMath.bset(A.flags(),Ability.FLAG_UNHOLY); if(holySet && !unholySet) { if((checkProtection("HOLY")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("Holy energies from <S-NAME> are repelled from <T-NAME>.")); return false; } } else if(unholySet && !holySet) { if((checkProtection("UNHOLY")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("Unholy energies from <S-NAME> are repelled from <T-NAME>.")); return false; } } } if((A.classificationCode()&Ability.ALL_ACODES)==Ability.ACODE_DISEASE) { if((checkProtection("DISEASE")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("A disease from <S-NAME> is repelled from <T-NAME>.")); return false; } } else if((A.classificationCode()&Ability.ALL_ACODES)==Ability.ACODE_POISON) { if((checkProtection("POISON")) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false)))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("The poison from <S-NAME> is repelled from <T-NAME>.")); return false; } } else { final String abilityType = CMLib.flags().getAbilityType_(A); final String abilityDomain = CMLib.flags().getAbilityDomain(A); if((checkProtection(abilityType)) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false))) &&(CMLib.dice().rollPercentage() <= getProtection(abilityType))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("<T-NAME> repell(s) the @x1 effects from <S-NAME>.",CMLib.flags().getAbilityType(A).toLowerCase())); return false; } else if((checkProtection(abilityDomain)) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false))) &&(CMLib.dice().rollPercentage() <= getProtection(abilityDomain))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("<T-NAME> repell(s) the @x1 effects from <S-NAME>.",abilityDomain.toLowerCase().replace('_',' '))); return false; } else if((checkProtection(A.ID().toUpperCase())) &&((maskString.length()==0)||(CMLib.masking().maskCheck(maskString,mob,false))) &&(CMLib.dice().rollPercentage() <= getProtection(A.ID().toUpperCase()))) { mob.location().show(msg.source(),mob,CMMsg.MSG_OK_VISUAL,L("<T-NAME> repell(s) @x1 from <S-NAME>.",abilityDomain.toLowerCase().replace('_',' '))); return false; } } } } } return true; } public String describeResistance(final String text) { final StringBuilder parmString=new StringBuilder(""); for(final String parmKey : this.prots.keySet()) parmString.append(parmKey.toLowerCase()).append(" ").append(this.prots.get(parmKey)).append("% "); String id=parmString.toString().trim()+"."; if(maskString.length()>0) id+=" Restrictions: "+CMLib.masking().maskDesc(maskString)+"."; return id; } public boolean canResist(final Environmental E) { if((affected instanceof Item) &&(E instanceof MOB) &&(!((Item)affected).amDestroyed()) &&(E==((Item)affected).owner())) return true; return false; } @Override public boolean okMessage(final Environmental myHost, final CMMsg msg) { if((canResist(msg.target())) &&(msg.target() instanceof MOB) &&(((MOB)msg.target()).location()!=null)) { if((msg.value()<=0)&&(!isOk(msg,this,(MOB)msg.target(),maskString))) return false; resistAffect(msg,(MOB)msg.target(),this,maskString); } return true; } @Override public String getStat(String statVar) { if(statVar != null) { statVar=statVar.toUpperCase(); if(statVar.equals("STAT-LEVEL")) { int levelChange=0; if(checkProtection("NON-MAGICAL-WEAPONS")) levelChange += 5*getProtection("NON-MAGICAL-WEAPONS"); if(checkProtection("NON-SILVER-WEAPONS")) levelChange += 5*getProtection("NON-SILVER-WEAPONS"); if(checkProtection("WEAPONS")) levelChange += 10*getProtection("WEAPONS"); if(checkProtection("BLUNT")) levelChange += 3*getProtection("BLUNT"); if(checkProtection("PIERCE")) levelChange += 3*getProtection("PIERCE"); if(checkProtection("SLASH")) levelChange += 3*getProtection("SLASH"); if(checkProtection("HOLY")) levelChange += 2*getProtection("HOLY"); if(checkProtection("UNHOLY")) levelChange += 2*getProtection("UNHOLY"); if(checkProtection("DISEASE")) levelChange += 2*getProtection("DISEASE"); if(checkProtection("POISON")) levelChange += 2*getProtection("POISON"); if(checkProtection("TELEPORT")) levelChange += 2*getProtection("TELEPORT"); for(final String acode : Ability.ACODE_DESCS_) if(checkProtection(acode)) levelChange += 5*getProtection(acode); for(final String domain : Ability.DOMAIN_DESCS) if(checkProtection(domain)) levelChange += getProtection(domain); for(final int i : CharStats.CODES.SAVING_THROWS()) { if(adjCharStats.getStat(i)!=0) { switch(i) { case CharStats.STAT_SAVE_GAS: case CharStats.STAT_SAVE_FIRE: case CharStats.STAT_SAVE_ELECTRIC: case CharStats.STAT_SAVE_MIND: case CharStats.STAT_SAVE_COLD: case CharStats.STAT_SAVE_ACID: case CharStats.STAT_SAVE_UNDEAD: case CharStats.STAT_SAVE_JUSTICE: case CharStats.STAT_SAVE_WATER: case CharStats.STAT_SAVE_PARALYSIS: levelChange+= adjCharStats.getStat(i); break; case CharStats.STAT_SAVE_POISON: case CharStats.STAT_SAVE_DISEASE: levelChange+= adjCharStats.getStat(i)*2; break; case CharStats.STAT_SAVE_SPELLS: case CharStats.STAT_SAVE_PRAYERS: case CharStats.STAT_SAVE_SONGS: case CharStats.STAT_SAVE_CHANTS: case CharStats.STAT_SAVE_TRAPS: case CharStats.STAT_SAVE_BLUNT: case CharStats.STAT_SAVE_PIERCE: case CharStats.STAT_SAVE_SLASH: levelChange+= adjCharStats.getStat(i)*5; break; case CharStats.STAT_SAVE_GENERAL: levelChange+= adjCharStats.getStat(i)*50; break; case CharStats.STAT_SAVE_MAGIC: levelChange+= adjCharStats.getStat(i)*10; break; case CharStats.STAT_CRIT_CHANCE_PCT_WEAPON: case CharStats.STAT_CRIT_CHANCE_PCT_MAGIC: case CharStats.STAT_CRIT_DAMAGE_PCT_WEAPON: case CharStats.STAT_CRIT_DAMAGE_PCT_MAGIC: levelChange+= adjCharStats.getStat(i); break; } } } return ""+levelChange; } else if(statVar.startsWith("TIDBITS")) { String parmText = text().toUpperCase(); if(statVar.startsWith("TIDBITS=")) parmText = statVar.substring(8).toUpperCase().trim(); final StringBuilder str=new StringBuilder(""); final List<String> parmParts = CMParms.parseSpaces(parmText, true); final List<String> previousSet = new LinkedList<String>(); for(String parts : parmParts) { Integer newPct = null; if(parts.endsWith("%")) { parts=parts.substring(0,parts.length()-1).trim(); } if(CMath.isInteger(parts)) { newPct = Integer.valueOf(CMath.s_int(parts)); } else if(CMath.isNumber(parts)) { final double d = CMath.s_double(parts); if((d > 1.0) || (d < -1.0)) { newPct = Integer.valueOf((int)Math.round(d)); } else { newPct = Integer.valueOf((int)Math.round(d * 100.0)); } } else if(parts.equals("ALWAYS")) { //this.alwaysWeapProt=true; } else { previousSet.add(parts); } if(newPct != null) { for(final String previousKey : previousSet) { if(newPct.intValue() < 0) { str.append(L("@x1% vulnerable to @x2\n\r",newPct.toString().substring(1),previousKey.toLowerCase())); } else str.append(L("@x1% resistant to @x2\n\r",newPct.toString(),previousKey.toLowerCase())); } previousSet.clear(); } } return str.toString(); } } return ""; } }