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 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 Prop_Doppleganger extends Property
{
@Override
public String ID()
{
return "Prop_Doppleganger";
}
@Override
public String name()
{
return "Doppleganger";
}
@Override
protected int canAffectCode()
{
return Ability.CAN_MOBS|Ability.CAN_ITEMS;
}
//protected boolean lastLevelChangers=true;
protected Physical lastOwner = null;
private int maxLevel = Integer.MAX_VALUE;
private int minLevel = Integer.MIN_VALUE;
protected int lastLevel = Integer.MIN_VALUE;
protected int levelAdd = 0;
protected double levelPct = 1.0;
protected int diffAdd = 0;
protected double diffPct = 1.0;
protected boolean matchPlayersOnly = false;
protected boolean matchPlayersFollowersOnly = false;
protected int asMaterial = -1;
@Override
public long flags()
{
return Ability.FLAG_ADJUSTER;
}
@Override
public String accountForYourself()
{
return "Level Changer";
}
@Override
public void setMiscText(final String text)
{
super.setMiscText(text);
levelAdd=0;
diffAdd=0;
levelPct=1.0;
diffPct=1.0;
asMaterial=-1;
maxLevel=Integer.MAX_VALUE;
minLevel=Integer.MIN_VALUE;
if(CMath.isInteger(text))
levelAdd=CMath.s_int(text);
else
if(CMath.isPct(text))
levelPct=CMath.s_pct(text);
else
{
maxLevel=CMParms.getParmInt(text,"MAX",Integer.MAX_VALUE);
minLevel=CMParms.getParmInt(text,"MIN",Integer.MIN_VALUE);
levelAdd=CMParms.getParmInt(text, "LEVELADD", 0);
levelPct=CMParms.getParmInt(text, "LEVELPCT", 100)/100.0;
diffAdd=CMParms.getParmInt(text, "DIFFLADD", 0);
diffPct=CMParms.getParmInt(text, "DIFFPCT", 100)/100.0;
matchPlayersFollowersOnly=CMParms.getParmBool(text, "PLAYERSNFOLS", false);
matchPlayersOnly=CMParms.getParmBool(text, "PLAYERSONLY", false);
final String asMat = CMParms.getParmStr(text, "ASMATERIAL", "");
if((asMat!=null)&&(asMat.trim().length()>0))
asMaterial = RawMaterial.CODES.FIND_IgnoreCase(asMat);
}
}
protected void doppleGangItem(final Item I)
{
final ItemPossessor owner=I.owner();
if(owner!=null)
{
lastOwner=owner;
lastLevel=owner.phyStats().level();
int level=(int)Math.round(CMath.mul(((MOB)owner).phyStats().level(),levelPct))+levelAdd;
if(level<minLevel)
level=minLevel;
if(level>maxLevel)
level=maxLevel;
int difflevel=(int)Math.round(CMath.mul(level,diffPct))+diffAdd;
I.basePhyStats().setLevel(difflevel);
I.phyStats().setLevel(difflevel);
final int oldMaterial=I.material();
if(asMaterial != -1)
I.setMaterial(asMaterial);
CMLib.itemBuilder().balanceItemByLevel(I);
I.basePhyStats().setLevel(level);
I.phyStats().setLevel(level);
I.setMaterial(oldMaterial);
level=((MOB)owner).phyStats().level();
if(level<minLevel)
level=minLevel;
if(level>maxLevel)
level=maxLevel;
I.basePhyStats().setLevel(level);
I.phyStats().setLevel(level);
owner.recoverPhyStats();
if(owner instanceof Room)
((Room)owner).recoverRoomStats();
else
if(owner instanceof MOB)
{
final MOB M=(MOB)owner;
M.recoverCharStats();
M.recoverMaxState();
M.recoverPhyStats();
}
}
}
@Override
public void executeMsg(final Environmental myHost, final CMMsg msg)
{
if((affected instanceof Item)
&&((((Item)affected).owner()!=lastOwner)||((lastOwner!=null)&&(lastOwner.phyStats().level()!=lastLevel)))
&&(((Item)affected).owner() instanceof MOB))
doppleGangItem((Item)affected);
super.executeMsg(myHost,msg);
}
protected void doppleGangMob(final MOB entrantM, final MOB mob, final Room R)
{
if((R!=null)
&&(CMLib.flags().isAliveAwakeMobile(mob,true))
&&(mob.curState().getHitPoints()>=mob.maxState().getHitPoints()))
{
int total=0;
int num=0;
final MOB victim=mob.getVictim();
if(canMimic(victim,R))
{
total+=victim.phyStats().level();
num++;
}
if(canMimic(entrantM,R))
{
total+=entrantM.phyStats().level();
num++;
}
for(int i=0;i<R.numInhabitants();i++)
{
final MOB M=R.fetchInhabitant(i);
if((M!=null)
&&(M!=mob)
&&((M.getVictim()==mob)||(victim==null))
&&((M!=victim)&&(M!=entrantM))
&&(canMimic(M,R)))
{
total+=M.phyStats().level();
num++;
}
}
if(num>0)
{
int level=(int)Math.round(CMath.mul(CMath.div(total,num),levelPct))+levelAdd;
if(level<minLevel)
level=minLevel;
if(level>maxLevel)
level=maxLevel;
int difflevel=(int)Math.round(CMath.mul(level,diffPct))+diffAdd;
if(level!=mob.basePhyStats().level())
{
mob.basePhyStats().setLevel(difflevel);
mob.phyStats().setLevel(difflevel);
mob.basePhyStats().setArmor(CMLib.leveler().getLevelMOBArmor(mob));
mob.basePhyStats().setAttackAdjustment(CMLib.leveler().getLevelAttack(mob));
mob.basePhyStats().setDamage(CMLib.leveler().getLevelMOBDamage(mob));
mob.basePhyStats().setSpeed(1.0+(CMath.div(level,100)*4.0));
mob.baseState().setHitPoints(CMLib.leveler().getPlayerHitPoints(mob));
mob.baseState().setMana(CMLib.leveler().getLevelMana(mob));
mob.baseState().setMovement(CMLib.leveler().getLevelMove(mob));
mob.basePhyStats().setLevel(level);
mob.phyStats().setLevel(level);
mob.recoverPhyStats();
mob.recoverCharStats();
mob.recoverMaxState();
mob.resetToMaxState();
}
}
}
}
public boolean canMimic(final MOB mob, final Room R)
{
if((mob==affected)
||(mob==null))
return false;
if((affected instanceof Room)
&&(mob.isMonster())
&&(mob.getStartRoom()!=null)
&&(mob.getStartRoom().getArea()==((Room)affected).getArea()))
return false;
if((affected instanceof Area)
&&(mob.isMonster())
&&(mob.getStartRoom()!=null)
&&(mob.getStartRoom().getArea()==(Area)affected))
return false;
if(mob.fetchEffect(ID())!=null)
return false;
if(mob.isMonster())
{
if(matchPlayersFollowersOnly)
{
final MOB folM=mob.amUltimatelyFollowing();
return (folM!=null)&& (!folM.isMonster());
}
return (!matchPlayersOnly);
}
if((!CMSecurity.isAllowed(mob,R,CMSecurity.SecFlag.CMDMOBS))
&&(!CMSecurity.isAllowed(mob,R,CMSecurity.SecFlag.CMDROOMS))
&&(!CMLib.flags().isUnattackable(mob)))
return true;
return false;
}
@Override
public boolean okMessage(final Environmental myHost, final CMMsg msg)
{
if((((msg.target() instanceof Room)&&(msg.sourceMinor()==CMMsg.TYP_ENTER))
||(msg.sourceMinor()==CMMsg.TYP_LIFE))
&&(!(affected instanceof Item)))
//&&(lastLevelChangers))
{
//lastLevelChangers=false;
if(affected instanceof MOB)
{
final Room R=(msg.target() instanceof Room)?((Room)msg.target()):msg.source().location();
this.doppleGangMob(msg.source(), (MOB)affected, R);
}
else
if(affected instanceof Room)
{
final Room R=(Room)affected;
for(final Enumeration<MOB> m=R.inhabitants();m.hasMoreElements();)
{
final MOB M=m.nextElement();
if((M.isMonster())
&&(M.getStartRoom()!=null)
&&(M.amFollowing()==null)
&&(M.getStartRoom().getArea()==R.getArea()))
this.doppleGangMob(msg.source(), M, R);
}
}
else
if(affected instanceof Area)
{
final Room R=(msg.target() instanceof Room)?((Room)msg.target()):msg.source().location();
for(final Enumeration<MOB> m=R.inhabitants();m.hasMoreElements();)
{
final MOB M=m.nextElement();
if((M.isMonster())
&&(M.getStartRoom()!=null)
&&(M.amFollowing()==null)
&&(M.getStartRoom().getArea()==affected))
this.doppleGangMob(msg.source(), M, R);
}
}
}
return super.okMessage(myHost,msg);
}
}