/
com/planet_ink/coffee_mud/Abilities/Common/
com/planet_ink/coffee_mud/Abilities/Diseases/
com/planet_ink/coffee_mud/Abilities/Druid/
com/planet_ink/coffee_mud/Abilities/Fighter/
com/planet_ink/coffee_mud/Abilities/Languages/
com/planet_ink/coffee_mud/Abilities/Misc/
com/planet_ink/coffee_mud/Abilities/Prayers/
com/planet_ink/coffee_mud/Abilities/Properties/
com/planet_ink/coffee_mud/Abilities/Skills/
com/planet_ink/coffee_mud/Abilities/Songs/
com/planet_ink/coffee_mud/Abilities/Specializations/
com/planet_ink/coffee_mud/Abilities/Spells/
com/planet_ink/coffee_mud/Abilities/Thief/
com/planet_ink/coffee_mud/Abilities/Traps/
com/planet_ink/coffee_mud/Behaviors/
com/planet_ink/coffee_mud/CharClasses/
com/planet_ink/coffee_mud/CharClasses/interfaces/
com/planet_ink/coffee_mud/Commands/
com/planet_ink/coffee_mud/Commands/interfaces/
com/planet_ink/coffee_mud/Common/
com/planet_ink/coffee_mud/Common/interfaces/
com/planet_ink/coffee_mud/Exits/interfaces/
com/planet_ink/coffee_mud/Items/Armor/
com/planet_ink/coffee_mud/Items/Basic/
com/planet_ink/coffee_mud/Items/BasicTech/
com/planet_ink/coffee_mud/Items/CompTech/
com/planet_ink/coffee_mud/Items/MiscMagic/
com/planet_ink/coffee_mud/Items/Weapons/
com/planet_ink/coffee_mud/Items/interfaces/
com/planet_ink/coffee_mud/Libraries/
com/planet_ink/coffee_mud/Libraries/interfaces/
com/planet_ink/coffee_mud/Locales/
com/planet_ink/coffee_mud/MOBS/
com/planet_ink/coffee_mud/Races/
com/planet_ink/coffee_mud/Races/interfaces/
com/planet_ink/coffee_mud/WebMacros/
com/planet_ink/coffee_mud/WebMacros/interfaces/
com/planet_ink/coffee_mud/core/
com/planet_ink/coffee_mud/core/collections/
com/planet_ink/coffee_mud/core/interfaces/
com/planet_ink/coffee_mud/core/intermud/
com/planet_ink/coffee_mud/core/intermud/i3/
com/planet_ink/coffee_web/server/
com/planet_ink/siplet/applet/
lib/
resources/factions/
resources/fakedb/
resources/progs/autoplayer/
resources/quests/holidays/
web/
web/admin.templates/
web/admin/grinder/
web/admin/images/
web/clan.templates/
web/pub.templates/
web/pub/images/mxp/
web/pub/sounds/
web/pub/textedit/
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 "";
	}
}