package com.planet_ink.coffee_mud.Locales; import com.planet_ink.coffee_mud.Libraries.interfaces.*; import com.planet_ink.coffee_mud.core.interfaces.*; import com.planet_ink.coffee_mud.core.interfaces.EachApplicable.ApplyAffectPhyStats; import com.planet_ink.coffee_mud.core.interfaces.ItemPossessor.Expire; import com.planet_ink.coffee_mud.core.interfaces.ItemPossessor.Move; 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.Locales.interfaces.*; import com.planet_ink.coffee_mud.MOBS.interfaces.*; import com.planet_ink.coffee_mud.Races.interfaces.*; import java.util.*; /* Copyright 2001-2016 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 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 StdRoom implements Room { @Override public String ID() { return "StdRoom"; } protected String myID = ""; protected String name = "the room"; protected String displayText = L("Standard Room"); protected String rawImageName = null; protected String cachedImageName = null; protected Object description = null; protected Area myArea = null; protected PhyStats phyStats = (PhyStats) CMClass.getCommon("DefaultPhyStats"); protected PhyStats basePhyStats = (PhyStats) CMClass.getCommon("DefaultPhyStats"); protected Exit[] exits = new Exit[Directions.NUM_DIRECTIONS()]; protected Room[] doors = new Room[Directions.NUM_DIRECTIONS()]; protected String[] xtraValues = null; protected boolean mobility = true; protected GridLocale gridParent = null; protected int tickStatus = Tickable.STATUS_NOT; protected long expirationDate = 0; protected int atmosphere = ATMOSPHERE_INHERIT; protected int climask = CLIMASK_INHERIT; protected int myResource = -1; protected long lastResourceTime = 0; protected boolean amDestroyed = false; protected boolean skyedYet = false; protected volatile short combatTurnMobIndex = 0; protected final short[] roomRecoverMarker = new short[1]; protected SVector<Ability> affects = null; protected SVector<Behavior> behaviors = null; protected SVector<ScriptingEngine> scripts = null; protected SVector<MOB> inhabitants = new SVector<MOB>(1); protected SVector<Item> contents = new SVector<Item>(1); protected Room me = this; @SuppressWarnings("rawtypes") protected ApplyAffectPhyStats affectPhyStats = new ApplyAffectPhyStats<Physical>(this); // base move points and thirst points per round public StdRoom() { super(); //CMClass.bumpCounter(this,CMClass.CMObjectType.LOCALE);//removed for mem & perf xtraValues=CMProps.getExtraStatCodesHolder(this); basePhyStats.setWeight(2); recoverPhyStats(); } //protected void finalize(){ CMClass.unbumpCounter(this,CMClass.CMObjectType.LOCALE); }//removed for mem & perf @Override public void initializeClass() { } @Override public CMObject newInstance() { try { return this.getClass().newInstance(); } catch(final Exception e) { Log.errOut(ID(),e); } return new StdRoom(); } @Override public String roomID() { return myID ; } @Override public String Name() { return name; } @Override public void setName(String newName) { name=newName; } @Override public String name() { if(phyStats().newName()!=null) return phyStats().newName(); return name; } @Override public String name(MOB viewerMob) { return name(); } @Override public int getAtmosphereCode() { return atmosphere; } @Override public void setAtmosphere(int resourceCode) { atmosphere=resourceCode; } @Override public int getAtmosphere() { if(getGridParent()!=null) return getGridParent().getAtmosphere(); else if(getAtmosphereCode()!=ATMOSPHERE_INHERIT) return getAtmosphereCode(); else return (myArea==null)?RawMaterial.RESOURCE_AIR:myArea.getAtmosphere(); } @Override public String image() { if(cachedImageName==null) { if((rawImageName!=null)&&(rawImageName.length()>0)) cachedImageName=rawImageName; else cachedImageName=CMLib.protocol().getDefaultMXPImage(this); } return cachedImageName; } @Override public String rawImage() { if(rawImageName==null) return ""; return rawImageName; } @Override public void setImage(String newImage) { if((newImage==null)||(newImage.trim().length()==0)) rawImageName=null; else rawImageName=newImage; if((cachedImageName!=null)&&(!cachedImageName.equals(newImage))) cachedImageName=null; } @Override public boolean isGeneric() { return false; } @SuppressWarnings("rawtypes") protected void cloneFix(Room R) { me=this; basePhyStats=(PhyStats)R.basePhyStats().copyOf(); phyStats=(PhyStats)R.phyStats().copyOf(); affectPhyStats = new ApplyAffectPhyStats(this); contents=new SVector<Item>(); inhabitants=new SVector<MOB>(); affects=null; behaviors=null; scripts=null; exits=new Exit[exits.length]; Arrays.fill(exits, null); doors=new Room[doors.length]; Arrays.fill(doors, null); for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { if(R.getRawExit(d)!=null) exits[d]=(Exit)R.getRawExit(d).copyOf(); if(R.rawDoors()[d]!=null) doors[d]=R.rawDoors()[d]; } for(int i=0;i<R.numItems();i++) { final Item I2=R.getItem(i); if(I2!=null) { final Item I=(Item)I2.copyOf(); I.setOwner(this); contents.addElement(I); } } for(int i=0;i<numItems();i++) { final Item I2=getItem(i); if((I2!=null) &&(I2.container()!=null) &&(!isContent(I2.container()))) { for(int ii=0;ii<R.numItems();ii++) { if((R.getItem(ii)==I2.container())&&(ii<numItems())) { I2.setContainer((Container)getItem(ii)); break; } } } } for(final Enumeration<MOB> m=R.inhabitants();m.hasMoreElements();) { final MOB M2=m.nextElement(); if(M2.isSavable()) { final MOB M=(MOB)M2.copyOf(); if(M.getStartRoom()==R) M.setStartRoom(this); M.setLocation(this); inhabitants.addElement(M); } } for(final Enumeration<Ability> a=R.effects();a.hasMoreElements();) { final Ability A=a.nextElement(); if(!A.canBeUninvoked()) addEffect((Ability)A.copyOf()); } for(final Enumeration<Behavior> e=R.behaviors();e.hasMoreElements();) { final Behavior B=e.nextElement(); addBehavior((Behavior)B.copyOf()); } for(final Enumeration<ScriptingEngine> e=R.scripts();e.hasMoreElements();) { final ScriptingEngine SE=e.nextElement(); addScript((ScriptingEngine)SE.copyOf()); } } @Override public CMObject copyOf() { try { final StdRoom R=(StdRoom)this.clone(); //CMClass.bumpCounter(R,CMClass.CMObjectType.LOCALE);//removed for mem & perf R.xtraValues=(xtraValues==null)?null:(String[])xtraValues.clone(); R.cloneFix(this); return R; } catch(final CloneNotSupportedException e) { return this.newInstance(); } } @Override public int domainType() { return Room.DOMAIN_OUTDOORS_CITY; } @Override public int getClimateTypeCode() { return climask; } @Override public void setClimateType(int climask) { this.climask=climask; } @Override public int getClimateType() { if(getGridParent()!=null) return getGridParent().getClimateType(); else if(getClimateTypeCode()!=CLIMASK_INHERIT) return getClimateTypeCode(); else return (myArea==null)?CLIMASK_NORMAL:myArea.getClimateType(); } @Override public long expirationDate() { return expirationDate; } @Override public void setExpirationDate(long time) { expirationDate=time; } @Override public void setRawExit(int direction, Exit to) { if((direction<0)||(direction>=exits.length)) return; final Exit E=exits[direction]; if(E==to) return; if(E!=null) E.exitUsage((short)-1); if(to !=null ) { to.exitUsage((short)1); exits[direction]=to; } else exits[direction]=null; /** * cant be done * if((E!=null)&&(E.exitUsage((short)0)==0)) E.destroy(); */ } @Override public String displayText() { return displayText; } @Override public void setDisplayText(String newDisplayText) { displayText=newDisplayText; } @Override public String description() { if(description == null) { if(CMProps.getBoolVar(CMProps.Bool.ROOMDNOCACHE)&&(roomID().trim().length()>0)) { final String txt=CMLib.database().DBReadRoomDesc(roomID()); if(txt==null) { Log.errOut("Unable to recover description for "+roomID()+"."); return ""; } return txt; } return ""; } else if(description instanceof byte[]) { final byte[] descriptionBytes=(byte[])description; if(CMProps.getBoolVar(CMProps.Bool.ROOMDCOMPRESS)) return CMLib.encoder().decompressString(descriptionBytes); return CMStrings.bytesToStr(descriptionBytes); } else return ((String)description); } @Override public void setDescription(String newDescription) { if(newDescription.length()==0) description=null; else if(CMProps.getBoolVar(CMProps.Bool.ROOMDCOMPRESS)) description=CMLib.encoder().compressString(newDescription); else description=newDescription; } @Override public String text() { return CMLib.coffeeMaker().getPropertiesStr(this,true); } @Override public String miscTextFormat() { return CMParms.FORMAT_UNDEFINED; } @Override public void setMiscText(String newMiscText) { if(newMiscText.trim().length()>0) CMLib.coffeeMaker().setPropertiesStr(this,newMiscText,true); } @Override public void setRoomID(String newID) { if((myID!=null)&&(!myID.equals(newID))) { myID=newID; if(myArea!=null) { // force the re-sort myArea.delProperRoom(this); myArea.addProperRoom(this); } } else myID=newID; } @Override public Area getArea() { if(myArea==null) return CMClass.randomArea(); return myArea; } @Override public void setArea(Area newArea) { if(newArea!=myArea) { if(myArea!=null) myArea.delProperRoom(this); myArea=newArea; if(myArea!=null) myArea.addProperRoom(this); } } @Override public void setGridParent(GridLocale room) { gridParent=room; } @Override public GridLocale getGridParent() { return gridParent; } @Override public void giveASky(int depth) { if(skyedYet) return; if(depth>1000) return; skyedYet=true; if((roomID().length()==0) &&(getGridParent()!=null) &&(getGridParent().roomID().length()==0)) return; if((rawDoors()[Directions.UP]==null) &&((domainType()&Room.INDOORS)==0) &&(domainType()!=Room.DOMAIN_OUTDOORS_UNDERWATER) &&(domainType()!=Room.DOMAIN_OUTDOORS_AIR) &&(CMProps.getIntVar(CMProps.Int.SKYSIZE)!=0)) { Exit upE=null; final Exit dnE=CMClass.getExit("StdOpenDoorway"); if(CMProps.getIntVar(CMProps.Int.SKYSIZE)>0) upE=dnE; else upE=CMClass.getExit("UnseenWalkway"); final GridLocale sky=(GridLocale)CMClass.getLocale("EndlessThinSky"); sky.setRoomID(""); sky.setArea(getArea()); rawDoors()[Directions.UP]=sky; setRawExit(Directions.UP,upE); sky.rawDoors()[Directions.DOWN]=this; sky.setRawExit(Directions.DOWN,dnE); if(!(getArea() instanceof BoardableShip)) { for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { if((d!=Directions.UP)&&(d!=Directions.DOWN)) { Room thatRoom=rawDoors()[d]; if((thatRoom!=null)&&(getRawExit(d)!=null)) { thatRoom=CMLib.map().getRoom(thatRoom); thatRoom.giveASky(depth+1); final Room thatSky=thatRoom.rawDoors()[Directions.UP]; if((thatSky!=null) &&(thatSky.roomID().length()==0) &&((thatSky instanceof EndlessThinSky)||(thatSky instanceof EndlessSky))) { Exit xo=getRawExit(d); if(xo!=null) { if(xo.hasADoor()) xo=dnE; sky.rawDoors()[d]=thatSky; sky.setRawExit(d,xo); } final int opDir=Directions.getOpDirectionCode(d); xo=thatRoom.getRawExit(opDir); if(xo!=null) { if(xo.hasADoor()) xo=dnE; thatSky.rawDoors()[opDir]=sky; thatSky.setRawExit(opDir,xo); } ((GridLocale)thatSky).clearGrid(null); } } } } } sky.clearGrid(null); } } @Override public void clearSky() { if(!skyedYet) return; final Room skyGridRoom=rawDoors()[Directions.UP]; if(skyGridRoom!=null) { if(((skyGridRoom.roomID()==null)||(skyGridRoom.roomID().length()==0)) &&((skyGridRoom instanceof EndlessSky)||(skyGridRoom instanceof EndlessThinSky))) { ((GridLocale)skyGridRoom).clearGrid(null); rawDoors()[Directions.UP]=null; setRawExit(Directions.UP,null); for(int d=0;d<Directions.NUM_DIRECTIONS();d++) { final Room thatSky=skyGridRoom.rawDoors()[d]; final int opDir=Directions.getOpDirectionCode(d); if((thatSky!=null)&&(thatSky.rawDoors()[opDir]==skyGridRoom)) { thatSky.rawDoors()[opDir]=null; thatSky.setRawExit(opDir, null); } skyGridRoom.rawDoors()[d]=null; skyGridRoom.setRawExit(d,null); } CMLib.map().emptyRoom(skyGridRoom,null,true); skyGridRoom.destroy(); skyedYet=false; } } } @Override public List<Integer> resourceChoices() { return null; } @Override public void setResource(int resourceCode) { myResource=resourceCode; lastResourceTime= (resourceCode>=0) ? lastResourceTime=System.currentTimeMillis() : 0; } @Override public int myResource() { if(lastResourceTime!=0) { if(lastResourceTime<(System.currentTimeMillis()-(30*TimeManager.MILI_MINUTE))) setResource(-1); } if(myResource<0) { if(resourceChoices()==null) setResource(-1); else { int totalChance=0; for(int i=0;i<resourceChoices().size();i++) { final int resource=resourceChoices().get(i).intValue(); totalChance+=RawMaterial.CODES.FREQUENCY(resource); } setResource(-1); final int theRoll=CMLib.dice().roll(1,totalChance,0); totalChance=0; for(int i=0;i<resourceChoices().size();i++) { final int resource=resourceChoices().get(i).intValue(); totalChance+=RawMaterial.CODES.FREQUENCY(resource); if(theRoll<=totalChance) { setResource(resource); break; } } } } return myResource; } @Override public void toggleMobility(boolean onoff) { mobility=onoff; } @Override public boolean getMobility() { return mobility; } @Override public boolean okMessage(final Environmental myHost, final CMMsg msg) { if(!getArea().okMessage(this,msg)) return false; if(msg.amITarget(this)) { final MOB mob=msg.source(); switch(msg.targetMinor()) { case CMMsg.TYP_EXPIRE: { if((gridParent!=null)&&(!gridParent.okMessage(myHost,msg))) return false; if(!CMLib.map().isClearableRoom(this)) return false; for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { final Room R2=rawDoors()[d]; if((R2!=null)&&(!CMLib.map().isClearableRoom(R2))) return false; } break; } case CMMsg.TYP_LEAVE: if((!CMLib.flags().allowsMovement(this))||(!getMobility())) return false; break; case CMMsg.TYP_FLEE: case CMMsg.TYP_ENTER: if((!CMLib.flags().allowsMovement(this))||(!getMobility())) return false; if(!mob.isMonster()) { final Room R=mob.location(); if((R!=null)&&(R.getArea()!=getArea())) CMLib.factions().updatePlayerFactions(mob,this,false); giveASky(0); } break; case CMMsg.TYP_AREAAFFECT: // obsolete with the area objects break; case CMMsg.TYP_CAST_SPELL: case CMMsg.TYP_DELICATE_HANDS_ACT: case CMMsg.TYP_OK_ACTION: case CMMsg.TYP_JUSTICE: case CMMsg.TYP_OK_VISUAL: case CMMsg.TYP_SNIFF: break; case CMMsg.TYP_LIST: case CMMsg.TYP_BUY: case CMMsg.TYP_BID: case CMMsg.TYP_SELL: case CMMsg.TYP_VIEW: case CMMsg.TYP_VALUE: if(CMLib.coffeeShops().getShopKeeper(this)==null) { mob.tell(L("You can't shop here.")); return false; } break; case CMMsg.TYP_SPEAK: break; case CMMsg.TYP_DIG: if(CMLib.map().getExtendedRoomID(this).length()==0) { mob.tell(L("You can't really dig here.")); return false; } switch(this.domainType()) { case Room.DOMAIN_OUTDOORS_DESERT: case Room.DOMAIN_OUTDOORS_HILLS: case Room.DOMAIN_OUTDOORS_JUNGLE: case Room.DOMAIN_OUTDOORS_PLAINS: case Room.DOMAIN_OUTDOORS_SWAMP: case Room.DOMAIN_OUTDOORS_WOODS: break; case Room.DOMAIN_OUTDOORS_WATERSURFACE: case Room.DOMAIN_OUTDOORS_UNDERWATER: { if(getRoomInDir(Directions.DOWN)==null) break; } //$FALL-THROUGH$ default: mob.tell(L("You can't really dig here.")); return false; } break; default: if(((msg.targetMajor(CMMsg.MASK_HANDS))||(msg.targetMajor(CMMsg.MASK_MOUTH))) &&(!CMath.bset(msg.targetMajor(), CMMsg.MASK_MAGIC)) &&(msg.targetMinor()!=CMMsg.TYP_THROW) &&(isInhabitant(msg.source()))) { mob.tell(L("You can't do that here.")); return false; } break; } } if(isInhabitant(msg.source())) { if(!msg.source().okMessage(this,msg)) return false; } for(final Enumeration<MOB> m=inhabitants();m.hasMoreElements();) { final MOB M=m.nextElement(); if((M!=msg.source()) &&(!M.okMessage(this,msg))) return false; } for(final Enumeration<Item> i=items();i.hasMoreElements();) { final Item I=i.nextElement(); if(!I.okMessage(this,msg)) return false; } for(final Enumeration<Ability> a=effects();a.hasMoreElements();) { final Ability A=a.nextElement(); if(!A.okMessage(this,msg)) return false; } for(final Enumeration<Behavior> b=behaviors();b.hasMoreElements();) { final Behavior B=b.nextElement(); if(!B.okMessage(this,msg)) return false; } for(final Enumeration<ScriptingEngine> s=scripts();s.hasMoreElements();) { final ScriptingEngine S=s.nextElement(); if(!S.okMessage(this,msg)) return false; } for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { final Exit thisExit=getRawExit(d); if(thisExit!=null) { if(!thisExit.okMessage(this,msg)) return false; } } return true; } @Override public void executeMsg(final Environmental myHost, final CMMsg msg) { getArea().executeMsg(this,msg); if(msg.amITarget(this)) { final MOB mob=msg.source(); switch(msg.targetMinor()) { case CMMsg.TYP_LEAVE: { if(!CMath.bset(msg.targetMajor(),CMMsg.MASK_OPTIMIZE)) recoverRoomStats(); break; } case CMMsg.TYP_FLEE: { if(!CMath.bset(msg.targetMajor(),CMMsg.MASK_OPTIMIZE)) recoverRoomStats(); break; } case CMMsg.TYP_ENTER: case CMMsg.TYP_RECALL: { if(!CMath.bset(msg.targetMajor(),CMMsg.MASK_OPTIMIZE)) recoverRoomStats(); if((msg.source().playerStats()!=null)&&(msg.source().soulMate()==null)) { if(msg.source().playerStats().addRoomVisit(this)) CMLib.players().bumpPrideStat(msg.source(),AccountStats.PrideStat.ROOMS_EXPLORED, 1); } break; } case CMMsg.TYP_LOOK_EXITS: if(msg.value()==CMMsg.MASK_OPTIMIZE) CMLib.commands().lookAtExitsShort(this, msg.source()); else CMLib.commands().lookAtExits(this, msg.source()); break; case CMMsg.TYP_LOOK: case CMMsg.TYP_EXAMINE: CMLib.commands().handleBeingLookedAt(msg); break; case CMMsg.TYP_SNIFF: CMLib.commands().handleBeingSniffed(msg); break; case CMMsg.TYP_READ: if(CMLib.flags().canBeSeenBy(this,mob)) mob.tell(L("There is nothing written here.")); else mob.tell(L("You can't see that!")); break; case CMMsg.TYP_AREAAFFECT: // obsolete with the area objects break; case CMMsg.TYP_DIG: if(findItem("HoleInTheGround")==null) addItem(CMClass.getBasicItem("HoleInTheGround")); break; default: break; } } if(numItems()>0) { eachItem(new EachApplicable<Item>() { @Override public final void apply(final Item I) { I.executeMsg(me, msg); } }); } for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { final Exit E=getRawExit(d); if(E!=null) E.executeMsg(this,msg); } if(numBehaviors()>0) { eachBehavior(new EachApplicable<Behavior>() { @Override public final void apply(final Behavior B) { B.executeMsg(me, msg); } }); } if(numScripts()>0) { eachScript(new EachApplicable<ScriptingEngine>() { @Override public final void apply(final ScriptingEngine S) { S.executeMsg(me, msg); } }); } if(numEffects()>0) { eachEffect(new EachApplicable<Ability>() { @Override public final void apply(final Ability A) { A.executeMsg(me,msg); } }); } if(msg.sourceMinor()==CMMsg.TYP_SHUTDOWN) { try { if(CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMMOBS)) { if(roomID().length()==0) { eachInhabitant(new EachApplicable<MOB>() { @Override public final void apply(final MOB M) { if((M.isSavable()) &&(M.getStartRoom()!=me) &&(M.getStartRoom()!=null) &&(M.getStartRoom().roomID().length()>0)) M.getStartRoom().bringMobHere(M,false); } }); } } else if(CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMSHOPS)) { eachInhabitant(new EachApplicable<MOB>() { @Override public final void apply(final MOB M) { if((M instanceof ShopKeeper) &&(M.isSavable()) &&(M.getStartRoom()!=me) &&(M.getStartRoom()!=null)) M.getStartRoom().bringMobHere(M,false); } }); } }catch(final NoSuchElementException e){} } if(msg.amITarget(this)&&(msg.targetMinor()==CMMsg.TYP_EXPIRE)) { synchronized(("SYNC"+roomID()).intern()) { final LinkedList<DeadBody> deadBodies=new LinkedList<DeadBody>(); eachItem(new EachApplicable<Item>() { @Override public final void apply(final Item I) { if((I instanceof DeadBody) &&(((DeadBody)I).isPlayerCorpse())) deadBodies.add((DeadBody)I); } }); for(final DeadBody D : deadBodies) { MOB M=CMLib.players().getLoadPlayer(D.getMobName()); if(M==null) M=D.getSavedMOB(); if((M!=null)&&(M.getStartRoom()!=null)) { final Room startRoom=CMLib.map().getRoom(M.getStartRoom()); M.tell(L("Your corpse has been moved to @x1",startRoom.displayText())); startRoom.moveItemTo(D); } } if(gridParent!=null) gridParent.executeMsg(myHost,msg); else if((roomID().length()>0) &&((getArea()==null)||(!CMath.bset(getArea().flags(), Area.FLAG_INSTANCE_CHILD))) &&(CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMMOBS) ||CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMITEMS) ||CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMSHOPS))) { final ArrayList<MOB> shopmobs=new ArrayList<MOB>(1); final ArrayList<Item> bodies=new ArrayList<Item>(1); if(CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMMOBS)) { eachInhabitant(new EachApplicable<MOB>() { @Override public final void apply(final MOB M) { if(M.isSavable()) { M.setStartRoom(me); M.text(); // this permanizes his current state } } }); CMLib.database().DBUpdateMOBs(this); } else if(CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMSHOPS)) { eachInhabitant(new EachApplicable<MOB>() { @Override public final void apply(final MOB M) { if((M.isSavable()) &&(M instanceof ShopKeeper) &&(M.getStartRoom()==me)) shopmobs.add(M); } }); if(!shopmobs.isEmpty()) CMLib.database().DBUpdateTheseMOBs(this,shopmobs); } if(CMSecurity.isSaveFlag(CMSecurity.SaveFlag.ROOMITEMS)) { eachItem(new EachApplicable<Item>() { @Override public final void apply(final Item I) { if(I instanceof DeadBody) bodies.add(I); } }); for(int i=0;i<bodies.size();i++) bodies.get(i).destroy(); CMLib.database().DBUpdateItems(this); } } final Area A=getArea(); final String roomID=roomID(); setGridParent(null); if(!CMProps.getBoolVar(CMProps.Bool.MUDSHUTTINGDOWN)) { CMLib.map().emptyRoom(this,null,true); destroy(); if(roomID.length()>0) A.addProperRoomnumber(roomID); } } } } @Override public void startItemRejuv() { eachItem(new EachApplicable<Item>() { @Override public final void apply(final Item item) { if(item.container()==null) { final ItemTicker I=(ItemTicker)CMClass.getAbility("ItemRejuv"); I.unloadIfNecessary(item); if((item.phyStats().rejuv()!=PhyStats.NO_REJUV) &&(item.phyStats().rejuv()>0)) I.loadMeUp(item,me); } } }); } @Override public int getTickStatus() { return tickStatus; } @Override public boolean tick(final Tickable ticking, final int tickID) { tickStatus=Tickable.STATUS_START; if(tickID==Tickable.TICKID_ROOM_BEHAVIOR) { if((numBehaviors()<=0)&&(numScripts()<=0)) return false; tickStatus=Tickable.STATUS_BEHAVIOR; eachBehavior(new EachApplicable<Behavior>() { @Override public final void apply(final Behavior B) { B.tick(ticking, tickID); } }); tickStatus=Tickable.STATUS_SCRIPT; eachScript(new EachApplicable<ScriptingEngine>() { @Override public final void apply(final ScriptingEngine S) { S.tick(ticking, tickID); } }); } else { tickStatus=Tickable.STATUS_AFFECT; if(numEffects()>0) { eachEffect(new EachApplicable<Ability>() { @Override public final void apply(final Ability A) { if(!A.tick(ticking,tickID)) A.unInvoke(); } }); } } tickStatus=Tickable.STATUS_NOT; return !amDestroyed(); } @Override public PhyStats phyStats() { return phyStats; } @Override public PhyStats basePhyStats() { return basePhyStats; } @SuppressWarnings("unchecked") @Override public void recoverPhyStats() { basePhyStats.copyInto(phyStats); final Area myArea=getArea(); if(myArea!=null) myArea.affectPhyStats(this,phyStats()); eachEffect(affectPhyStats); eachItem(affectPhyStats); eachInhabitant(affectPhyStats); } private final static EachApplicable<Item> recoverRoomStatsItemApplicable=new EachApplicable.ApplyRecoverPhyStats<Item>(); private final static EachApplicable<MOB> recoverRoomStatsInhabitantApplicable=new EachApplicable<MOB>() { @Override public final void apply(final MOB M) { M.recoverCharStats(); M.recoverPhyStats(); M.recoverMaxState(); } }; private void reallyRecoverRoomStats() { recoverPhyStats(); eachInhabitant(recoverRoomStatsInhabitantApplicable); for(final Exit X : exits) { if(X!=null) X.recoverPhyStats(); } eachItem(recoverRoomStatsItemApplicable); } @Override public void recoverRoomStats() { synchronized(roomRecoverMarker) { roomRecoverMarker[0]++; if(roomRecoverMarker[0]!=1) { return; } try { reallyRecoverRoomStats(); roomRecoverMarker[0]--; if(roomRecoverMarker[0]>0) reallyRecoverRoomStats(); } finally { roomRecoverMarker[0]=0; } } } @Override public void setBasePhyStats(PhyStats newStats) { basePhyStats=(PhyStats)newStats.copyOf(); } private final static int phyStatsMaskOut = ~(PhyStats.IS_DARK|PhyStats.IS_LIGHTSOURCE|PhyStats.IS_SLEEPING|PhyStats.IS_HIDDEN|PhyStats.IS_SWIMMING|PhyStats.IS_NOT_SEEN); @Override public void affectPhyStats(final Physical affected, final PhyStats affectableStats) { getArea().affectPhyStats(affected,affectableStats); final int disposition=phyStats().disposition() & phyStatsMaskOut; if(disposition>0) affectableStats.setDisposition(affectableStats.disposition()|disposition); eachEffect(new EachApplicable<Ability>() { @Override public final void apply(final Ability A) { if(A.bubbleAffect()) A.affectPhyStats(affected,affectableStats); } }); } @Override public void affectCharStats(final MOB affectedMob, final CharStats affectableStats) { getArea().affectCharStats(affectedMob,affectableStats); eachEffect(new EachApplicable<Ability>(){ @Override public final void apply(final Ability A) { if(A.bubbleAffect()) A.affectCharStats(affectedMob,affectableStats); } }); } @Override public void affectCharState(final MOB affectedMob, final CharState affectableMaxState) { getArea().affectCharState(affectedMob,affectableMaxState); eachEffect(new EachApplicable<Ability>(){ @Override public final void apply(final Ability A) { if(A.bubbleAffect()) A.affectCharState(affectedMob,affectableMaxState); } }); } @Override public int compareTo(CMObject o) { return CMClass.classID(this).compareToIgnoreCase(CMClass.classID(o)); } protected String parseVariesCodes(final MOB mob, final Area A, final String text) { final StringBuilder buf=new StringBuilder(""); int aligatorDex=text.indexOf('<'); int curDex=0; boolean addMe = true; while(aligatorDex>=0) { for(final VariationCode code : VariationCode.values()) { if(text.startsWith(code.openTag, aligatorDex)) { buf.append(text.substring(curDex, aligatorDex)); int y=text.indexOf(code.closeTag,aligatorDex+code.openTag.length()); if(y<0) { curDex = text.length(); y=text.length(); } else curDex = y+code.closeTag.length(); switch(code.c) { case '\n': addMe = !addMe; break; case '\r': addMe = true; break; case 'W': addMe = A.getClimateObj().weatherType(null) == code.num; break; case 'C': addMe = A.getTimeObj().getTODCode().ordinal() == code.num; break; case 'S': addMe = A.getTimeObj().getSeasonCode().ordinal() == code.num; break; case 'M': addMe = ((mob != null) && (CMath.bset(mob.phyStats().disposition(), code.num))); break; case 'V': addMe = ((mob != null) && (mob.playerStats() != null) && (mob.playerStats().hasVisited(this))); break; } if(addMe) buf.append(parseVariesCodes(mob,A,text.substring(aligatorDex+code.openTag.length(),y))); aligatorDex=curDex-1; break; } } if(aligatorDex >= text.length()-1) break; aligatorDex=text.indexOf('<',aligatorDex+1); } if(curDex < text.length()) { if(text.startsWith("</VARIES>",curDex)) buf.append(text.substring(curDex+9)); else buf.append(text.substring(curDex)); } return buf.toString(); } protected String parseVaries(MOB mob, String text) { if(text.startsWith("<VARIES>")) return parseVariesCodes(mob,getArea(),text.substring(8)); return text; } @Override public String displayText(MOB mob) { return parseVaries(mob,displayText()); } @Override public String description(MOB mob) { return parseVaries(mob,description()); } @Override public void bringMobHere(MOB mob, boolean andFollowers) { if(mob==null) return; final Room oldRoom=mob.location(); if(oldRoom!=null) oldRoom.delInhabitant(mob); addInhabitant(mob); mob.setLocation(this); if((andFollowers)&&(oldRoom!=null)) { for(final Enumeration<Pair<MOB,Short>> f=mob.followers();f.hasMoreElements();) { final MOB folM=f.nextElement().first; if(folM.location()==oldRoom) bringMobHere(folM,true); } } final Rideable RI=mob.riding(); if((RI!=null)&&(CMLib.map().roomLocation(RI)==oldRoom)) { if((RI.isMobileRideBasis()) &&((!(RI instanceof Item))||(!CMLib.flags().isMobile(RI)))) { if(RI instanceof MOB) bringMobHere((MOB)RI,andFollowers); else if(RI instanceof Item) { if(andFollowers) moveItemTo((Item)RI,ItemPossessor.Expire.Player_Drop,Move.Followers); else moveItemTo((Item)RI,ItemPossessor.Expire.Player_Drop); } // refuse is good for above, since mostly moving player stuff around } else mob.setRiding(null); } if((oldRoom!=null)&&(mob instanceof Rideable)&&(oldRoom!=this)) { for(final Enumeration<Rider> r=((Rideable)mob).riders();r.hasMoreElements();) { final Rider RR=r.nextElement(); if(CMLib.map().roomLocation(RR)==oldRoom) { if(((Rideable)mob).isMobileRideBasis()) { if((RR instanceof MOB)&&(andFollowers)) bringMobHere((MOB)RR,andFollowers); else if(RR instanceof Item) { if(andFollowers) moveItemTo((Item)RR,ItemPossessor.Expire.Player_Drop,Move.Followers); else moveItemTo((Item)RR,ItemPossessor.Expire.Player_Drop); // refuse is good for above, since mostly moving player stuff around } } else RR.setRiding(null); } } } if(oldRoom!=null) oldRoom.recoverRoomStats(); recoverRoomStats(); } @Override public void moveItemTo(Item container) { moveItemTo(container, Expire.Never); } @Override public void moveItemTo(Item item, Expire expire, Move... moveFlags) { if(item==null) return; if(item.owner()==null) return; final Environmental o=item.owner(); List<Item> V=new ArrayList<Item>(); if(item instanceof Container) V=((Container)item).getDeepContents(); if(o instanceof MOB) ((MOB)o).delItem(item); if(o instanceof Room) ((Room)o).delItem(item); addItem(item, expire); for(int v=0;v<V.size();v++) { final Item i2=V.get(v); if(o instanceof MOB) ((MOB)o).delItem(i2); if(o instanceof Room) ((Room)o).delItem(i2); addItem(i2); } item.setContainer(null); final Rideable RI=item.riding(); if((RI!=null)&&(o instanceof Room)&&(CMLib.map().roomLocation(RI)==o)) { if((RI.isMobileRideBasis()) &&((!(RI instanceof Item))||(!CMLib.flags().isMobile(RI)))) { if(RI instanceof MOB) bringMobHere((MOB)RI,true); else if(RI instanceof Item) moveItemTo((Item)RI,null,ItemPossessor.Move.Followers); } else item.setRiding(null); } if(CMParms.contains(moveFlags, Move.Followers) &&(o instanceof Room) &&(item instanceof Rideable) &&(o!=this)) { Rider RR=null; for(int r=0;r<((Rideable)item).numRiders();r++) { RR=((Rideable)item).fetchRider(r); if(CMLib.map().roomLocation(RR)==o) { if((((Rideable)item).isMobileRideBasis()) &&(!CMLib.flags().isMobile(item))) { if(RR instanceof MOB) bringMobHere((MOB)RR,true); else if(RR instanceof Item) moveItemTo((Item)RR,ItemPossessor.Expire.Player_Drop,ItemPossessor.Move.Followers); } else RR.setRiding(null); } } } if(o instanceof Room) ((Room)o).recoverRoomStats(); else if(o instanceof MOB) { ((MOB)o).recoverCharStats(); ((MOB)o).recoverPhyStats(); ((MOB)o).recoverMaxState(); } recoverRoomStats(); } @Override public Exit getReverseExit(int direction) { if((direction<0)||(direction>=Directions.NUM_DIRECTIONS())) return null; final Room opRoom=getRoomInDir(direction); if(opRoom!=null) return opRoom.getExitInDir(Directions.getOpDirectionCode(direction)); return null; } @Override public Exit getPairedExit(int direction) { final Exit opExit=getReverseExit(direction); final Exit myExit=getExitInDir(direction); if((myExit==null)||(opExit==null)) return null; if(myExit.hasADoor()!=opExit.hasADoor()) return null; return opExit; } @Override public Room prepareRoomInDir(Room R, int direction) { if(amDestroyed) { if(roomID().length()>0) { final Room thinMeR=CMClass.getLocale("ThinRoom"); thinMeR.setRoomID(roomID()); thinMeR.setArea(myArea); if(R.rawDoors()[direction]==this) R.rawDoors()[direction]=thinMeR; return thinMeR.prepareRoomInDir(R,direction); } return null; } if(expirationDate()!=0) setExpirationDate(System.currentTimeMillis()+WorldMap.ROOM_EXPIRATION_MILLIS); return this; } @Override public Room getRoomInDir(int direction) { if((direction<0)||(direction>=doors.length)||(amDestroyed)) return null; Room nextRoom=rawDoors()[direction]; if(gridParent!=null) nextRoom=gridParent.prepareGridLocale(this,nextRoom,direction); if(nextRoom!=null) { nextRoom=nextRoom.prepareRoomInDir(this,direction); if((nextRoom!=null)&&(nextRoom.amDestroyed())) return null; } return nextRoom; } @Override public Exit getExitInDir(int direction) { if((direction<0)||(direction>=exits.length)) return null; if((gridParent!=null)&&(getRawExit(direction)==null)) getRoomInDir(direction); return getRawExit(direction); } protected void reallyReallySend(MOB source, CMMsg msg) { if((Log.debugChannelOn())&&(CMSecurity.isDebugging(CMSecurity.DbgFlag.MESSAGES))) Log.debugOut("StdRoom",msg.source().ID()+":"+msg.sourceCode()+":"+msg.sourceMessage()+"/"+((msg.target()!=null)?msg.target().ID():"null")+":"+msg.targetCode()+":"+msg.targetMessage()+"/"+((msg.tool()!=null)?msg.tool().ID():"null")+"/"+msg.othersCode()+":"+msg.othersMessage()); for(final MOB otherMOB : inhabitants) { if(otherMOB!=source) otherMOB.executeMsg(otherMOB,msg); } executeMsg(source,msg); CMLib.commands().monitorGlobalMessage(this, msg); } protected void reallySend(MOB source, CMMsg msg, int depth) { reallyReallySend(source,msg); // now handle trailer msgs if(depth<3) { if(msg.trailerMsgs()!=null) { for(final CMMsg msg2 : msg.trailerMsgs()) { if((msg!=msg2) &&((msg2.target()==null) ||(!(msg2.target() instanceof MOB)) ||((!((MOB)msg2.target()).amDead())||(msg2.sourceMinor()==CMMsg.TYP_DEATH))) &&(okMessage(source,msg2))) { source.executeMsg(source,msg2); reallySend(source,msg2,depth+1); } } } if(msg.trailerRunnables()!=null) { for(final Runnable r : msg.trailerRunnables()) CMLib.threads().executeRunnable(r); } } } @Override public void send(MOB source, CMMsg msg) { source.executeMsg(source,msg); reallySend(source,msg,0); } @Override public void sendOthers(MOB source, CMMsg msg) { reallySend(source,msg,0); } @Override public void showHappens(int allCode, String allMessage) { final MOB everywhereMOB=CMLib.map().getFactoryMOB(this); final CMMsg msg=CMClass.getMsg(everywhereMOB,null,null,allCode,allCode,allCode,allMessage); sendOthers(everywhereMOB,msg); everywhereMOB.destroy(); } @Override public void showHappens(int allCode, Environmental like, String allMessage) { final MOB everywhereMOB=CMClass.getFactoryMOB(); everywhereMOB.setName(like.name()); if(like instanceof Physical) everywhereMOB.setBasePhyStats(((Physical)like).phyStats()); everywhereMOB.setLocation(this); everywhereMOB.recoverPhyStats(); final CMMsg msg=CMClass.getMsg(everywhereMOB,null,null,allCode,allCode,allCode,allMessage); send(everywhereMOB,msg); everywhereMOB.destroy(); } @Override public boolean show(MOB source, Environmental target, int allCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,null,allCode,allCode,allCode,allMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; send(source,msg); return true; } @Override public boolean show(MOB source, Environmental target, Environmental tool, int allCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,tool,allCode,allCode,allCode,allMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; send(source,msg); return true; } @Override public boolean show(MOB source, Environmental target, Environmental tool, int srcCode, int tarCode, int othCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,tool,srcCode,tarCode,othCode,allMessage); if((!CMath.bset(srcCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; send(source,msg); return true; } @Override public boolean show(MOB source, Environmental target, Environmental tool, int allCode, String srcMessage, String tarMessage, String othMessage) { final CMMsg msg=CMClass.getMsg(source,target,tool,allCode,srcMessage,allCode,tarMessage,allCode,othMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; send(source,msg); return true; } @Override public boolean show(MOB source, Environmental target, Environmental tool, int srcCode, String srcMessage, int tarCode, String tarMessage, int othCode, String othMessage) { final CMMsg msg=CMClass.getMsg(source,target,tool,srcCode,srcMessage,tarCode,tarMessage,othCode,othMessage); if((!CMath.bset(srcCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; send(source,msg); return true; } @Override public boolean showOthers(MOB source, Environmental target, int allCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,null,allCode,allCode,allCode,allMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; reallySend(source,msg,0); return true; } @Override public boolean showOthers(MOB source, Environmental target, Environmental tool, int allCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,tool,allCode,allCode,allCode,allMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; reallySend(source,msg,0); return true; } @Override public boolean showSource(MOB source, Environmental target, int allCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,null,allCode,allCode,allCode,allMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; source.executeMsg(source,msg); return true; } @Override public boolean showSource(MOB source, Environmental target, Environmental tool, int allCode, String allMessage) { final CMMsg msg=CMClass.getMsg(source,target,tool,allCode,allCode,allCode,allMessage); if((!CMath.bset(allCode,CMMsg.MASK_ALWAYS))&&(!okMessage(source,msg))) return false; source.executeMsg(source,msg); return true; } @Override public Exit getRawExit(final int dir) { if(dir<exits.length) return exits[dir]; return null; } @Override public Room[] rawDoors() { return doors; } @Override public boolean isSavable() { return ((roomID().length()>0) &&((getArea()==null) || (!CMath.bset(getArea().flags(),Area.FLAG_INSTANCE_CHILD))) &&(CMLib.flags().isSavable(this))); } @Override public void setSavable(boolean truefalse) { CMLib.flags().setSavable(this, truefalse); } @Override public void destroy() { if((phyStats().sensesMask()&PhyStats.SENSE_UNDESTROYABLE)>0) return; CMLib.map().registerWorldObjectDestroyed(getArea(),this,this); delAllEffects(true); delAllInhabitants(true); delAllBehaviors(); delAllScripts(); CMLib.threads().deleteTick(this,-1); delAllItems(true); if(this instanceof GridLocale) ((GridLocale)this).clearGrid(null); clearSky(); if((roomID().length()==0)&&(rawDoors()!=null)) { Room roomDir=null; for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { roomDir=rawDoors()[d]; if((roomDir!=null)&&(roomDir.rawDoors()!=null)) { for(int d2=Directions.NUM_DIRECTIONS()-1;d2>=0;d2--) { if(roomDir.rawDoors()[d2]==this) { roomDir.rawDoors()[d2]=null; roomDir.setRawExit(d2,null); } } } } } rawImageName=null; cachedImageName=null; setArea(null); // this actually deletes the room from the cache map basePhyStats=(PhyStats)CMClass.getCommon("DefaultPhyStats"); phyStats=basePhyStats; for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) setRawExit(d,null); Arrays.fill(exits, null); Arrays.fill(doors, null); affects=null; behaviors=null; scripts=null; contents.setSize(0); inhabitants.setSize(0); gridParent=null; amDestroyed=true; } @Override public boolean amDestroyed() { return amDestroyed; } @Override public int getCombatTurnMobIndex() { return combatTurnMobIndex; } @Override public void setCombatTurnMobIndex(final int index) { combatTurnMobIndex = (short)index; } @Override public boolean isHere(Environmental E) { if(E instanceof Item) return isContent((Item)E); else if(E instanceof MOB) return isInhabitant((MOB)E); else if(E instanceof Exit) { for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--) { if(getRawExit(d)==E) return true; } } else if(E instanceof Room) { if (E == this) return true; if (E instanceof Room) { if (CMLib.map().getExtendedRoomID(this).equals(CMLib.map().getExtendedRoomID((Room) E))) return true; } return false; } else if(E instanceof Ability) return fetchEffect(E.ID())!=null; else if(E instanceof Behavior) return fetchBehavior(E.ID())!=null; return false; } @Override public MOB fetchRandomInhabitant() { if(inhabitants.isEmpty()) return null; return fetchInhabitant(CMLib.dice().roll(1,numInhabitants(),-1)); } @Override public MOB fetchInhabitant(String inhabitantID) { if(inhabitants.isEmpty()) return null; MOB mob=(MOB)CMLib.english().fetchEnvironmental(inhabitants,inhabitantID,true); if(mob==null) mob=(MOB)CMLib.english().fetchEnvironmental(inhabitants,inhabitantID, false); return mob; } private static final ReadOnlyVector<MOB> emptyMOBV=new ReadOnlyVector<MOB>(1); @SuppressWarnings("unchecked") @Override public List<MOB> fetchInhabitants(String inhabitantID) { if(inhabitants.isEmpty()) return emptyMOBV; @SuppressWarnings("rawtypes") List inhabs=CMLib.english().fetchEnvironmentals(inhabitants,inhabitantID,true); if(inhabs.isEmpty()) inhabs=CMLib.english().fetchEnvironmentals(inhabitants,inhabitantID, false); return inhabs; } @Override public void addInhabitant(MOB mob) { inhabitants.addElement(mob); } @Override public Enumeration<MOB> inhabitants() { return inhabitants.elements(); } @Override public int numInhabitants() { return inhabitants.size(); } @Override public int numPCInhabitants() { final Set<MOB> playerInhabitants=CMLib.players().getPlayersHere(this); if(playerInhabitants.isEmpty()) return 0; int num=0; for(final MOB M : playerInhabitants) { if((M!=null)&&(M.session()!=null)) num++; } return num; } @Override public boolean isInhabitant(MOB mob) { return inhabitants.contains(mob); } @Override public MOB fetchInhabitant(int i) { try { return inhabitants.elementAt(i); } catch(final java.lang.ArrayIndexOutOfBoundsException x) { } return null; } @Override public void eachInhabitant(final EachApplicable<MOB> applier) { final List<MOB> inhabitants=this.inhabitants; if((inhabitants!=null)&&(!inhabitants.isEmpty())) { try { for(int a=0;a<inhabitants.size();a++) { final MOB M=inhabitants.get(a); if(M!=null) applier.apply(M); } } catch(final ArrayIndexOutOfBoundsException e) { } } } @Override public void delInhabitant(MOB mob) { inhabitants.removeElement(mob); } @Override public void delAllInhabitants(boolean destroy) { try { for(int i=numInhabitants()-1;i>=0;i--) { final MOB M=fetchInhabitant(i); if(M!=null) { if(destroy || (M.location()==this)) M.setLocation(null); M.destroy(); } } inhabitants.clear(); } catch(final Exception e) { } } @Override public Exit fetchExit(String itemID) { int dir=Directions.getGoodDirectionCode(itemID); Exit E=null; if(dir >= 0) E=getExitInDir(dir); final List<Exit> exitList=Arrays.asList(exits); if(E==null) E=CMLib.english().fetchExit(exitList, itemID, true); if(E==null) E=CMLib.english().fetchExit(exitList, itemID, false); if(contents.isEmpty()) return E; if(E==null) E=CMLib.english().fetchExit(contents, itemID, true); if(E==null) E=CMLib.english().fetchExit(contents, itemID, false); return E; } @Override public Item findItem(String itemID) { if(contents.isEmpty()) return null; Item item=(Item)CMLib.english().fetchEnvironmental(contents,itemID,true); if(item==null) item=(Item)CMLib.english().fetchEnvironmental(contents,itemID,false); return item; } @Override public Enumeration<Item> items() { return contents.elements(); } @Override public Item findItem(Item goodLocation, String itemID) { if(contents.isEmpty()) return null; Item item=CMLib.english().fetchAvailableItem(contents,itemID,goodLocation,Wearable.FILTER_ANY,true); if(item==null) item=CMLib.english().fetchAvailableItem(contents,itemID,goodLocation,Wearable.FILTER_ANY,false); return item; } @Override public List<Item> findItems(Item goodLocation, String itemID) { if(contents.isEmpty()) return new Vector<Item>(1); List<Item> items=CMLib.english().fetchAvailableItems(contents,itemID,goodLocation,Wearable.FILTER_ANY,true); if(items.isEmpty()) items=CMLib.english().fetchAvailableItems(contents,itemID,goodLocation,Wearable.FILTER_ANY,false); return items; } @SuppressWarnings({ "rawtypes", "unchecked" }) @Override public List<Item> findItems(String itemID) { if(contents.isEmpty()) return new Vector<Item>(1); List items=CMLib.english().fetchEnvironmentals(contents,itemID,true); if(items.isEmpty()) items=CMLib.english().fetchEnvironmentals(contents,itemID, false); return items; } @Override public void addItem(Item item, Expire expire) { if(expire == null) { expire=Expire.Never; } int numMins = 0; switch(expire) { case Monster_EQ: addItem(item); numMins = CMProps.getIntVar(CMProps.Int.EXPIRE_MONSTER_EQ); break; case Monster_Body: insertItemUpTop(item); numMins = CMProps.getIntVar(CMProps.Int.EXPIRE_MONSTER_BODY); break; case Player_Body: insertItemUpTop(item); numMins = CMProps.getIntVar(CMProps.Int.EXPIRE_PLAYER_BODY); break; case Player_Drop: addItem(item); numMins = CMProps.getIntVar(CMProps.Int.EXPIRE_PLAYER_DROP); break; case Resource: addItem(item); numMins = CMProps.getIntVar(CMProps.Int.EXPIRE_RESOURCE); break; case Never: addItem(item); break; } if(numMins==0) item.setExpirationDate(0); else item.setExpirationDate(System.currentTimeMillis()+(numMins * TimeManager.MILI_MINUTE)); } protected void insertItemUpTop(Item item) { if((item!=null)&&(!item.amDestroyed())) { item.setOwner(this); if(!contents.isEmpty()) contents.add(0,item); else contents.add(item); item.recoverPhyStats(); } } @Override public void addItem(Item item) { if((item!=null)&&(!item.amDestroyed())) { item.setOwner(this); contents.addElement(item); item.recoverPhyStats(); } } @Override public void delItem(Item item) { contents.removeElement(item); item.recoverPhyStats(); } @Override public void delAllItems(boolean destroy) { if(destroy) { for(int i=numItems()-1;i>=0;i--) { final Item I=getItem(i); if(I!=null) { // since were deleting you AND all your peers, no need for Item to do it. I.setOwner(null); I.destroy(); } } } contents.clear(); } @Override public int numItems() { return contents.size(); } @Override public boolean isContent(Item item) { return contents.contains(item); } @Override public Item getItem(int i) { try { return contents.elementAt(i); } catch(final java.lang.ArrayIndexOutOfBoundsException x) { } return null; } @Override public void eachItem(final EachApplicable<Item> applier) { final List<Item> contents=this.contents; if((contents!=null)&&(!contents.isEmpty())) { try { for(int a=0;a<contents.size();a++) { final Item I=contents.get(a); if(I!=null) applier.apply(I); } } catch(final ArrayIndexOutOfBoundsException e) { } } } @Override public Item getRandomItem() { if(numItems()==0) return null; return getItem(CMLib.dice().roll(1,numItems(),-1)); } @Override public String getContextName(Environmental E) { if(E instanceof Exit) { for(int e=0;e<exits.length;e++) { if(exits[e]==E) { if((this instanceof BoardableShip)||(this.getArea() instanceof BoardableShip)) return Directions.getShipDirectionName(e); else return Directions.getDirectionName(e); } } return E.Name(); } else if(E instanceof MOB) { final String ctxName=CMLib.english().getContextName(inhabitants,E); if(ctxName!=null) return ctxName; } else if(E instanceof Item) { final String ctxName=CMLib.english().getContextName(contents,E); if(ctxName!=null) return ctxName; } else if(E!=null) return E.name(); return "nothing"; } @Override public PhysicalAgent fetchFromMOBRoomItemExit(MOB mob, Item goodLocation, String thingName, final Filterer<Environmental> filter) { PhysicalAgent found=null; String newThingName=CMLib.lang().preItemParser(thingName); if(newThingName!=null) thingName=newThingName; final boolean mineOnly=(mob!=null)&&(thingName.toUpperCase().trim().startsWith("MY ")); if(mineOnly) thingName=thingName.trim().substring(3).trim(); if((mob!=null)&&((filter!=Wearable.FILTER_WORNONLY))) { found=mob.fetchItem(goodLocation, new Filterer<Environmental>() { @Override public boolean passesFilter(Environmental obj) { return filter.passesFilter(obj) && Wearable.FILTER_UNWORNONLY.passesFilter(obj); } }, thingName); } if((found==null)&&(!mineOnly)) { found=(Exit)CMLib.english().fetchEnvironmental(Arrays.asList(exits),thingName,true); if(found==null) found=CMLib.english().fetchAvailableItem(contents,thingName,goodLocation,filter,true); if(found==null) found=(Exit)CMLib.english().fetchEnvironmental(Arrays.asList(exits),thingName,false); if(found==null) found=CMLib.english().fetchAvailableItem(contents,thingName,goodLocation,filter,false); if((found instanceof Item) // the smurfy well/gate exception &&(goodLocation==null) &&(found.displayText().length()==0) &&(thingName.indexOf('.')<0)) { PhysicalAgent visibleItem=null; visibleItem=(Exit)CMLib.english().fetchEnvironmental(Arrays.asList(exits),thingName,false); if(visibleItem==null) visibleItem=fetchFromMOBRoomItemExit(null,null,thingName+".2",filter); if(visibleItem!=null) found=visibleItem; } if((found!=null)&&(CMLib.flags().canBeSeenBy(found,mob))) return found; while((found!=null)&&(!CMLib.flags().canBeSeenBy(found,mob))) { newThingName=CMLib.english().bumpDotNumber(thingName); if(!newThingName.equals(thingName)) { thingName=newThingName; found=fetchFromRoomFavorItems(goodLocation, thingName); } else found=null; } } if((mob!=null)&&(found==null)&&((filter!=Wearable.FILTER_UNWORNONLY))) { found=mob.fetchItem(null, new Filterer<Environmental>() { @Override public boolean passesFilter(Environmental obj) { return filter.passesFilter(obj) && Wearable.FILTER_WORNONLY.passesFilter(obj); } }, thingName); } if(found==null) { newThingName=CMLib.lang().failedItemParser(thingName); if(newThingName!=null) return fetchFromMOBRoomItemExit(mob,goodLocation,newThingName,filter); } return found; } @Override public PhysicalAgent fetchFromRoomFavorItems(Item goodLocation, String thingName) { // def was Wearable.FILTER_UNWORNONLY; String newThingName=CMLib.lang().preItemParser(thingName); if(newThingName!=null) thingName=newThingName; PhysicalAgent found=null; final int[] contextNumber=new int[]{0}; if(!contents.isEmpty()) found=(PhysicalAgent)CMLib.english().fetchAvailable(contents,thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); if(found==null) found=(PhysicalAgent)CMLib.english().fetchAvailable(Arrays.asList(exits),thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); if((found==null)&&(!inhabitants.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(inhabitants,thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); contextNumber[0]=0; if((found==null)&&(!contents.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(contents,thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if(found==null) found=(PhysicalAgent)CMLib.english().fetchAvailable(Arrays.asList(exits),thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if((found==null)&&(!inhabitants.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(inhabitants,thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if((found!=null) // the smurfy well exception &&(found instanceof Item) &&(goodLocation==null) &&(found.displayText().length()==0) &&(thingName.indexOf('.')<0)) { final PhysicalAgent visibleItem=fetchFromRoomFavorItems(null,thingName+".2"); if(visibleItem!=null) found=visibleItem; } if(found==null) { newThingName=CMLib.lang().failedItemParser(thingName); if(newThingName!=null) return fetchFromRoomFavorItems(goodLocation,newThingName); } return found; } @Override public PhysicalAgent fetchFromRoomFavorExits(String thingName) { String newThingName=CMLib.lang().preItemParser(thingName); if(newThingName!=null) thingName=newThingName; final Item goodLocation=null; PhysicalAgent found=null; final int dirCode = Directions.getGoodDirectionCode(thingName); final int[] contextNumber=new int[]{0}; if(dirCode>=0) found=getRoomInDir(dirCode); if(found==null) found=(PhysicalAgent)CMLib.english().fetchAvailable(Arrays.asList(exits),thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); if((found==null)&&(!contents.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(contents,thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); if((found==null)&&(!inhabitants.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(inhabitants,thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); contextNumber[0]=0; if(found==null) found=(PhysicalAgent)CMLib.english().fetchAvailable(Arrays.asList(exits),thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if((found==null)&&(!contents.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(contents,thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if((found==null)&&(!inhabitants.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(inhabitants,thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if(found==null) { newThingName=CMLib.lang().failedItemParser(thingName); if(newThingName!=null) return fetchFromRoomFavorMOBs(goodLocation,newThingName); } return found; } @Override public PhysicalAgent fetchFromRoomFavorMOBs(Item goodLocation, String thingName) { // def was Wearable.FILTER_UNWORNONLY; String newThingName=CMLib.lang().preItemParser(thingName); if(newThingName!=null) thingName=newThingName; final int[] contextNumber = new int[]{0}; PhysicalAgent found=null; if(!inhabitants.isEmpty()) found=(PhysicalAgent)CMLib.english().fetchAvailable(inhabitants,thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); if((found==null)&&(!contents.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(contents,thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); if(found==null) found=(PhysicalAgent)CMLib.english().fetchAvailable(Arrays.asList(exits),thingName,goodLocation,Wearable.FILTER_ANY,true,contextNumber); contextNumber[0]=0; if((found==null)&&(!inhabitants.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(inhabitants,thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if((found==null)&&(!contents.isEmpty())) found=(PhysicalAgent)CMLib.english().fetchAvailable(contents,thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if(found==null) found=(PhysicalAgent)CMLib.english().fetchAvailable(Arrays.asList(exits),thingName,goodLocation,Wearable.FILTER_ANY,false,contextNumber); if(found==null) { newThingName=CMLib.lang().failedItemParser(thingName); if(newThingName!=null) return fetchFromRoomFavorMOBs(goodLocation,newThingName); } return found; } @Override public PhysicalAgent fetchFromMOBRoomFavorsItems(MOB mob, Item goodLocation, String thingName, Filterer<Environmental> filter) { return fetchFromMOBRoom(mob,goodLocation,thingName,filter,true); } @Override public PhysicalAgent fetchFromMOBRoomFavorsMOBs(MOB mob, Item goodLocation, String thingName, Filterer<Environmental> filter) { return fetchFromMOBRoom(mob,goodLocation,thingName,filter,false); } private PhysicalAgent fetchFromMOBRoom(MOB mob, Item goodLocation, String thingName, final Filterer<Environmental> filter, boolean favorItems) { PhysicalAgent found=null; String newThingName=CMLib.lang().preItemParser(thingName); if(newThingName!=null) thingName=newThingName; final boolean mineOnly=(mob!=null)&&(thingName.toUpperCase().trim().startsWith("MY ")); if(mineOnly) thingName=thingName.trim().substring(3).trim(); if((mob!=null)&&(favorItems)&&(filter!=Wearable.FILTER_WORNONLY)) { found=mob.fetchItem(goodLocation, new Filterer<Environmental>() { @Override public boolean passesFilter(Environmental obj) { return filter.passesFilter(obj) && Wearable.FILTER_UNWORNONLY.passesFilter(obj); } }, thingName); } if((found==null)&&(!mineOnly)) { if(favorItems) found=fetchFromRoomFavorItems(goodLocation, thingName); else found=fetchFromRoomFavorMOBs(goodLocation, thingName); if((found!=null)&&(CMLib.flags().canBeSeenBy(found,mob))) return found; while((found!=null)&&(!CMLib.flags().canBeSeenBy(found,mob))) { newThingName=CMLib.english().bumpDotNumber(thingName); if(!newThingName.equals(thingName)) { thingName=newThingName; if(favorItems) found=fetchFromRoomFavorItems(goodLocation, thingName); else found=fetchFromRoomFavorMOBs(goodLocation, thingName); } else found=null; } } if((mob!=null)&&(!favorItems)&&(filter!=Wearable.FILTER_WORNONLY)) { found=mob.fetchItem(goodLocation, new Filterer<Environmental>() { @Override public boolean passesFilter(Environmental obj) { return filter.passesFilter(obj) && Wearable.FILTER_UNWORNONLY.passesFilter(obj); } }, thingName); } if((mob!=null)&&(found==null)&&(filter!=Wearable.FILTER_UNWORNONLY)) { found=mob.fetchItem(null, new Filterer<Environmental>() { @Override public boolean passesFilter(Environmental obj) { return filter.passesFilter(obj) && Wearable.FILTER_WORNONLY.passesFilter(obj); } }, thingName); } if((mob!=null)&&(found==null)) found=mob.fetchItem(goodLocation,filter,thingName); if(found==null) { final boolean inShip=(this instanceof BoardableShip)||(this.getArea() instanceof BoardableShip); for(int d=0;d<exits.length;d++) { if((exits[d]!=null) &&(thingName.equalsIgnoreCase(inShip?Directions.getShipDirectionName(d):Directions.getDirectionName(d)))) return getExitInDir(d); } } if(found==null) { newThingName=CMLib.lang().failedItemParser(thingName); if(newThingName!=null) return fetchFromMOBRoom(mob,goodLocation,newThingName,filter,favorItems); } return found; } @Override public int pointsPerMove() { return getArea().getClimateObj().adjustMovement(phyStats().weight(),this); } protected int baseThirst() { return 1; } @Override public int thirstPerRound() { final int derivedClimate=getClimateType(); int adjustment=0; if(CMath.bset(derivedClimate, Places.CLIMASK_HOT)) adjustment+=1; if(CMath.bset(derivedClimate, Places.CLIMASK_WET)) adjustment-=1; if(CMath.bset(derivedClimate, Places.CLIMASK_DRY)) adjustment+=1; if(CMath.bset(derivedClimate, Places.CLIMASK_WINDY)) adjustment+=1; if(getArea().getClimateObj()!=null) return getArea().getClimateObj().adjustWaterConsumption(baseThirst()+adjustment,this); return 0; } @Override public int minRange() { return Integer.MIN_VALUE; } @Override public int maxRange() { return((domainType()&Room.INDOORS)>0)?1:10; } @Override public void addEffect(Ability to) { if(to==null) return; if(fetchEffect(to.ID())!=null) return; if(affects==null) affects=new SVector<Ability>(1); if(affects.contains(to)) return; affects.addElement(to); to.setAffectedOne(this); } @Override public void addNonUninvokableEffect(Ability to) { if(to==null) return; if(fetchEffect(to.ID())!=null) return; if(affects==null) affects=new SVector<Ability>(1); to.makeNonUninvokable(); to.makeLongLasting(); affects.addElement(to); to.setAffectedOne(this); } @Override public void delEffect(Ability to) { if(affects==null) return; if(affects.remove(to)) { to.setAffectedOne(null); if(affects.isEmpty()) affects=new SVector<Ability>(1); } } @Override public void eachEffect(final EachApplicable<Ability> applier) { final List<Ability> affects=this.affects; if((affects!=null)&&(!affects.isEmpty())) try { for(int a=0;a<affects.size();a++) { final Ability A=affects.get(a); if(A!=null) applier.apply(A); } } catch(final ArrayIndexOutOfBoundsException e) { } } @Override public void delAllEffects(boolean unInvoke) { if(affects==null) return; for(int a=numEffects()-1;a>=0;a--) { final Ability A=fetchEffect(a); if(A!=null) { if(unInvoke) A.unInvoke(); A.setAffectedOne(null); } } affects=new SVector<Ability>(1); } @Override public int numEffects() { if(affects==null) return 0; return affects.size(); } @SuppressWarnings("unchecked") @Override public Enumeration<Ability> effects() { return (affects==null)?EmptyEnumeration.INSTANCE:affects.elements(); } @Override public Ability fetchEffect(int index) { if(affects==null) return null; try { return affects.elementAt(index); } catch(final java.lang.ArrayIndexOutOfBoundsException x) { } return null; } @Override public Ability fetchEffect(String ID) { if(affects==null) return null; for(final Enumeration<Ability> a=effects();a.hasMoreElements();) { final Ability A=a.nextElement(); if(A.ID().equals(ID)) return A; } return null; } /** Manipulation of Behavior objects, which includes * movement, speech, spellcasting, etc, etc.*/ @Override public void addBehavior(Behavior to) { if(to==null) return; if(behaviors==null) behaviors=new SVector<Behavior>(1); for(final Behavior B : behaviors) if(B.ID().equals(to.ID())) return; if(behaviors.isEmpty()) CMLib.threads().startTickDown(this,Tickable.TICKID_ROOM_BEHAVIOR,1); to.startBehavior(this); behaviors.addElement(to); } @Override public void delBehavior(Behavior to) { if(behaviors==null) return; if(behaviors.remove(to)) { if(behaviors.isEmpty()) behaviors=new SVector<Behavior>(1); if(((behaviors==null)||(behaviors.isEmpty()))&&((scripts==null)||(scripts.isEmpty()))) CMLib.threads().deleteTick(this,Tickable.TICKID_ROOM_BEHAVIOR); } } @Override public void delAllBehaviors() { final boolean didSomething=(behaviors!=null)&&(!behaviors.isEmpty()); if(didSomething) behaviors.clear(); behaviors=null; if(didSomething && ((scripts==null)||(scripts.isEmpty()))) CMLib.threads().deleteTick(this,Tickable.TICKID_ROOM_BEHAVIOR); } @Override public int numBehaviors() { if(behaviors==null) return 0; return behaviors.size(); } @SuppressWarnings("unchecked") @Override public Enumeration<Behavior> behaviors() { return (behaviors==null)?EmptyEnumeration.INSTANCE:behaviors.elements(); } @Override public Behavior fetchBehavior(int index) { if(behaviors==null) return null; try { return behaviors.elementAt(index); } catch(final java.lang.ArrayIndexOutOfBoundsException x) { } return null; } @Override public Behavior fetchBehavior(String ID) { if(behaviors==null) return null; for(final Behavior B : behaviors) { if(B.ID().equalsIgnoreCase(ID)) return B; } return null; } @Override public void eachBehavior(final EachApplicable<Behavior> applier) { final List<Behavior> behaviors=this.behaviors; if((behaviors!=null)&&(!behaviors.isEmpty())) try { for(int a=0;a<behaviors.size();a++) { final Behavior B=behaviors.get(a); if(B!=null) applier.apply(B); } } catch(final ArrayIndexOutOfBoundsException e) { } } /** Manipulation of the scripts list */ @Override public void addScript(ScriptingEngine S) { if(scripts==null) scripts=new SVector<ScriptingEngine>(1); if(S==null) return; if(!scripts.contains(S)) { for(final Enumeration<ScriptingEngine> s2=scripts();s2.hasMoreElements();) { final ScriptingEngine S2=s2.nextElement(); if(S2.getScript().equalsIgnoreCase(S.getScript())) return; } if(scripts.isEmpty()) CMLib.threads().startTickDown(this,Tickable.TICKID_ROOM_BEHAVIOR,1); scripts.addElement(S); } } @Override public void delScript(ScriptingEngine S) { if(scripts!=null) { if(scripts.remove(S)) { if(scripts.isEmpty()) scripts=new SVector<ScriptingEngine>(1); if(((behaviors==null)||(behaviors.isEmpty()))&&((scripts==null)||(scripts.isEmpty()))) CMLib.threads().deleteTick(this,Tickable.TICKID_ROOM_BEHAVIOR); } } } @Override public void delAllScripts() { final boolean didSomething=(scripts!=null)&&(!scripts.isEmpty()); if(didSomething) scripts.clear(); scripts=null; if(didSomething && ((behaviors==null)||(behaviors.isEmpty()))) CMLib.threads().deleteTick(this,Tickable.TICKID_ROOM_BEHAVIOR); } @Override public int numScripts() { return (scripts==null)?0:scripts.size(); } @SuppressWarnings("unchecked") @Override public Enumeration<ScriptingEngine> scripts() { return (scripts==null)?EmptyEnumeration.INSTANCE:scripts.elements(); } @Override public ScriptingEngine fetchScript(int x) { try { return scripts.elementAt(x); } catch (final Exception e) { } return null; } @Override public void eachScript(final EachApplicable<ScriptingEngine> applier) { final List<ScriptingEngine> scripts=this.scripts; if((scripts!=null)&&(!scripts.isEmpty())) try { for(int a=0;a<scripts.size();a++) { final ScriptingEngine S=scripts.get(a); if(S!=null) applier.apply(S); } } catch(final ArrayIndexOutOfBoundsException e) { } } @Override public String L(final String str, final String... xs) { return CMLib.lang().fullSessionTranslation(str, xs); } @Override public int getSaveStatIndex() { return (xtraValues == null) ? getStatCodes().length : getStatCodes().length - xtraValues.length; } protected static final String[] STDCODES = { "CLASS", "DISPLAY", "DESCRIPTION", "TEXT", "AFFBEHAV", "IMAGE", "CLIMATE", "ATMOSPHERE" }; private static String[] codes = null; @Override public String[] getStatCodes() { if (codes == null) codes = CMProps.getStatCodesList(STDCODES, this); return codes; } @Override public boolean isStat(String code) { return CMParms.indexOf(getStatCodes(), code.toUpperCase().trim()) >= 0; } protected int getCodeNum(String code) { return CMParms.indexOf(codes, code.toUpperCase()); } @Override public String getStat(String code) { switch (getCodeNum(code)) { case 0: return CMClass.classID(this); case 1: return displayText(); case 2: return description(); case 3: return text(); case 4: return CMLib.coffeeMaker().getExtraEnvPropertiesStr(this); case 5: return rawImage(); case 6: return "" + getClimateTypeCode(); case 7: return "" + getAtmosphereCode(); default: return CMProps.getStatCodeExtensionValue(getStatCodes(), xtraValues, code); } } @Override public void setStat(String code, String val) { switch (getCodeNum(code)) { case 0: return; case 1: setDisplayText(val); break; case 2: setDescription(val); break; case 3: setMiscText(val); break; case 4: { delAllEffects(true); delAllBehaviors(); CMLib.coffeeMaker().setExtraEnvProperties(this, CMLib.xml().parseAllXML(val)); break; } case 5: setImage(val); break; case 6: setClimateType((CMath.s_int(val) < 0) ? -1 : CMath.s_parseBitIntExpression(Places.CLIMATE_DESCS, val)); break; case 7: { if (CMath.isMathExpression(val)) setAtmosphere(CMath.s_parseIntExpression(val)); final int matCode = RawMaterial.CODES.FIND_IgnoreCase(val); if (matCode >= 0) setAtmosphere(matCode); break; } default: CMProps.setStatCodeExtensionValue(getStatCodes(), xtraValues, code, val); break; } } @Override public boolean sameAs(Environmental E) { if (!(E instanceof StdRoom)) return false; final String[] codes = getStatCodes(); for (int i = 0; i < codes.length; i++) { if((!E.getStat(codes[i]).equals(getStat(codes[i]))) &&(!codes[i].equals("ATMOSPHERE"))) return false; } return true; } }