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

import java.lang.ref.WeakReference;
import java.util.*;

/*
   Copyright 2008-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 StdThinInstance extends StdThinArea
{
	@Override
	public String ID()
	{
		return "StdThinInstance";
	}

	protected long	flags	= Area.FLAG_THIN | Area.FLAG_INSTANCE_PARENT;

	@Override
	public long flags()
	{
		return flags;
	}

	protected final List<AreaInstanceChild>	instanceChildren	= new SVector<AreaInstanceChild>();
	protected volatile int					instanceCounter		= 0;
	protected long							childCheckDown		= CMProps.getMillisPerMudHour() / CMProps.getTickMillis();
	protected WeakReference<Area>			parentArea			= null;

	protected String getStrippedRoomID(final String roomID)
	{
		final int x=roomID.indexOf('#');
		if(x<0)
			return null;
		return roomID.substring(x);
	}

	protected String convertToMyArea(final String roomID)
	{
		final String strippedID=getStrippedRoomID(roomID);
		if(strippedID==null)
			return null;
		return Name()+strippedID;
	}

	protected boolean qualifiesToBeParentArea(final Area parentA)
	{
		return (CMath.bset(parentA.flags(),Area.FLAG_INSTANCE_PARENT))
		&&(!CMath.bset(parentA.flags(),Area.FLAG_INSTANCE_CHILD));
	}

	protected Area getParentArea()
	{
		if((parentArea!=null)&&(parentArea.get()!=null))
			return parentArea.get();
		final int x=Name().indexOf('_');
		if(x<0)
			return null;
		if(!CMath.isNumber(Name().substring(0,x)))
			return null;
		final Area parentA = CMLib.map().getArea(Name().substring(x+1));
		if((parentA==null)
		||(!qualifiesToBeParentArea(parentA)))
			return null;
		parentArea=new WeakReference<Area>(parentA);
		return parentA;
	}

	@Override
	public Room getRoom(String roomID)
	{
		if(!CMath.bset(flags(),Area.FLAG_INSTANCE_CHILD))
			return super.getRoom(roomID);

		if(!isRoom(roomID))
			return null;
		final Room existingR=super.getRoom(roomID);
		if(((existingR==null)||(existingR.amDestroyed()))&&(roomID!=null))
		{
			final Area parentA=getParentArea();
			if(parentA==null)
				return null;

			if(roomID.toUpperCase().startsWith(Name().toUpperCase()+"#"))
				roomID=Name()+roomID.substring(Name().length()); // for case sensitive situations
			final Room parentR=parentA.getRoom(parentA.Name()+getStrippedRoomID(roomID));
			if(parentR==null)
				return null;

			final Room newR=CMLib.database().DBReadRoomObject(parentR.roomID(), false);
			if(newR==null)
				return null;
			final TreeMap<String,Room> V=new TreeMap<String,Room>();
			V.put(roomID,newR);
			CMLib.database().DBReadRoomExits(newR.roomID(), newR, false);
			CMLib.database().DBReadContent(newR.roomID(), newR, true);
			newR.clearSky();
			if(newR instanceof GridLocale)
				((GridLocale)newR).clearGrid(null);
			for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
				newR.rawDoors()[d]=null;
			newR.setRoomID(roomID);
			newR.setArea(this);
			addProperRoom(newR);

			synchronized(("SYNC"+roomID).intern())
			{
				final Room[] newDoors = newR.rawDoors();
				for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
				{
					final Room dirR=parentR.rawDoors()[d];
					if(dirR!=null)
					{
						final String myRID=dirR.roomID();
						if((myRID!=null)
						&&(myRID.length()>0)
						&&(dirR.getArea()==parentA))
						{
							final String localDirRID=convertToMyArea(myRID);
							final Room localDirR=getProperRoom(localDirRID);
							if(localDirR!=null)
								newDoors[d]=localDirR;
							else
							if(localDirRID==null)
								Log.errOut("StdThinInstance","Error in linked room ID "+parentR.roomID()+", dir="+d);
							else
							{
								newDoors[d]=CMClass.getLocale("ThinRoom");
								newDoors[d].setRoomID(localDirRID);
								newDoors[d].setArea(this);
							}
						}
						else
							newDoors[d]=dirR;
					}
				}
			}
			//for(final Enumeration<MOB> e=newR.inhabitants();e.hasMoreElements();) // is this really necc?
			//	e.nextElement().bringToLife(newR,true);
			newR.startItemRejuv();
			fillInAreaRoom(newR);
			newR.setExpirationDate(System.currentTimeMillis()+WorldMap.ROOM_EXPIRATION_MILLIS);
			final MOB mob=CMClass.getFactoryMOB("the wind",1,newR);
			try
			{
				newR.executeMsg(mob, CMClass.getMsg(mob, newR, CMMsg.MSG_NEWROOM, null));
			}
			finally
			{
				mob.destroy();
			}
			return newR;
		}
		return existingR;
	}

	protected static boolean isInThisArea(final Area childA, final MOB mob)
	{
		if(mob != null)
		{
			final Room R=mob.location();
			if(R!=null)
			{
				final Area A=R.getArea();
				if(R.getArea()==childA)
					return true;
				if(A instanceof BoardableShip)
				{
					final Room R2=CMLib.map().roomLocation(((BoardableShip)A).getShipItem());
					if((R2!=null)
					&&(R2.getArea()==childA))
						return true;
				}
			}
		}
		return false;
	}

	protected Set<Physical> getProtectedSet(final Area childA, final List<WeakReference<MOB>> registeredMobList)
	{
		final Set<Physical> protectTheseList=new HashSet<Physical>();
		for(final WeakReference<MOB> wmob : registeredMobList)
		{
			final MOB M=wmob.get();
			if(M!=null)
			{
				final Room R=M.location();
				if(CMLib.flags().isInTheGame(M,true)
				&&(R!=null)
				&&(R.getArea()==childA))
					protectTheseList.add(M);
			}
		}
		for(final Enumeration<MOB> m = CMLib.players().players();m.hasMoreElements();)
		{
			final MOB M=m.nextElement();
			if(M!=null)
			{
				final Room R=M.location();
				if((R!=null)
				&&(R.getArea()==childA))
					protectTheseList.add(M);
			}
		}
		for(final Enumeration<Room> r=childA.getFilledCompleteMap();r.hasMoreElements();)
		{
			final Room R=r.nextElement();
			if(R!=null)
			{
				for(final Enumeration<Item> i=R.items();i.hasMoreElements();)
				{
					final Item I=i.nextElement();
					if((I instanceof DeadBody)
					&&(((DeadBody)I).isPlayerCorpse()))
						protectTheseList.add(I);
				}
			}
		}
		for(final Enumeration<BoardableShip> s = CMLib.map().ships();s.hasMoreElements();)
		{
			final BoardableShip B=s.nextElement();
			if(B!=null)
			{
				final Item I=B.getShipItem();
				if((I!=null)
				&&(I.baseGoldValue()>2)
				&&(I.owner() instanceof Room))
				{
					final Room R=(Room)I.owner();
					if((R!=null)
					&&(R.getArea()==childA)
					&&(B instanceof PrivateProperty)
					&&(((PrivateProperty)B).getOwnerName()!=null)
					&&(((PrivateProperty)B).getOwnerName().length()>0))
					{
						protectTheseList.add(I);
					}
				}
			}
		}
		final List<MOB> addFollowers=new ArrayList<MOB>(0);
		for(final Physical p : protectTheseList)
		{
			if(p instanceof MOB)
			{
				final MOB pmob=(MOB)p;
				final Set<MOB> grp=pmob.getGroupMembers(new HashSet<MOB>());
				for(final MOB M : grp)
				{
					if(pmob!=M)
					{
						final Room R=M.location();
						if((R!=null)
						&&(R.getArea()==childA))
							addFollowers.add(M);
					}
				}
			}
		}
		protectTheseList.addAll(addFollowers);
		return protectTheseList;
	}

	protected static class EmptyFilters implements RFilters
	{
		@Override
		public boolean isFilteredOut(final Room hostR, final Room R, final Exit E, final int dir)
		{
			return false;
		}

		@Override
		public RFilters plus(final RFilter filter)
		{
			return this;
		}

		@Override
		public RFilters minus(final RFilter filter)
		{
			return this;
		}

		@Override
		public RFilters copyOf()
		{
			return this;
		}
	}

	protected static class AllWaterFilters extends EmptyFilters
	{
		@Override
		public boolean isFilteredOut(final Room hostR, final Room R, final Exit E, final int dir)
		{
			return (!(CMLib.flags().isWateryRoom(hostR)||CMLib.flags().isWateryRoom(R)));
		}
	}

	protected static class OutOfAreaFilter implements RFilter
	{
		protected final Area A1;

		public OutOfAreaFilter(final Area A1)
		{
			this.A1=A1;
		}

		@Override
		public boolean isFilteredOut(final Room hostR, final Room R, final Exit E, final int dir)
		{
			return (R.getArea()==A1);
		}
	}

	protected void flushInstance(final AreaInstanceChild child)
	{
		final Area childA=child.A;
		final Set<Physical> protectedList = getProtectedSet(childA, child.mobs);
		childA.setAreaState(State.STOPPED); // kill motion!
		Room returnBoatsToR = null;
		for(final Physical P : protectedList)
		{
			if(P instanceof MOB)
			{
				final MOB wmob=(MOB)P;
				final Room R=wmob.location();
				if((R!=null)
				&&(R.getArea()==childA))
				{
					final Room startRoom=wmob.getStartRoom();
					if(startRoom != null)
					{
						if(R.isInhabitant(wmob))
							startRoom.bringMobHere(wmob, true);
						if(wmob.location()!=startRoom)
							wmob.setLocation(startRoom);
					}
				}
			}
			else
			if(P instanceof Item)
			{
				final Item I = (Item)P;
				Room startRoom = returnBoatsToR;
				Room R=CMLib.map().roomLocation(I);
				if(R!=null)
				{
					if((startRoom == null)
					&&(I instanceof SailingShip))
					{
						final Room R1=CMLib.tracking().getRadiantRoomTarget(R, new AllWaterFilters(), new OutOfAreaFilter(childA));
						if(R1!=null)
						{
							returnBoatsToR=R1;
							startRoom=R1;
						}
					}
					if(startRoom == null)
					{
						final Room R1=CMLib.tracking().getRadiantRoomTarget(R, new EmptyFilters(), new OutOfAreaFilter(childA));
						if(R1!=null)
							startRoom=R1;
					}
					if(startRoom != null)
					{
						if(R.isHere(I))
							startRoom.moveItemTo(I);
						R=CMLib.map().roomLocation(I);
						if(R!=startRoom)
							I.setOwner(startRoom);
					}
				}
			}
		}
		final MOB mob=CMClass.getFactoryMOB();
		try
		{
			final CMMsg msg=CMClass.getMsg(mob,null,null,CMMsg.MSG_EXPIRE,null);
			final LinkedList<Room> propRooms = new LinkedList<Room>();
			for(final Enumeration<Room> r=childA.getFilledProperMap();r.hasMoreElements();)
				propRooms.add(r.nextElement());
			// msgs only, handles saves and stuff, but ignores grid rooms!!
			for(final Iterator<Room> r=propRooms.iterator();r.hasNext();)
			{
				final Room R=r.next();
				try
				{
					try
					{
						R.clearSky();
						msg.setTarget(R);
						R.executeMsg(mob,msg);
						try
						{
							CMLib.map().emptyRoom(R, null, true);
						}
						catch(final Exception e)
						{
							Log.errOut(e);
						}
					}
					catch(final Exception e)
					{
						Log.errOut(e);
					}
					R.destroy(); // destroys the mobs and items.  the Deadly Thing.
				}
				catch(final Exception e)
				{
					Log.errOut(e);
				}
			}
			propRooms.clear();
			CMLib.map().delArea(childA);
			childA.destroy();
		}
		finally
		{
			mob.destroy();
		}
	}

	@Override
	public void destroy()
	{
		super.destroy();
		if(this.doesManageChildAreas())
		{
			final List<AreaInstanceChild> cs = new ArrayList<AreaInstanceChild>(instanceChildren.size());
			synchronized(instanceChildren)
			{
				for(int i=instanceChildren.size()-1;i>=0;i--)
					cs.add(instanceChildren.get(i));
				instanceChildren.clear();
			}
			for(final AreaInstanceChild I : cs)
			{
				final Area A=I.A;
				try
				{
					flushInstance(I);
				}
				catch(final Exception e)
				{
					Log.errOut(e);
				}
				CMLib.map().delArea(A);
				A.destroy();
			}
		}
	}

	protected boolean doesManageChildAreas()
	{
		if(CMath.bset(flags(),Area.FLAG_INSTANCE_CHILD))
			return false;
		return true;
	}

	@Override
	public boolean tick(final Tickable ticking, final int tickID)
	{
		if(!super.tick(ticking, tickID))
			return false;
		if(!doesManageChildAreas())
			return true;
		if((--childCheckDown)<=0)
		{
			childCheckDown=CMProps.getMillisPerMudHour()/CMProps.getTickMillis();
			final List<AreaInstanceChild> flushThese = new ArrayList<AreaInstanceChild>(1);
			synchronized(instanceChildren)
			{
				for(int i=instanceChildren.size()-1;i>=0;i--)
				{
					final Area childA=instanceChildren.get(i).A;
					if(childA.getAreaState() != Area.State.ACTIVE) // this is the one and only criteria -- if its not active, flush it.
					{
						final AreaInstanceChild child = instanceChildren.remove(i);
						flushThese.add(child);
					}
				}
			}
			for(final AreaInstanceChild inst : flushThese)
				flushInstance(inst);
		}
		return true;
	}

	@Override
	public void executeMsg(final Environmental myHost, final CMMsg msg)
	{
		super.executeMsg(myHost, msg);
		if(CMath.bset(flags(),Area.FLAG_INSTANCE_CHILD))
		{
			if((msg.sourceMinor()==CMMsg.TYP_SPEAK)
			&&(msg.sourceMessage()!=null)
			&&((msg.sourceMajor()&CMMsg.MASK_MAGIC)==0))
			{
				final String said=CMStrings.getSayFromMessage(msg.sourceMessage());
				if("RESET INSTANCE".equalsIgnoreCase(said))
				{
					Room returnToRoom=null;
					final Room thisRoom=msg.source().location();
					if(thisRoom.getArea()==this)
					{
						for(int d=0;d<Directions.NUM_DIRECTIONS();d++)
						{
							final Room R=thisRoom.getRoomInDir(d);
							if((R!=null)&&(R.getArea()!=null)&&(R.getArea()!=this))
								returnToRoom=R;
						}
					}
					if(returnToRoom==null)
					{
						msg.addTrailerMsg(CMClass.getMsg(msg.source(),null,null,CMMsg.MSG_OK_ACTION,CMMsg.NO_EFFECT,CMMsg.NO_EFFECT, L("You must be at an entrance to reset the area.")));
						return;
					}
					final Area A=this.getParentArea();
					if(A instanceof StdThinInstance)
					{
						final StdThinInstance parentA=(StdThinInstance)A;
						final List<AreaInstanceChild> resetThese = new LinkedList<AreaInstanceChild>();
						synchronized(instanceChildren)
						{
							for(int i=0;i<parentA.instanceChildren.size();i++)
							{
								if(parentA.instanceChildren.get(i).A==this)
								{
									final AreaInstanceChild child = parentA.instanceChildren.remove(i);
									resetThese.add(child);
								}
							}
						}
						for(final AreaInstanceChild child : resetThese)
						{
							final List<WeakReference<MOB>> V=child.mobs;
							for(final WeakReference<MOB> wM : V)
							{
								final MOB M=wM.get();
								if((M!=null)
								&&CMLib.flags().isInTheGame(M,true)
								&&(M.location()!=null)
								&&(M.location()!=returnToRoom)
								&&(M.location().getArea()==this))
								{
									returnToRoom.bringMobHere(M, true);
									CMLib.commands().postLook(M, true);
								}
							}
							setAreaState(Area.State.PASSIVE);
							flushInstance(child);
							msg.addTrailerMsg(CMClass.getMsg(msg.source(),CMMsg.MSG_OK_ACTION,L("The instance has been reset.")));
						}
					}
					msg.addTrailerMsg(CMClass.getMsg(msg.source(),CMMsg.MSG_OK_ACTION,L("The instance failed to reset.")));
				}
			}
			else
			if((msg.sourceMinor()==CMMsg.TYP_QUIT)&&(CMLib.map().isHere(msg.source(), this)))
			{
				final MOB mob = msg.source();
				CMLib.tracking().forceRecall(mob, true);
			}
		}
	}

	@Override
	public int[] getAreaIStats()
	{
		if(!CMProps.getBoolVar(CMProps.Bool.MUDSTARTED))
			return emptyStats;
		final Area parentArea=getParentArea();
		final String areaName = (parentArea==null)?Name():parentArea.Name();
		int[] statData=(int[])Resources.getResource("STATS_"+areaName.toUpperCase());
		if(statData!=null)
			return statData;
		synchronized(("STATS_"+areaName).intern())
		{
			if(parentArea==null)
			{
				statData=super.getAreaIStats();
				if(statData==emptyStats)
				{
					final Iterator<AreaInstanceChild> childE=instanceChildren.iterator();
					int ct=0;
					if(childE.hasNext())
					{
						statData=new int[Area.Stats.values().length];
						for(;childE.hasNext();)
						{
							final int[] theseStats=childE.next().A.getAreaIStats();
							if(theseStats != emptyStats)
							{
								ct++;
								for(int i=0;i<theseStats.length;i++)
									statData[i]+=theseStats[i];
							}
						}
					}
					if(ct==0)
						return emptyStats;
					for(int i=0;i<statData.length;i++)
						statData[i]=statData[i]/ct;
				}
				Resources.removeResource("HELP_"+areaName.toUpperCase());
				Resources.submitResource("STATS_"+areaName.toUpperCase(),statData);
			}
			else
			{
				return super.getAreaIStats();
			}
		}
		return statData;
	}

	protected boolean doesManageMobLists()
	{
		return CMath.bset(flags(),Area.FLAG_INSTANCE_PARENT);
	}

	protected Area createRedirectArea(final List<MOB> mobs)
	{
		final StdThinInstance newA=(StdThinInstance)this.copyOf();
		newA.properRooms=new STreeMap<String, Room>(new Area.RoomIDComparator());
		newA.parentArea=null;
		newA.properRoomIDSet = null;
		newA.metroRoomIDSet = null;
		newA.blurbFlags=new STreeMap<String,String>();
		newA.setName((++instanceCounter)+"_"+Name());
		newA.flags |= Area.FLAG_INSTANCE_CHILD;
		for(final Enumeration<String> e=getProperRoomnumbers().getRoomIDs();e.hasMoreElements();)
			newA.addProperRoomnumber(newA.convertToMyArea(e.nextElement()));
		CMLib.map().addArea(newA);
		newA.setAreaState(Area.State.ACTIVE); // starts ticking
		final List<WeakReference<MOB>> newMobList = new SVector<WeakReference<MOB>>(5);
		for(final MOB mob : mobs)
			newMobList.add(new WeakReference<MOB>(mob));
		final AreaInstanceChild child = new AreaInstanceChild(newA,newMobList);
		instanceChildren.add(child);
		return newA;
	}

	@Override
	public boolean okMessage(final Environmental myHost, final CMMsg msg)
	{
		if(!super.okMessage(myHost, msg))
			return false;
		if(!this.doesManageChildAreas())
			return true;
		if(CMath.bset(flags(),Area.FLAG_INSTANCE_PARENT))
			setAreaState(Area.State.PASSIVE);
		if((msg.sourceMinor()==CMMsg.TYP_ENTER)
		&&(msg.target() instanceof Room)
		&&(doesManageMobLists())
		&&(isRoom((Room)msg.target()))
		&&((!CMSecurity.isAllowed(msg.source(),(Room)msg.target(),CMSecurity.SecFlag.CMDAREAS))
			||(!msg.source().isAttributeSet(MOB.Attrib.SYSOPMSGS)))
		&&(((msg.source().getStartRoom()==null)||(msg.source().getStartRoom().getArea()!=this))))
		{
			int myDex=-1;
			Area redirectA = null;
			final Set<MOB> grp = msg.source().getGroupMembers(new HashSet<MOB>());
			if(msg.source().isMonster()
			&&(msg.source().riding() instanceof BoardableShip))
			{
				final List<MOB> mobSet=new LinkedList<MOB>();
				boolean playerFound=false;
				final Area subA=((BoardableShip)msg.source().riding()).getShipArea();
				for(final Enumeration<Room> r=subA.getProperMap();r.hasMoreElements();)
				{
					final Room R=r.nextElement();
					if(R!=null)
					{
						for(final Enumeration<MOB> m=R.inhabitants();m.hasMoreElements();)
						{
							final MOB M=m.nextElement();
							if(M!=null)
							{
								mobSet.add(M);
								playerFound = playerFound || (!M.isMonster());
							}
						}
					}
				}
				if(playerFound)
				{
					grp.addAll(mobSet);
				}
			}
			synchronized(instanceChildren)
			{
				for(int i=0;i<instanceChildren.size();i++)
				{
					final List<WeakReference<MOB>> V=instanceChildren.get(i).mobs;
					for (final WeakReference<MOB> weakReference : V)
					{
						if(msg.source() == weakReference.get())
						{
							myDex=i;
							break;
						}
					}
				}
				for(int i=0;i<instanceChildren.size();i++)
				{
					if(i!=myDex)
					{
						final List<WeakReference<MOB>> V=instanceChildren.get(i).mobs;
						for(int v=V.size()-1;v>=0;v--)
						{
							final WeakReference<MOB> wmob=V.get(v);
							if(wmob==null)
								continue;
							final MOB M=wmob.get();
							if(grp.contains(M))
							{
								if(myDex<0)
								{
									myDex=i;
									break;
								}
								else
								if((CMLib.flags().isInTheGame(M,true))
								&&(M.location().getArea()!=instanceChildren.get(i).A))
								{
									V.remove(wmob);
									instanceChildren.get(myDex).mobs.add(new WeakReference<MOB>(M));
								}
							}
						}
					}
				}
				if(myDex>=0)
					redirectA=instanceChildren.get(myDex).A;
			}
			if(myDex<0)
				redirectA = createRedirectArea(new XVector<MOB>(grp));
			if(redirectA instanceof StdThinInstance)
			{
				Room R=redirectA.getRoom(((StdThinInstance)redirectA).convertToMyArea(CMLib.map().getExtendedRoomID((Room)msg.target())));
				int tries=1000;
				while((R==null)&&((--tries)>0)&&(redirectA.numberOfProperIDedRooms()>0))
				{
					R=redirectA.getRandomProperRoom();
					if(R!=null)
					{
						msg.setTarget(R);
						if((!CMLib.flags().canAccess(msg.source(),R))
						||(CMLib.law().getLandTitle(R)!=null)
						||(!R.okMessage(msg.source(), msg)))
							R=null;
					}
				}
				if(R!=null)
					msg.setTarget(R);
			}
		}
		return true;
	}
}