package com.planet_ink.coffee_mud.Behaviors;
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.Basic.StdItem;
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 WaterCurrents extends ActiveTicker
{
@Override
public String ID()
{
return "WaterCurrents";
}
@Override
protected int canImproveCode()
{
return Behavior.CAN_ROOMS | Behavior.CAN_AREAS;
}
protected String dirs = "";
protected boolean doBoats = false;
public WaterCurrents()
{
super();
minTicks=3;maxTicks=5;chance=75;
tickReset();
}
@Override
public String accountForYourself()
{
return "water current moving";
}
@Override
public void setParms(final String newParms)
{
super.setParms(newParms);
final Vector<String> V=CMParms.parse(newParms);
dirs="";
for(int v=0;v<V.size();v++)
{
final String str=V.get(v);
if(str.equalsIgnoreCase("BOATS"))
this.doBoats=true;
else
{
final int dir=CMLib.directions().getGoodDirectionCode(str);
if(dir>=0)
dirs=dirs+CMLib.directions().getDirectionChar(dir);
}
}
if(dirs.length()==0)
dirs="NE";
}
public void applyCurrents(final Room R, final Vector<Physical> done)
{
final Vector<Physical> todo=new Vector<Physical>();
if((R!=null)&&(R.numInhabitants()>0))
{
MOB M=null;
for(int m=0;m<R.numInhabitants();m++)
{
M=R.fetchInhabitant(m);
if((M!=null)
&&(!M.isMonster())
&&(M.riding()==null)
&&(!CMLib.flags().isInFlight(M))
&&((!(M instanceof Rideable))||(((Rideable)M).numRiders()==0))
&&(!M.isInCombat())
&&(!CMLib.flags().isMobile(M))
&&(!done.contains(M)))
{
todo.addElement(M);
done.addElement(M);
}
}
}
if((R!=null)&&(R.numItems()>0))
{
Item I=null;
for(int i=0;i<R.numItems();i++)
{
I=R.getItem(i);
if((I!=null)
&&(I.container()==null)
&&((!(I instanceof Rideable))
||(((Rideable)I).rideBasis()!=Rideable.RIDEABLE_WATER)
||(((Rideable)I).numRiders()==0)
||(doBoats))
&&(!CMLib.flags().isInFlight(I))
&&(!CMLib.flags().isMobile(I))
&&((CMLib.flags().isGettable(I))||(I instanceof BoardableShip))
&&((!(I instanceof Exit))||(doBoats))
&&(!done.contains(I)))
{
todo.addElement(I);
done.addElement(I);
}
}
}
if((todo.size()>0)&&(R!=null))
{
int dir=-1;
Room R2=null;
for(int dl=0;dl<dirs.length();dl++)
{
dir=CMLib.directions().getDirectionCode(""+dirs.charAt(dl));
if(dir>=0)
{
R2=R.getRoomInDir(dir);
if(R2!=null)
{
if((R.getExitInDir(dir)!=null)
&&(R.getExitInDir(dir).isOpen())
&&((R2.domainType()==R.domainType())
||(CMLib.flags().isWateryRoom(R2))))
break;
R2=null;
}
}
}
if(R2!=null)
{
MOB M=null;
Item I=null;
final MOB srcM=CMClass.getFactoryMOB("the water", 1, R2);
for(int m=0;m<todo.size();m++)
{
if(todo.elementAt(m) instanceof MOB)
{
M=(MOB)todo.elementAt(m);
final CMMsg themsg=CMClass.getMsg(srcM,M,new AWaterCurrent(),CMMsg.MSG_OK_ACTION,L("<T-NAME> <T-IS-ARE> swept @x1 by the current.",
CMLib.directions().getDirectionName(dir).toLowerCase()));
if(R.okMessage(M,themsg))
{
R.send(M,themsg);
R2.bringMobHere(M,true);
R2.showOthers(srcM,M,new AWaterCurrent(),CMMsg.MSG_OK_ACTION,L("<T-NAME> <T-IS-ARE> swept in from @x1 by the current.",
CMLib.directions().getFromCompassDirectionName(R.getReverseDir(dir)).toLowerCase()));
CMLib.commands().postLook(M,true);
}
}
else
if(todo.elementAt(m) instanceof Item)
{
I=(Item)todo.elementAt(m);
if(R.show(srcM,I,new AWaterCurrent(),CMMsg.MSG_OK_ACTION,L("@x1 is swept @x2 by the current.",
I.name(),CMLib.directions().getDirectionName(dir).toLowerCase())))
{
if(I instanceof BoardableShip)
{
for(final Enumeration<Room> r = ((BoardableShip)I).getShipArea().getProperMap();r.hasMoreElements();)
{
final Room R3=r.nextElement();
if((R3!=null)&&((R3.domainType()&Room.INDOORS)==0))
R3.showHappens(CMMsg.MSG_OK_ACTION, L("@x1 is swept @x2 by the current.",I.name(),CMLib.directions().getDirectionName(dir).toLowerCase()));
}
}
R2.moveItemTo(I,ItemPossessor.Expire.Player_Drop);
R2.showOthers(srcM,I,new AWaterCurrent(),CMMsg.MSG_OK_ACTION,L("@x1 is swept in from @x2 by the current.",
I.name(),CMLib.directions().getFromCompassDirectionName(R.getReverseDir(dir)).toLowerCase()));
}
}
}
srcM.destroy();
}
}
}
@Override
public boolean tick(final Tickable ticking, final int tickID)
{
super.tick(ticking,tickID);
if(canAct(ticking,tickID))
{
final Vector<Physical> sweeps=new Vector<Physical>();
if(ticking instanceof Room)
{
final Room R=(Room)ticking;
applyCurrents(R,sweeps);
final Room below=R.rawDoors()[Directions.DOWN];
if((below!=null)
&&(below.roomID().length()==0)
&&(below instanceof GridLocale)
&&((below.domainType()==Room.DOMAIN_INDOORS_UNDERWATER)
||(below.domainType()==Room.DOMAIN_OUTDOORS_UNDERWATER)))
{
for (final Room R2 : ((GridLocale)below).getAllRooms())
{
applyCurrents(R2,sweeps);
}
}
}
else
if(ticking instanceof Area)
{
for(final Enumeration<Room> r=((Area)ticking).getMetroMap();r.hasMoreElements();)
{
final Room R=r.nextElement();
if(CMLib.flags().isWateryRoom(R))
applyCurrents(R,sweeps);
}
}
}
return true;
}
protected static final String[] empty={};
protected static final String[] CODES={"CLASS","TEXT"};
protected static class AWaterCurrent implements Ability, Cloneable
{
public AWaterCurrent()
{
super();
// CMClass.bumpCounter(this,CMClass.CMObjectType.ABILITY);//removed
// for mem & perf
}
@Override
public String ID()
{
return "AWaterCurrent";
}
@Override
public String name()
{
return "a water current";
}
@Override
public String Name()
{
return name();
}
@Override
public String description()
{
return "";
}
@Override
public String displayText()
{
return "";
}
protected boolean savable = true;
protected String miscText = "";
protected Physical affected = null;
protected int canAffectCode()
{
return 0;
}
protected int canTargetCode()
{
return 0;
}
@Override
public boolean canTarget(final int can_code)
{
return false;
}
@Override
public boolean canAffect(final int can_code)
{
return false;
}
@Override
public double castingTime(final MOB mob, final List<String> cmds)
{
return 0.0;
}
@Override
public double combatCastingTime(final MOB mob, final List<String> cmds)
{
return 0.0;
}
@Override
public double checkedCastingCost(final MOB mob, final List<String> cmds)
{
return 0.0;
}
@Override
public void initializeClass()
{
}
@Override
public int abilityCode()
{
return 0;
}
@Override
public void setAbilityCode(final int newCode)
{
}
@Override
public int adjustedLevel(final MOB mob, final int asLevel)
{
return -1;
}
@Override
public boolean bubbleAffect()
{
return false;
}
@Override
public ExpertiseLibrary.SkillCost getTrainingCost(final MOB mob)
{
return CMLib.expertises().createNewSkillCost(ExpertiseLibrary.CostType.TRAIN, Double.valueOf(1.0));
}
@Override
public String L(final String str, final String... xs)
{
return CMLib.lang().fullSessionTranslation(str, xs);
}
@Override
public long flags()
{
return Ability.FLAG_TRANSPORTING;
}
@Override
public int getTickStatus()
{
return Tickable.STATUS_NOT;
}
@Override
public int usageType()
{
return 0;
}
/*
protected void finalize()
{
CMClass.unbumpCounter(this, CMClass.CMObjectType.ABILITY);
}// removed for mem & perf
*/
@Override
public long expirationDate()
{
return 0;
}
@Override
public void setExpirationDate(final long time)
{
}
@Override
public void setName(final String newName)
{
}
@Override
public void setDescription(final String newDescription)
{
}
@Override
public void setDisplayText(final String newDisplayText)
{
}
@Override
public String image()
{
return "";
}
@Override
public String rawImage()
{
return "";
}
@Override
public void setImage(final String newImage)
{
}
@Override
public MOB invoker()
{
return null;
}
@Override
public void setInvoker(final MOB mob)
{
}
@Override
public String[] triggerStrings()
{
return empty;
}
@Override
public boolean preInvoke(final MOB mob, final List<String> commands, final Physical givenTarget, final boolean auto, final int asLevel, final int secondsElapsed, final double actionsRemaining)
{
return true;
}
@Override
public boolean invoke(final MOB mob, final List<String> commands, final Physical target, final boolean auto, final int asLevel)
{
return false;
}
@Override
public boolean invoke(final MOB mob, final Physical target, final boolean auto, final int asLevel)
{
return false;
}
@Override
public boolean autoInvocation(final MOB mob, final boolean force)
{
return false;
}
@Override
public void unInvoke()
{
}
@Override
public boolean canBeUninvoked()
{
return false;
}
@Override
public boolean isAutoInvoked()
{
return true;
}
@Override
public boolean isNowAnAutoEffect()
{
return true;
}
@Override
public List<String> externalFiles()
{
return null;
}
@Override
public boolean canBeTaughtBy(final MOB teacher, final MOB student)
{
return false;
}
@Override
public boolean canBePracticedBy(final MOB teacher, final MOB student)
{
return false;
}
@Override
public boolean canBeLearnedBy(final MOB teacher, final MOB student)
{
return false;
}
@Override
public void teach(final MOB teacher, final MOB student)
{
}
@Override
public void unlearn(final MOB student)
{
}
@Override
public void practice(final MOB teacher, final MOB student)
{
}
@Override
public int maxRange()
{
return Integer.MAX_VALUE;
}
@Override
public int minRange()
{
return Integer.MIN_VALUE;
}
@Override
public void startTickDown(final MOB invokerMOB, final Physical affected, final int tickTime)
{
if (affected.fetchEffect(ID()) == null)
affected.addEffect(this);
}
@Override
public int proficiency()
{
return 0;
}
@Override
public void setProficiency(final int newProficiency)
{
}
@Override
public boolean proficiencyCheck(final MOB mob, final int adjustment, final boolean auto)
{
return false;
}
@Override
public void helpProficiency(final MOB mob, final int adjustment)
{
}
@Override
public Physical affecting()
{
return affected;
}
@Override
public void setAffectedOne(final Physical P)
{
affected = P;
}
@Override
public boolean putInCommandlist()
{
return false;
}
@Override
public int abstractQuality()
{
return Ability.QUALITY_INDIFFERENT;
}
@Override
public int enchantQuality()
{
return Ability.QUALITY_INDIFFERENT;
}
@Override
public int castingQuality(final MOB mob, final Physical target)
{
return Ability.QUALITY_INDIFFERENT;
}
@Override
public int classificationCode()
{
return Ability.ACODE_PROPERTY;
}
@Override
public boolean isSavable()
{
return savable;
}
@Override
public void setSavable(final boolean truefalse)
{
savable = truefalse;
}
protected boolean amDestroyed = false;
@Override
public void destroy()
{
amDestroyed = true;
}
@Override
public boolean amDestroyed()
{
return amDestroyed;
}
@Override
public CMObject newInstance()
{
try
{
return this.getClass().newInstance();
}
catch (final Exception e)
{
Log.errOut(ID(), e);
}
return new AWaterCurrent();
}
@Override
public int getSaveStatIndex()
{
return getStatCodes().length;
}
@Override
public String[] getStatCodes()
{
return CODES;
}
@Override
public boolean isStat(final String code)
{
return CMParms.indexOf(getStatCodes(), code.toUpperCase().trim()) >= 0;
}
protected int getCodeNum(final String code)
{
for(int i=0;i<CODES.length;i++)
{
if(code.equalsIgnoreCase(CODES[i]))
return i;
}
return -1;
}
@Override
public String getStat(final String code)
{
switch(getCodeNum(code))
{
case 0:
return ID();
case 1:
return text();
}
return "";
}
@Override
public void setStat(final String code, final String val)
{
switch(getCodeNum(code))
{
case 0:
return;
case 1:
setMiscText(val);
break;
}
}
@Override
public boolean sameAs(final Environmental E)
{
if(!(E instanceof AWaterCurrent))
return false;
final String[] codes=getStatCodes();
for(int i=0;i<codes.length;i++)
{
if(!E.getStat(codes[i]).equals(getStat(codes[i])))
return false;
}
return true;
}
protected void cloneFix(final Ability E)
{
}
@Override
public CMObject copyOf()
{
try
{
final AWaterCurrent E=(AWaterCurrent)this.clone();
//CMClass.bumpCounter(E,CMClass.CMObjectType.ABILITY);//removed for mem & perf
E.cloneFix(this);
return E;
}
catch(final CloneNotSupportedException e)
{
return this.newInstance();
}
}
@Override
public int compareTo(final CMObject o)
{
return CMClass.classID(this).compareToIgnoreCase(CMClass.classID(o));
}
@Override
public void setMiscText(final String newMiscText)
{
miscText = newMiscText;
}
@Override
public String text()
{
return miscText;
}
@Override
public String miscTextFormat()
{
return CMParms.FORMAT_UNDEFINED;
}
@Override
public boolean appropriateToMyFactions(final MOB mob)
{
return true;
}
@Override
public String accountForYourself()
{
return "";
}
@Override
public String requirements(final MOB mob)
{
return "";
}
@Override
public boolean canAffect(final Physical P)
{
if((P==null)&&(canAffectCode()==0))
return true;
if(P==null)
return false;
if((P instanceof MOB)&&((canAffectCode()&Ability.CAN_MOBS)>0))
return true;
if((P instanceof Item)&&((canAffectCode()&Ability.CAN_ITEMS)>0))
return true;
if((P instanceof Exit)&&((canAffectCode()&Ability.CAN_EXITS)>0))
return true;
if((P instanceof Room)&&((canAffectCode()&Ability.CAN_ROOMS)>0))
return true;
if((P instanceof Area)&&((canAffectCode()&Ability.CAN_AREAS)>0))
return true;
return false;
}
@Override
public boolean canTarget(final Physical P)
{
return false;
}
@Override
public void affectPhyStats(final Physical affected, final PhyStats affectableStats)
{
}
@Override
public void affectCharStats(final MOB affectedMob, final CharStats affectableStats)
{
}
@Override
public void affectCharState(final MOB affectedMob, final CharState affectableMaxState)
{
}
@Override
public void executeMsg(final Environmental myHost, final CMMsg msg)
{
return;
}
@Override
public boolean okMessage(final Environmental myHost, final CMMsg msg)
{
return true;
}
@Override
public boolean tick(final Tickable ticking, final int tickID)
{
return true;
}
@Override
public void makeLongLasting()
{
}
@Override
public void makeNonUninvokable()
{
}
protected static final int[] cost = new int[Ability.CACHEINDEX_TOTAL];
@Override
public int[] usageCost(final MOB mob, final boolean ignoreClassOverride)
{
return cost;
}
@Override
public boolean isGeneric()
{
return false;
}
}
}