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 2002-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 Burning extends StdAbility
{
@Override
public String ID()
{
return "Burning";
}
private final static String localizedName = CMLib.lang().L("Burning");
@Override
public String name()
{
return localizedName;
}
private final static String localizedStaticDisplay = CMLib.lang().L("(Burning)");
@Override
public String displayText()
{
return localizedStaticDisplay;
}
@Override
protected int canAffectCode()
{
return Ability.CAN_ITEMS;
}
@Override
protected int canTargetCode()
{
return 0;
}
@Override
public long flags()
{
return Ability.FLAG_HEATING | Ability.FLAG_FIREBASED;
}
protected static final int FIREFLAG_WEATHERMASK = 255;
protected static final int FIREFLAG_PERSISTFLAGS = 256;
protected static final int FIREFLAG_DESTROYHOST = 512;
protected static final int FIREFLAG_NEVERDESTROYHOST = 1024;
protected static final int FIREFLAG_UNEXTINGUISHABLE = 2048;
protected static final int FIREFLAG_NOSPREAD = 4096;
protected int abilityCode = 0;
@Override
public int abilityCode()
{
return abilityCode;
}
@Override
public void setAbilityCode(final int newCode)
{
this.abilityCode = newCode;
}
@Override
public void setMiscText(final String newMiscText)
{
super.setMiscText(newMiscText);
final List<String> flags=CMParms.parse(newMiscText.toUpperCase().trim());
if(flags.contains("PERMANENT")||flags.contains("UNEXTINGUISHABLE"))
this.setAbilityCode(this.abilityCode|Burning.FIREFLAG_UNEXTINGUISHABLE);
if(flags.contains("PERSISTFLAGS"))
this.setAbilityCode(this.abilityCode|Burning.FIREFLAG_PERSISTFLAGS);
if(flags.contains("DESTROY")||flags.contains("DESTROYHOST"))
this.setAbilityCode(this.abilityCode|Burning.FIREFLAG_DESTROYHOST);
if(flags.contains("NODESTROY")||flags.contains("NODESTROYHOST"))
this.setAbilityCode(this.abilityCode|Burning.FIREFLAG_NEVERDESTROYHOST);
if(flags.contains("NOSPREAD"))
this.setAbilityCode(this.abilityCode|Burning.FIREFLAG_NOSPREAD);
}
@Override
public void unInvoke()
{
final Physical affected=this.affected;
super.unInvoke();
if(canBeUninvoked() && (affected != null) && (!affected.amDestroyed()))
{
if(isFlagSet(FIREFLAG_DESTROYHOST))
affected.destroy();
}
}
public boolean isFlagSet(final int flag)
{
if(abilityCode()<0)
return CMath.bset(-abilityCode(), flag);
else
return CMath.bset(abilityCode(), flag);
}
@Override
public boolean tick(final Tickable ticking, final int tickID)
{
Physical affected=this.affected;
if((affected instanceof Item)
&&(((Item)affected).owner() instanceof Room)
&&(!CMath.bset(abilityCode(), Burning.FIREFLAG_UNEXTINGUISHABLE)))
{
int unInvokeChance;
if(abilityCode() < 0)
unInvokeChance=((-abilityCode()) & FIREFLAG_WEATHERMASK);
else
unInvokeChance=-(abilityCode() & FIREFLAG_WEATHERMASK);
String what=null;
switch(((Room)(((Item)affected).owner())).getArea().getClimateObj().weatherType(((Room)(((Item)affected).owner()))))
{
case Climate.WEATHER_RAIN:
what="rain";
unInvokeChance+=10;
break;
case Climate.WEATHER_THUNDERSTORM:
what="pounding rain";
unInvokeChance+=15;
break;
case Climate.WEATHER_SLEET:
what="sleet";
unInvokeChance+=5;
break;
case Climate.WEATHER_BLIZZARD:
what="swirling snow";
unInvokeChance+=10;
break;
case Climate.WEATHER_SNOW:
what="snow";
unInvokeChance+=10;
break;
}
if(CMLib.dice().rollPercentage()<unInvokeChance)
{
final Room R=((Room)(((Item)affected).owner()));
if(R.numInhabitants()>0)
R.showHappens(CMMsg.MSG_OK_ACTION,L("The @x1 puts out @x2.",what,affected.name()));
unInvoke();
return false;
}
}
if((tickDown<2)&&(affected!=null))
{
if(affected instanceof Item)
{
final Environmental E=((Item)affected).owner();
if(E==null)
{
if(!isFlagSet(FIREFLAG_NEVERDESTROYHOST))
((Item)affected).destroy();
}
else
if(E instanceof Room)
{
final Room room=(Room)E;
if((affected instanceof RawMaterial)
&&(!CMath.bset(abilityCode(), Burning.FIREFLAG_NOSPREAD))
&&(room.isContent((Item)affected)))
{
for(int i=0;i<room.numItems();i++)
{
final Item I=room.getItem(i);
if(I.name().equals(affected.name())
&&(I!=affected)
&&(I instanceof RawMaterial)
&&(I.material()==((Item)affected).material()))
{
int durationOfBurn=CMLib.materials().getBurnDuration(I);
if(durationOfBurn<=0)
durationOfBurn=5;
final Burning B=new Burning();
if(isFlagSet(FIREFLAG_PERSISTFLAGS))
B.setAbilityCode(abilityCode());
B.invoke(invoker,I,true,durationOfBurn);
break;
}
}
}
if(!(affected instanceof ClanItem))
{
switch(((Item)affected).material()&RawMaterial.MATERIAL_MASK)
{
case RawMaterial.MATERIAL_LIQUID:
case RawMaterial.MATERIAL_METAL:
case RawMaterial.MATERIAL_MITHRIL:
case RawMaterial.MATERIAL_ENERGY:
case RawMaterial.MATERIAL_GAS:
case RawMaterial.MATERIAL_PRECIOUS:
case RawMaterial.MATERIAL_ROCK:
case RawMaterial.MATERIAL_UNKNOWN:
break;
default:
{
if(CMLib.flags().isABonusItems(affected))
{
if(invoker==null)
{
invoker=CMClass.getMOB("StdMOB");
invoker.setLocation(CMClass.getLocale("StdRoom"));
invoker.basePhyStats().setLevel(affected.phyStats().level());
invoker.phyStats().setLevel(affected.phyStats().level());
}
room.showHappens(CMMsg.MSG_OK_ACTION,L("@x1 EXPLODES!!!",affected.name()));
for(int i=0;i<room.numInhabitants();i++)
{
final MOB target=room.fetchInhabitant(i);
CMLib.combat().postDamage(invoker(),target,null,CMLib.dice().roll(affected.phyStats().level(),5,1),CMMsg.MASK_ALWAYS|CMMsg.TYP_FIRE,Weapon.TYPE_BURNING,L("The blast <DAMAGE> <T-NAME>!"));
}
if(!isFlagSet(FIREFLAG_NEVERDESTROYHOST))
((Item)affected).destroy();
}
else
{
final Item ash=CMClass.getItem("GenResource");
ash.setName(L("some ash"));
ash.setDisplayText(L("a small pile of ash is here"));
ash.setMaterial(RawMaterial.RESOURCE_ASH);
ash.basePhyStats().setWeight(1);
ash.recoverPhyStats();
room.addItem(ash,ItemPossessor.Expire.Monster_EQ);
((RawMaterial)ash).rebundle();
if((affected instanceof RawMaterial)
&&(affected.basePhyStats().weight()>1)
&&(CMLib.materials().getBurnDuration(affected)>0))
{
affected.basePhyStats().setWeight(affected.basePhyStats().weight()-1);
affected.recoverPhyStats();
this.tickDown = CMLib.materials().getBurnDuration(affected);
CMLib.materials().adjustResourceName((Item)affected);
room.recoverRoomStats();
return super.tick(ticking,tickID);
}
room.showHappens(CMMsg.MSG_OK_VISUAL, L("@x1 is no longer burning.",affected.name()));
if(!isFlagSet(FIREFLAG_NEVERDESTROYHOST))
((Item)affected).destroy();
}
break;
}
}
}
((Room)E).recoverRoomStats();
}
else
if(E instanceof MOB)
{
switch(((Item)affected).material()&RawMaterial.MATERIAL_MASK)
{
case RawMaterial.MATERIAL_LIQUID:
case RawMaterial.MATERIAL_METAL:
case RawMaterial.MATERIAL_ENERGY:
case RawMaterial.MATERIAL_MITHRIL:
case RawMaterial.MATERIAL_PRECIOUS:
case RawMaterial.MATERIAL_ROCK:
case RawMaterial.MATERIAL_UNKNOWN:
break;
default:
if(!isFlagSet(FIREFLAG_NEVERDESTROYHOST))
((Item)affected).destroy();
break;
}
((MOB)E).location().recoverRoomStats();
}
return false;
}
}
if(!super.tick(ticking,tickID))
return false;
if(tickID!=Tickable.TICKID_MOB)
return true;
affected=this.affected;
if(affected==null)
return false;
if(affected instanceof Item)
{
final Item I=(Item)affected;
final Environmental owner=I.owner();
if((owner instanceof MOB)&&(((MOB)owner).location()!=null))
{
if(!ouch((MOB)owner))
CMLib.commands().postDrop((MOB)owner,I,false,false,false);
}
if((I.subjectToWearAndTear())
&&(I.usesRemaining()>1))
{
if(owner instanceof MOB)
CMLib.combat().postItemDamage((MOB)owner, I, null, 1, CMMsg.TYP_FIRE, null);
else
if(owner instanceof Room)
{
final MOB M=CMLib.map().getFactoryMOB((Room)owner);
CMLib.combat().postItemDamage(M, I, null, 1, CMMsg.TYP_FIRE, null);
M.destroy();
}
}
}
// might want to add the ability for it to spread
return true;
}
public boolean ouch(final MOB mob)
{
if(CMLib.dice().rollPercentage()>(mob.charStats().getSave(CharStats.STAT_SAVE_FIRE)-50))
{
if(affected instanceof Item)
switch(((Item)affected).material()&RawMaterial.MATERIAL_MASK)
{
case RawMaterial.MATERIAL_LIQUID:
case RawMaterial.MATERIAL_METAL:
case RawMaterial.MATERIAL_MITHRIL:
case RawMaterial.MATERIAL_PRECIOUS:
case RawMaterial.MATERIAL_ENERGY:
case RawMaterial.MATERIAL_GAS:
case RawMaterial.MATERIAL_ROCK:
case RawMaterial.MATERIAL_UNKNOWN:
mob.tell(L("Ouch!! @x1 is HOT!",CMStrings.capitalizeAndLower(affected.name())));
break;
default:
mob.tell(L("Ouch!! @x1 is on fire!",CMStrings.capitalizeAndLower(affected.name())));
break;
}
CMLib.combat().postDamage(invoker,mob,this,CMLib.dice().roll(1,5,5),CMMsg.MASK_ALWAYS|CMMsg.TYP_FIRE,Weapon.TYPE_BURNING,null);
return false;
}
return true;
}
@Override
public boolean okMessage(final Environmental myHost, final CMMsg msg)
{
if(!super.okMessage(myHost,msg))
return false;
if((affected instanceof Item)
&&(msg.amITarget(affected))
&&((msg.targetMinor()==CMMsg.TYP_GET)
||(msg.targetMinor()==CMMsg.TYP_PUSH)
||(msg.targetMinor()==CMMsg.TYP_PULL)))
{
if((msg.tool()==null)||(!(msg.tool() instanceof Item)))
return ouch(msg.source());
// the "oven" exception
final Item container=(Item)affected;
final Item target=(Item)msg.tool();
if((target.owner()==container.owner())
&&(target.container()==container))
{
switch(container.material()&RawMaterial.MATERIAL_MASK)
{
case RawMaterial.MATERIAL_METAL:
case RawMaterial.MATERIAL_MITHRIL:
case RawMaterial.MATERIAL_PRECIOUS:
case RawMaterial.MATERIAL_ENERGY:
case RawMaterial.MATERIAL_GAS:
case RawMaterial.MATERIAL_ROCK:
case RawMaterial.MATERIAL_UNKNOWN:
return true;
default:
break;
}
}
return ouch(msg.source());
}
return true;
}
@Override
public void executeMsg(final Environmental myHost, final CMMsg msg)
{
if((affected instanceof Item)
&&(msg.tool()==affected)
&&(msg.target() instanceof Container)
&&(msg.targetMinor()==CMMsg.TYP_PUT)
&&(!CMath.bset(abilityCode(), Burning.FIREFLAG_UNEXTINGUISHABLE)))
{
final Item I=(Item)affected;
final Item C=(Container)msg.target();
if((C instanceof Drink)
&&(((Drink)C).containsDrink()))
{
msg.addTrailerMsg(CMClass.getMsg(invoker,null,CMMsg.MSG_OK_VISUAL,L("@x1 is extinguished.",I.name())));
I.delEffect(this);
}
}
super.executeMsg(myHost,msg);
}
@Override
public void affectPhyStats(final Physical affected, final PhyStats affectableStats)
{
super.affectPhyStats(affected,affectableStats);
affectableStats.setDisposition(affectableStats.disposition()|PhyStats.IS_LIGHTSOURCE);
}
@Override
public boolean invoke(final MOB mob, final List<String> commands, Physical target, final boolean auto, int asLevel)
{
if(!auto)
return false;
if(target==null)
return false;
if(target instanceof MOB)
{
final MOB targM=(MOB)target;
for(int i=0;i<10;i++)
{
final Item I=targM.getRandomItem();
if((I.container()==null)&&(CMLib.materials().getBurnDuration(I)>0))
{
target=I;
break;
}
}
// mobs don't burn, only their stuff
if(target instanceof MOB)
{
final int dmg=CMLib.dice().roll(1, targM.baseState().getHitPoints() / 10, 0);
CMLib.combat().postDamage(mob, targM, this, dmg, CMMsg.MASK_ALWAYS | CMMsg.MASK_MALICIOUS | CMMsg.TYP_FIRE, Weapon.TYPE_BURNING, L("A fire <DAMAGE> <T-NAME>!"));
return true;
}
}
if(target.fetchEffect("Burning")==null)
{
if(((target instanceof Item)&&(((Item)target).material()==RawMaterial.RESOURCE_NOTHING))
||(target instanceof ClanItem))
return false;
if(mob!=null)
{
final Room room=mob.location();
if(room!=null)
{
final CMMsg msg=CMClass.getMsg(mob,target,CMMsg.MASK_ALWAYS|CMMsg.TYP_FIRE,null);
if(room.okMessage(mob,msg))
room.send(mob,msg);
}
}
if(asLevel == 0)
asLevel = CMLib.materials().getBurnDuration(target);
if(asLevel < 0)
asLevel = 0;
beneficialAffect(mob,target,0,asLevel);
target.recoverPhyStats();
if(target instanceof Item)
{
final ItemPossessor owner=((Item)target).owner();
if(owner!=null)
{
owner.recoverPhyStats();
if(owner instanceof Room)
((Room)owner).recoverRoomStats();
else
if(owner instanceof MOB)
{
final MOB ownerM=(MOB)owner;
final Room ownerRoom=ownerM.location();
if(ownerRoom != null)
ownerRoom.recoverRoomStats();
}
}
}
}
return true;
}
}