/
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.Behaviors;
import com.planet_ink.coffee_mud.core.interfaces.*;
import com.planet_ink.coffee_mud.core.*;
import com.planet_ink.coffee_mud.core.CMLib.Library;
import com.planet_ink.coffee_mud.core.collections.*;
import com.planet_ink.coffee_mud.Abilities.interfaces.*;
import com.planet_ink.coffee_mud.Areas.interfaces.*;
import com.planet_ink.coffee_mud.Behaviors.interfaces.*;
import com.planet_ink.coffee_mud.CharClasses.interfaces.*;
import com.planet_ink.coffee_mud.Commands.interfaces.*;
import com.planet_ink.coffee_mud.Common.interfaces.*;
import com.planet_ink.coffee_mud.Exits.interfaces.*;
import com.planet_ink.coffee_mud.Items.interfaces.*;
import com.planet_ink.coffee_mud.Libraries.interfaces.*;
import com.planet_ink.coffee_mud.Libraries.interfaces.XMLLibrary.XMLTag;
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 2006-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 Nanny extends StdBehavior
{
	@Override
	public String ID()
	{
		return "Nanny";
	}

	@Override
	protected int canImproveCode()
	{
		return Behavior.CAN_MOBS;
	}

	protected boolean	watchesBabies			= true;
	protected boolean	watchesChildren			= true;
	protected boolean	watchesMounts			= false;
	protected boolean	watchesWagons			= false;
	protected boolean	watchesCars				= false;
	protected boolean	watchesBoats			= false;
	protected boolean	watchesAirCars			= false;
	protected boolean	watchesMOBFollowers		= false;
	protected boolean	changedSinceLastSave	= false;
	protected String	place					= "Nursery";
	protected double	hourlyRate				= 1.0;

	protected List<DropOff>	dropOffs	= null;
	protected List<Payment>	payments	= new SVector<Payment>();
	protected DVector		sayLaters	= new DVector(2);
	// dynamic list of who belongs to what, before they leave
	// and get added to official drop-offs.
	protected List<DropOff> associations= new SVector<DropOff>();

	@Override
	public String accountForYourself()
	{
		return "caretaking and babysitting for a fee";
	}

	private static class DropOff
	{
		public String			mommyMname;
		public PhysicalAgent	baby;
		public long				dropOffTime;

		public DropOff(final String momMname, final PhysicalAgent baby, final long dropOff)
		{
			mommyMname = momMname;
			this.baby = baby;
			dropOffTime = dropOff;
		}
	}

	private static class Payment
	{
		public String	mommyMname;
		public double	paid;

		public Payment(final String mommyName, final double d)
		{
			this.mommyMname = mommyName;
			paid = d;
		}
	}

	public double getPaidBy(final MOB mob)
	{
		if(mob==null)
			return 0.0;
		double amt=0.0;
		for(final Payment P : payments)
		{
			if(P.mommyMname.equalsIgnoreCase(mob.Name()))
				amt+=P.paid;
		}
		return amt;
	}

	public boolean isDroppedOff(final PhysicalAgent P)
	{
		if(P==null)
			return false;
		for(final DropOff D : dropOffs)
		{
			if(D.baby==P)
				return true;
		}
		return false;
	}

	public boolean isAssociated(final PhysicalAgent P)
	{
		if(P==null)
			return false;
		for(final DropOff D : associations)
		{
			if((D.mommyMname.equalsIgnoreCase(P.Name()))
			||(D.baby==P))
				return true;
		}
		return false;
	}

	public void addPayment(final MOB mob,final double amt)
	{
		if(mob==null)
			return;
		for(final Payment P : payments)
		{
			if(P.mommyMname.equalsIgnoreCase(mob.Name()))
			{
				P.paid += amt;
				return;
			}
		}
		payments.add(new Payment(mob.Name(),amt));
	}

	public void clearTheSlate(final MOB mob)
	{
		if(mob==null)
			return;
		for(final Payment P : payments)
		{
			if(P.mommyMname.equalsIgnoreCase(mob.Name()))
				payments.remove(P);
		}
		if(dropOffs != null)
		{
			for(final DropOff D : dropOffs)
			{
				if(D.mommyMname.equalsIgnoreCase(mob.Name()))
				{
					boolean found=false;
					for(final DropOff A : associations)
						found = found || (A.mommyMname.equalsIgnoreCase(D.mommyMname));
					if(!found)
						associations.add(D);
					dropOffs.remove(D);
					changedSinceLastSave=true;
				}
			}
		}
	}

	public double getAllOwedBy(final MOB mob)
	{
		if(mob==null)
			return 0.0;
		final Room R=mob.location();
		if(R==null)
			return 0.0;
		final Area A=R.getArea();
		if(A==null)
			return 0.0;
		double amt=0.0;
		for(final DropOff D : dropOffs)
		{
			if(D.mommyMname.equalsIgnoreCase(mob.Name()))
			{
				long t=System.currentTimeMillis()-D.dropOffTime;
				t=Math.round(Math.ceil(CMath.div(t,CMProps.getMillisPerMudHour())));
				if(t>0)
					amt+=(t*hourlyRate);
			}
		}
		return amt;
	}

	public List<PhysicalAgent> getAllOwedFor(final MOB mob)
	{
		final List<PhysicalAgent> V=new Vector<PhysicalAgent>();
		if(mob!=null)
		{
			for(final DropOff D : dropOffs)
			{
				if(D.mommyMname.equalsIgnoreCase(mob.Name()))
					V.add(D.baby);
			}
		}
		return V;
	}

	public String getPronoun(final List<PhysicalAgent> V)
	{
		if(V.size()==0)
			return "your stuff";
		int babies=0;
		int friends=0;
		int objects=0;
		int mounts=0;
		for(int v=0;v<V.size();v++)
		{
			final PhysicalAgent E=V.get(v);
			if(CMLib.flags().isBaby(E)||CMLib.flags().isChild(E))
				babies++;
			else
			if(isMount(E))
				mounts++;
			else
			if(E instanceof Item)
				objects++;
			else
				friends++;
		}
		final Vector<String> pros=new Vector<String>();
		if(babies>0)
			pros.addElement("little one"+((babies>1)?"s":""));
		if(mounts>0)
			pros.addElement("mount"+((babies>1)?"s":""));
		if(friends>0)
			pros.addElement("friend"+((babies>1)?"s":""));
		if(objects>0)
			pros.addElement("thing"+((babies>1)?"s":""));
		final StringBuffer list=new StringBuffer("");
		for(int p=0;p<pros.size();p++)
		{
			list.append(pros.elementAt(p));
			if((pros.size()>1)&&(p==pros.size()-2))
				list.append(", and ");
			else
			if(pros.size()>1)
				list.append(", ");
		}
		return list.toString().trim();
	}

	public String getOwedFor(final String currency, final PhysicalAgent P)
	{
		for(final DropOff D : dropOffs)
		{
			if(D.baby==P)
			{
				long t=System.currentTimeMillis()-D.dropOffTime;
				t=Math.round(Math.floor(CMath.div(t,CMProps.getMillisPerMudHour())));
				if(t>0)
					return CMLib.beanCounter().abbreviatedPrice(currency, (t+hourlyRate))+" for watching "+P.name();
			}
		}
		return "";
	}

	public String getAllOwedBy(final String currency, final MOB mob)
	{
		if(mob==null)
			return "";
		final Room R=mob.location();
		if(R==null)
			return "";
		final Area A=R.getArea();
		if(A==null)
			return "";
		final StringBuffer owed=new StringBuffer("");
		for(final DropOff D : dropOffs)
		{
			if(D.mommyMname.equalsIgnoreCase(mob.Name()))
			{
				long t=System.currentTimeMillis()-D.dropOffTime;
				t=Math.round(Math.ceil(CMath.div(t,CMProps.getMillisPerMudHour())));
				if(t>0)
					owed.append(CMLib.beanCounter().abbreviatedPrice(currency, (t*hourlyRate))+" for "+D.baby.name()+", ");
			}
		}
		String s=owed.toString();
		if(s.endsWith(", "))
			s=s.substring(0,s.length()-2);
		return s;
	}

	public PhysicalAgent getDroppedOffObjIfAny(PhysicalAgent P)
	{
		if(P==null)
			return null;
		if(isDroppedOff(P))
			return P;

		if(P instanceof Container)
		{
			final List<Item> V=((Container)P).getDeepContents();
			Item I=null;
			for(int v=0;v<V.size();v++)
			{
				I=V.get(v);
				P=getDroppedOffObjIfAny(I);
				if(P!=null)
					return P;
			}
		}
		return null;
	}

	@Override
	public boolean okMessage(final Environmental host, final CMMsg msg)
	{
		if(!super.okMessage(host,msg))
			return false;
		if(dropOffs==null)
			return true;
		final int targMinor=msg.targetMinor();
		if((msg.target()==host)
		&&(targMinor==CMMsg.TYP_GIVE))
		{
			if(isDropOffable(msg.tool()))
			{
				final String pronoun=this.getPronoun(new XVector<PhysicalAgent>((PhysicalAgent)msg.tool()));
				msg.source().tell(msg.source(),host,msg.tool(),L("<T-NAME> won't accept <O-NAME>.  You should probably leave your @x1 here.",pronoun));
				return false;
			}
			if(msg.tool() instanceof Coins)
			{
				final Coins C=(Coins)msg.tool();
				final String myCurrency=CMLib.beanCounter().getCurrency(host);
				if(!C.getCurrency().equalsIgnoreCase(myCurrency))
				{
					if(host instanceof MOB)
						CMLib.commands().postSay((MOB)host,msg.source(),L("I'm don't accept @x1.  I can only accept @x2.",CMLib.beanCounter().getDenominationName(C.getCurrency(),C.getDenomination()),CMLib.beanCounter().getDenominationName(myCurrency)));
					else
						msg.source().tell(L("The @x1 doesn't accept @x2.  It only accepts @x3.",place,CMLib.beanCounter().getDenominationName(C.getCurrency(),C.getDenomination()),CMLib.beanCounter().getDenominationName(myCurrency)));
					return false;
				}
			}
		}
		else
		if(((targMinor==CMMsg.TYP_GET)
			||(targMinor==CMMsg.TYP_PULL)
			||(targMinor==CMMsg.TYP_PUSH)
			||(targMinor==CMMsg.TYP_CAST_SPELL))
		&&(msg.target() instanceof Item)
		&&(!msg.targetMajor(CMMsg.MASK_INTERMSG))
		&&(getDroppedOffObjIfAny((Item)msg.target()))!=null)
		{
			final PhysicalAgent obj=getDroppedOffObjIfAny((Item)msg.target());
			final String amt=getOwedFor(CMLib.beanCounter().getCurrency(host),obj);
			if((msg.source().location()==CMLib.map().roomLocation(host))
			&&(host instanceof MOB))
			{
				if(amt.length()>0)
					CMLib.commands().postSay((MOB)host,msg.source(),L("I'm afraid that @x1 fees of @x2 are still owed.",place,amt));
				else
					CMLib.commands().postSay((MOB)host,msg.source(),L("I'm afraid that @x1 fees are still owed on @x2.",place,obj.name()));
			}
			else
			if(amt.length()>0)
				msg.source().tell(L("You'll need to pay @x1 fees of @x2 first.",place,amt));
			else
				msg.source().tell(L("You'll need to pay your @x1 fees  for @x2 first.",place,obj.name()));
			return false;
		}
		else
		if((CMath.bset(msg.sourceMajor(),CMMsg.MASK_MALICIOUS)
			||CMath.bset(msg.targetMajor(),CMMsg.MASK_MALICIOUS))
		&&(msg.target() instanceof PhysicalAgent)
		&&((getDroppedOffObjIfAny((PhysicalAgent)msg.target())!=null)
				||(msg.target()==host)
				||(msg.target()==CMLib.map().roomLocation(host))))
		{
			if(msg.source()!=host)
			{
				if((host instanceof MOB)&&(msg.source().location()==CMLib.map().roomLocation(host)))
				{
					CMLib.commands().postSay((MOB)host,msg.source(),L("Not in my @x1 you dont!",place));
					final MOB victim=msg.source().getVictim();
					if(victim!=null)
						victim.makePeace(true);
					msg.source().makePeace(true);
				}
				else
					msg.source().tell(L("You can't do that here."));
			}
			return false;
		}
		else
		if((msg.sourceMinor()==CMMsg.TYP_LEAVE)
		&&(msg.target()==CMLib.map().roomLocation(host)))
		{
			final PhysicalAgent obj=getDroppedOffObjIfAny(msg.source());
			if(obj!=null)
			{
				if((msg.tool() instanceof Ability)
				&&(msg.source().location()!=null))
				{
					final Room R=CMLib.map().roomLocation(host);
					final boolean summon=CMath.bset(((Ability)msg.tool()).flags(),Ability.FLAG_SUMMONING);
					final boolean teleport=CMath.bset(((Ability)msg.tool()).flags(),Ability.FLAG_TRANSPORTING);
					final boolean shere=(msg.source().location()==R);
					if((!shere)&&((summon)||(teleport)))
					{
						if((msg.source().location()!=null)&&(msg.source().location()!=R))
							msg.source().location().showHappens(CMMsg.MSG_OK_VISUAL,L("Magical energy fizzles and is absorbed into the air!"));
						R.showHappens(CMMsg.MSG_OK_VISUAL,L("Magic energy fizzles and is absorbed into the air."));
						return false;
					}
				}
				else
				{
					final String amt=getOwedFor(CMLib.beanCounter().getCurrency(host),obj);
					if((msg.source().location()==CMLib.map().roomLocation(host))
					&&(host instanceof MOB))
					{
						if(amt.length()>0)
							CMLib.commands().postSay((MOB)host,msg.source(),L("I'm afraid that @x1 fees of @x2 are still owed on @x3.",place,amt,obj.name()));
						else
							CMLib.commands().postSay((MOB)host,msg.source(),L("I'm afraid that @x1 fees are still owed on @x2.",place,obj.name()));
					}
					else
					if(amt.length()>0)
						msg.source().tell(L("You'll need to pay @x1 fees of @x2 for @x3 first.",place,amt,obj.name()));
					else
						msg.source().tell(L("You'll need to pay your @x1 fees  for @x2 first.",place,obj.name()));
				}
				return false;
			}
		}
		return true;
	}

	public boolean isMount(final Environmental E)
	{
		if((E instanceof MOB)
		&&(E instanceof Rideable)
		&&((((Rideable)E).rideBasis()==Rideable.RIDEABLE_LAND)
				||(((Rideable)E).rideBasis()==Rideable.RIDEABLE_AIR)
				||(((Rideable)E).rideBasis()==Rideable.RIDEABLE_WATER)))
			return true;
		return false;
	}

	public boolean isDropOffable(final Environmental E)
	{
		if(E==null)
			return false;
		if((E instanceof MOB)&&(!((MOB)E).isMonster()))
			return false;
		if((watchesBabies)&&(CMLib.flags().isBaby(E)))
			return true;
		if((watchesChildren)&&(CMLib.flags().isChild(E))&&(!CMLib.flags().isBaby(E)))
			return true;
		if((watchesMounts)&&(isMount(E)))
			return true;
		if((watchesMOBFollowers)&&(E instanceof MOB)&&(!isMount(E))&&(!CMLib.flags().isChild(E))&&(!CMLib.flags().isBaby(E)))
			return true;
		if((this.watchesWagons)
		&&(E instanceof Rideable)
		&&(((Rideable)E).rideBasis()==Rideable.RIDEABLE_WAGON))
			return true;
		if((this.watchesCars)
		&&(E instanceof Item)
		&&(E instanceof Rideable)
		&&(((Rideable)E).rideBasis()==Rideable.RIDEABLE_LAND))
			return true;
		if((this.watchesBoats)
		&&(E instanceof Item)
		&&(E instanceof Rideable)
		&&(((Rideable)E).rideBasis()==Rideable.RIDEABLE_WATER))
			return true;
		if((this.watchesAirCars)
		&&(E instanceof Item)
		&&(E instanceof Rideable)
		&&(((Rideable)E).rideBasis()==Rideable.RIDEABLE_AIR))
			return true;
		return false;
	}

	public MOB ultimateFollowing(final Environmental E)
	{
		MOB ultimateFollowing=null;
		if(E instanceof MOB)
			ultimateFollowing=((MOB)E).amUltimatelyFollowing();
		return ultimateFollowing;
	}

	public MOB getMommyOf(final Physical P)
	{
		if((P instanceof Item)
		&&(((Item)P).owner() instanceof MOB)
		&&(!((MOB)((Item)P).owner()).isMonster()))
			return (MOB)((Item)P).owner();
		if((P instanceof MOB)
		&&(((MOB)P).amFollowing()!=null)
		&&(!((MOB)P).amFollowing().isMonster()))
			return ((MOB)P).amFollowing();
		if((P instanceof MOB)
		&&(ultimateFollowing(P)!=null)
		&&(!ultimateFollowing(P).isMonster()))
			return ultimateFollowing(P);
		if(P instanceof Rideable)
		{
			final Rideable R=(Rideable)P;
			Environmental E2=null;
			for(int r=0;r<R.numRiders();r++)
			{
				E2=R.fetchRider(r);
				if((E2 instanceof MOB)
				&&(!((MOB)E2).isMonster()))
					return (MOB)E2;
			}
		}
		if((P instanceof Rider)
		&&(((Rider)P).riding()!=null))
			return getMommyOf(((Rider)P).riding());
		return null;
	}

	protected void addAssociationsIfNecessaryRealTime(final Set<PhysicalAgent> H)
	{
		PhysicalAgent P=null;
		for(final Object o : H)
		{
			if(o instanceof PhysicalAgent)
			{
				P=(PhysicalAgent)o;
				if((P instanceof Rider)&&(((Rider)P).riding()!=null)&&(!H.contains(((Rider)P).riding())))
					H.add(P);
			}
		}
		for(final Object o : H)
		{
			if(o instanceof PhysicalAgent)
			{
				P=(PhysicalAgent)o;
				if((isDropOffable(P))&&(!isAssociated(P)))
				{
					final MOB source=getMommyOf(P);
					if(source!=null)
						associations.add(new DropOff(source.Name(),P,System.currentTimeMillis()));
				}
				if(P instanceof MOB)
				{
					final MOB mob=(MOB)P;
					for(int t=0;t<mob.numItems();t++)
					{
						final Item I=mob.getItem(t);
						if(isDropOffable(I)&&(!isAssociated(I)))
						{
							final MOB source=getMommyOf(I);
							if(source!=null)
								associations.add(new DropOff(source.Name(),I,System.currentTimeMillis()));
						}
					}
				}
			}
		}
	}

	public List<PhysicalAgent> myCurrentAssocs(final MOB mob)
	{
		final Vector<PhysicalAgent> V=new Vector<PhysicalAgent>();
		if(mob!=null)
		{
			for(final DropOff A : associations)
			{
				if(A.mommyMname.equalsIgnoreCase(mob.Name()))
					V.add(A.baby);
			}
		}
		return V;
	}

	@Override
	public void executeMsg(final Environmental host, final CMMsg msg)
	{
		super.executeMsg(host,msg);
		if(dropOffs==null)
			return;

		if((msg.targetMinor()==CMMsg.TYP_ENTER)
		&&(msg.target()==CMLib.map().roomLocation(host)))
		{
			final String currency=CMLib.beanCounter().getCurrency(host);
			final Set<PhysicalAgent> H=new HashSet<PhysicalAgent>();
			H.addAll(msg.source().getGroupMembers(new HashSet<MOB>()));
			H.addAll(msg.source().getRideBuddies(new HashSet<MOB>()));
			if(!H.contains(msg.source()))
				H.add(msg.source());
			HashSet<Environmental> H2 = null;
			int attempts=100000;
			do
			{
				H2 = new HashSet<Environmental>();
				H2.addAll(H);
				for (final Object element : H2)
				{
					final Environmental E = (Environmental)element;
					if(E instanceof Rideable)
					{
						final Rideable R = (Rideable)E;
						for(int r = 0; r<R.numRiders(); r++)
						{
							final Rider mR=R.fetchRider(r);
							if(!H.contains(mR))
								H.add(mR);
						}
					}
				}
			}
			while((H.size() > H2.size())&&((--attempts)>0))
				;

			addAssociationsIfNecessaryRealTime(H);
			final List<PhysicalAgent> myAssocs=myCurrentAssocs(msg.source());
			final StringBuffer list=new StringBuffer("");
			for(int m=0;m<myAssocs.size();m++)
			{
				list.append(myAssocs.get(m).name());
				if((myAssocs.size()>1)&&(m==myAssocs.size()-2))
					list.append(", and ");
				else
				if(myAssocs.size()>1)
					list.append(", ");
			}
			if(list.length()>0)
				sayLaters.addElement(msg.source(),"Welcome to my "+place+", "+msg.source().name()+"! You are welcome to leave " +
							list.toString()+" here under my care and protection.  Be aware that I charge "
							+CMLib.beanCounter().abbreviatedPrice(currency,hourlyRate)+" per hour, each.  " +
							"No payment is due until you return to fetch your "+getPronoun(myAssocs)+".");

			final double owed=getAllOwedBy(msg.source());
			final double paid=getPaidBy(msg.source());
			if(owed>0)
			{
				final List<PhysicalAgent> myStuff=getAllOwedFor(msg.source());
				final String pronoun=getPronoun(myStuff);
				sayLaters.addElement(msg.source(),"Welcome back, "+msg.source().name()+"! If are here for your "+pronoun
								+", the total bill is: "+getAllOwedBy(currency, msg.source())
								+" ("+CMLib.beanCounter().abbreviatedPrice(currency,owed-paid)+"). "
								+"You can just give me the money to settle the bill.");
			}
		}
		else
		if((msg.target()==host)
		&&(msg.targetMinor()==CMMsg.TYP_GIVE)
		&&(msg.tool() instanceof Coins))
		{
			addPayment(msg.source(),((Coins)msg.tool()).getTotalValue());
			final double owed=getAllOwedBy(msg.source());
			final double paid=getPaidBy(msg.source());
			final String currency=CMLib.beanCounter().getCurrency(host);
			if((paid>owed)&&(host instanceof MOB))
			{
				final double change=paid-owed;
				final Coins C=CMLib.beanCounter().makeBestCurrency(currency,change);
				final MOB source=msg.source();
				if((change>0.0)&&(C!=null))
				{
					// this message will actually end up triggering the hand-over.
					final CMMsg newMsg=CMClass.getMsg((MOB)host,source,C,CMMsg.MSG_SPEAK,L("^T<S-NAME> say(s) 'Heres your change.' to <T-NAMESELF>.^?"));
					C.setOwner((MOB)host);
					final long num=C.getNumberOfCoins();
					final String curr=C.getCurrency();
					final double denom=C.getDenomination();
					C.destroy();
					C.setNumberOfCoins(num);
					C.setCurrency(curr);
					C.setDenomination(denom);
					msg.addTrailerMsg(newMsg);
				}
				else
					CMLib.commands().postSay((MOB)host,source,L("Gee, thanks. :)"),true,false);
			}
			((Coins)msg.tool()).destroy();
			if(paid>=owed)
			{
				final List<PhysicalAgent> V=getAllOwedFor(msg.source());
				PhysicalAgent P=null;
				for(int v=0;v<V.size();v++)
				{
					P=V.get(v);
					if(P instanceof MOB)
					{
						CMLib.commands().postFollow((MOB)P,msg.source(),false);
						if(((MOB)P).isAttributeSet(MOB.Attrib.AUTOGUARD))
							((MOB)P).setAttribute(MOB.Attrib.AUTOGUARD,false);
						if(((MOB)P).amFollowing()!=msg.source())
						{
							CMLib.commands().postSay((MOB)host,msg.source(),L("Hmm, '@x1' doesn't seem ready to leave.  Now get along!",P.name()),true,false);
							msg.source().location().send((MOB)P,CMClass.getMsg((MOB)P,msg.source(),null,CMMsg.MSG_FOLLOW|CMMsg.MASK_ALWAYS,L("<S-NAME> follow(s) <T-NAMESELF>.")));
							if(((MOB)P).amFollowing()!=msg.source())
								((MOB)P).setFollowing(msg.source());
						}
					}
				}
				clearTheSlate(msg.source());
				sayLaters.addElement(msg.source(),"Thanks, come again!");
			}
			else
				sayLaters.addElement(msg.source(),"Thanks, but you still owe "+CMLib.beanCounter().abbreviatedPrice(currency,owed-paid)+".");
		}
		else
		if((msg.source()==host)
		&&(msg.targetMinor()==CMMsg.TYP_SPEAK)
		&&(msg.target() instanceof MOB)
		&&(msg.tool() instanceof Coins)
		&&(((Coins)msg.tool()).amDestroyed())
		&&(!msg.source().isMine(msg.tool()))
		&&(!((MOB)msg.target()).isMine(msg.tool())))
			CMLib.beanCounter().giveSomeoneMoney(msg.source(),(MOB)msg.target(),((Coins)msg.tool()).getTotalValue());
		else
		if((msg.source()!=host)
		&&(msg.targetMinor()==CMMsg.TYP_SPEAK)
		&&(msg.target() ==host)
		&&(CMSecurity.isAllowed(msg.source(), msg.source().location(), CMSecurity.SecFlag.CMDMOBS))
		&&(msg.sourceMessage()!=null))
		{
			final String say=CMStrings.getSayFromMessage(msg.sourceMessage());
			if((say != null)
			&&(say.toLowerCase().indexOf(I("report"))>=0))
			{
				final StringBuilder str=new StringBuilder(L("Drop Offs:\n\r"));
				for(final DropOff d : this.dropOffs)
					str.append(d.baby.Name()+" by "+d.mommyMname+" at "+CMLib.time().date2String(d.dropOffTime)).append("\n\r");
				str.append(L("\n\rAssociations:\n\r"));
				for(final DropOff d : this.associations)
					str.append(d.baby.Name()+" by "+d.mommyMname+" at "+CMLib.time().date2String(d.dropOffTime)).append("\n\r");
				str.append(L("\n\rPayments:\n\r"));
				for(final Payment p : this.payments)
					str.append(p.mommyMname+" paid "+CMLib.beanCounter().abbreviatedPrice((MOB)host, p.paid)).append("\n\r");
				CMLib.commands().postSay((MOB)host, str.toString());
			}
		}
	}

	protected int getNameCount(final List<String> V, final String name)
	{
		int index=0;
		for(int v=0;v<V.size();v++)
		{
			if(V.get(v).equals(name))
				index++;
		}
		return index;
	}

	@Override
	public String getParms()
	{
		final StringBuffer parms=new StringBuffer("");
		parms.append("RATE="+hourlyRate+" ");
		parms.append("NAME=\""+place+"\" ");
		parms.append("WATCHES=\"");
		if(watchesBabies)
			parms.append("Babies,");
		if(watchesChildren)
			parms.append("Children,");
		if(watchesMounts)
			parms.append("Mounts,");
		if(watchesWagons)
			parms.append("Wagons,");
		if(watchesCars)
			parms.append("Cars,");
		if(watchesBoats)
			parms.append("Boats,");
		if(watchesAirCars)
			parms.append("AirCars,");
		if(watchesMOBFollowers)
			parms.append("Followers,");
		parms.append("\"");
		if(dropOffs!=null)
		{
			parms.append(" |~| ");
			final Vector<String> oldNames=new Vector<String>();
			String eName=null;
			String oName=null;
			for(final DropOff D : dropOffs)
			{
				parms.append("<DROP>");

				eName=D.baby.Name();
				oName=D.mommyMname;
				if(oldNames.contains(eName))
					eName=getNameCount(oldNames,eName)+"."+eName;
				parms.append(CMLib.xml().convertXMLtoTag("ENAM",CMLib.xml().parseOutAngleBrackets(eName)));
				parms.append(CMLib.xml().convertXMLtoTag("ONAM",CMLib.xml().parseOutAngleBrackets(oName)));
				parms.append(CMLib.xml().convertXMLtoTag("TIME",D.dropOffTime));
				parms.append("</DROP>");
			}
		}
		return parms.toString().trim();
	}

	@Override
	public void setParms(final String parms)
	{
		super.setParms(parms);
		final int x=super.parms.indexOf("|~|");
		if(x>0)
			dropOffs=null;
		hourlyRate=CMParms.getParmDouble(parms,"RATE",2.0);
		place=CMParms.getParmStr(parms,"NAME","nursery");
		final List<String> watches=CMParms.parseCommas(CMParms.getParmStr(parms,"WATCHES","Babies,Children").toUpperCase(),true);
		String watch=null;
		watchesBabies=false;
		watchesChildren=false;
		watchesMounts=false;
		watchesWagons=false;
		watchesCars=false;
		watchesBoats=false;
		watchesAirCars=false;
		watchesMOBFollowers=false;
		for(int w=0;w<watches.size();w++)
		{
			watch=watches.get(w);
			if(watch.startsWith("BAB"))
				watchesBabies=true;
			if(watch.startsWith("CHI"))
				watchesChildren=true;
			if(watch.startsWith("MOU"))
				watchesMounts=true;
			if(watch.startsWith("WAG"))
				watchesWagons=true;
			if(watch.startsWith("CAR"))
				watchesCars=true;
			if(watch.startsWith("BOA"))
				watchesBoats=true;
			if(watch.startsWith("AIR"))
				watchesAirCars=true;
			if(watch.startsWith("FOL"))
				watchesMOBFollowers=true;
		}
	}

	@Override
	public boolean tick(final Tickable ticking, final int tickID)
	{
		if(!super.tick(ticking,tickID))
			return false;
		if((!CMProps.getBoolVar(CMProps.Bool.MUDSTARTED))
		||((!(ticking instanceof Environmental))))
			return true;
		if(dropOffs==null)
		{
			final int x=super.parms.indexOf("|~|");
			dropOffs=new SVector<DropOff>();
			if(x>0)
			{
				final String codes=super.parms.substring(x+3);
				parms=parms.substring(0,3);
				if(codes.trim().length()>0)
				{
					final List<XMLLibrary.XMLTag> V=CMLib.xml().parseAllXML(codes);
					XMLTag P=null;
					long time=0;
					String eName=null;
					PhysicalAgent PA=null;
					String oName=null;
					final Room R=CMLib.map().roomLocation((Environmental)ticking);
					if((V!=null)&&(R!=null))
					{
						for(int v=0;v<V.size();v++)
						{
							P=(V.get(v));
							if((P!=null)&&(P.contents()!=null)&&(P.contents().size()==3)&&(P.tag().equalsIgnoreCase("DROP")))
							{
								eName=CMLib.xml().restoreAngleBrackets(P.getValFromPieces("ENAM"));
								oName=CMLib.xml().restoreAngleBrackets(P.getValFromPieces("ONAM"));
								time=P.getLongFromPieces("TIME");
								if(!CMLib.players().playerExistsAllHosts(oName))
								{
									continue;
								}
								PA=R.fetchInhabitant(eName);
								if(PA==null)
									PA=R.findItem(eName);
								if(PA==null)
									Log.errOut("Nanny","Unable to find "+eName+" for "+oName+"!!");
								else
								if(!isDroppedOff(PA))
									dropOffs.add(new DropOff(oName,PA,time));
							}
							else
							if(P!=null)
								Log.errOut("Nanny","Unable to parse: "+codes+", specifically: "+P.value());
						}
					}
				}
			}
			changedSinceLastSave=false;
		}

		for(int s=sayLaters.size()-1;s>=0;s--)
		{
			if(ticking instanceof MOB)
				CMLib.commands().postSay((MOB)ticking,(MOB)sayLaters.elementAt(s,1),(String)sayLaters.elementAt(s,2));
			else
				((MOB)sayLaters.elementAt(s,1)).tell((String)sayLaters.elementAt(s,2));
			sayLaters.removeElementAt(s);
		}

		final Room R=CMLib.map().roomLocation((Environmental)ticking);
		if(R!=null)
		{
			for(final DropOff D : associations)
			{
				if(R.isHere(D.baby))
				{
					MOB hereM=R.fetchInhabitant("$"+D.mommyMname+"$");
					if(hereM==null)
						hereM=CMLib.players().getPlayerAllHosts(D.mommyMname);
					if((hereM ==null)
					||(!CMLib.flags().isInTheGame(hereM, true)))
					{
						if(!isDroppedOff(D.baby))
						{
							if((D.baby instanceof MOB)&&(((MOB)D.baby).amFollowing()!=null))
								((MOB)D.baby).setFollowing(null);
							D.dropOffTime=System.currentTimeMillis();
							dropOffs.add(D);
							associations.remove(D);
							changedSinceLastSave=true;
						}
					}
				}
				else
					associations.remove(D);
			}
		}
		if(!changedSinceLastSave)
		{
			for(final DropOff D : dropOffs)
			{
				if((D.baby instanceof MOB)
				&&(R!=null)
				&&(!R.isInhabitant((MOB)D.baby)))
				{
					dropOffs.remove(D);
					changedSinceLastSave=true;
				}
			}
		}
		for(final DropOff D : dropOffs)
		{
			if((D.baby instanceof Item)
			&&(R!=null)
			&&(!R.isContent((Item)D.baby)))
			{
				dropOffs.remove(D);
				changedSinceLastSave=true;
			}
		}

		if(changedSinceLastSave)
		{
			if(R!=null)
			{
				final List<MOB> mobsToSave=new ArrayList<MOB>();
				if(ticking instanceof MOB)
					mobsToSave.add((MOB)ticking);
				MOB M=null;
				for(int i=0;i<R.numInhabitants();i++)
				{
					M=R.fetchInhabitant(i);
					if((M!=null)
					&&(M.isSavable())
					&&(CMLib.flags().isMobile(M))
					&&(M.getStartRoom()==R)
					&&(!mobsToSave.contains(M)))
						mobsToSave.add(M);
				}
				for(final DropOff D : dropOffs)
				{
					if((D.baby instanceof MOB)
					&&(R.isInhabitant((MOB)D.baby))
					&&(!mobsToSave.contains(D.baby)))
						mobsToSave.add((MOB)D.baby);
				}
				CMLib.database().DBUpdateTheseMOBs(R,mobsToSave);
			}

			final List<Item> itemsToSave=new ArrayList<Item>();
			if(ticking instanceof Item)
				itemsToSave.add((Item)ticking);
			Item I=null;
			if(R!=null)
			{
				for(int i=0;i<R.numItems();i++)
				{
					I=R.getItem(i);
					if((I!=null)
					&&(I.isSavable())
					&&((!CMLib.flags().isGettable(I))||(I.displayText().length()==0))
					&&(!itemsToSave.contains(I)))
						itemsToSave.add(I);
				}
				for(final DropOff D : dropOffs)
				{
					if((D.baby instanceof Item)
					&&(R.isContent((Item)D.baby))
					&&(!itemsToSave.contains(D.baby)))
						itemsToSave.add((Item)D.baby);
				}
				CMLib.database().DBUpdateTheseItems(R,itemsToSave);
			}
			if(ticking instanceof Room)
				CMLib.database().DBUpdateRoom((Room)ticking);
			changedSinceLastSave=false;
		}

		if((dropOffs.size()>0)&&(ticking instanceof MOB)&&(CMLib.dice().rollPercentage()<10)&&(R!=null))
		{
			final MOB mob=(MOB)ticking;
			final PhysicalAgent PA=dropOffs.get(CMLib.dice().roll(1,dropOffs.size(),-1)).baby;
			if(CMLib.flags().isBaby(PA))
			{
				if(PA.fetchEffect("Soiled")!=null)
				{
					R.show(mob, PA, CMMsg.MSG_NOISYMOVEMENT,L("<S-NAME> change(s) <T-YOUPOSS> diaper."));
					PA.delEffect(PA.fetchEffect("Soiled"));
				}
				else
				if(CMLib.dice().rollPercentage()>50)
					R.show(mob, PA, CMMsg.MSG_NOISYMOVEMENT,L("<S-NAME> play(s) with <T-NAME>."));
				else
					R.show(mob, PA, CMMsg.MSG_NOISYMOVEMENT,L("<S-NAME> go(es) 'coochie-coochie coo' to <T-NAME>."));

			}
			else
			if(CMLib.flags().isChild(PA))
			{
				if(CMLib.dice().rollPercentage()>20)
					R.show(mob, PA, CMMsg.MSG_NOISYMOVEMENT,L("<S-NAME> play(s) with <T-NAME>."));
				else
				{
					R.show(mob, PA, CMMsg.MSG_NOISYMOVEMENT,L("<S-NAME> groom(s) <T-NAME>."));
					if(PA.fetchEffect("Soiled")!=null)
						PA.delEffect(PA.fetchEffect("Soiled"));
				}
			}
			else
			if(isMount(PA))
			{
				if(PA instanceof MOB)
				{
					if((!CMLib.flags().isAnimalIntelligence((MOB)PA))
					&&(CMLib.flags().canSpeak(mob)))
						R.show(mob, PA, CMMsg.MSG_NOISE,L("<S-NAME> speak(s) quietly with <T-NAME>."));
					else
					{
						final List<RawMaterial> V=((MOB)PA).charStats().getMyRace().myResources();
						boolean comb=false;
						if(V!=null)
						for(int v=0;v<V.size();v++)
						{
							if(((Item)V.get(v)).material()==RawMaterial.RESOURCE_FUR)
								comb=true;
						}
						if(comb)
							R.show(mob, PA, CMMsg.MSG_QUIETMOVEMENT,L("<S-NAME> groom(s) <T-NAME>."));
						else
							R.show(mob, PA, CMMsg.MSG_QUIETMOVEMENT,L("<S-NAME> pet(s) <T-NAME>."));
					}
				}
				else
					R.show(mob, PA, CMMsg.MSG_LOOK,L("<S-NAME> admire(s) <T-NAME>."));
			}
			else
			if(PA instanceof MOB)
			{
				if(CMLib.flags().isAnimalIntelligence((MOB)PA))
					R.show(mob, PA, CMMsg.MSG_QUIETMOVEMENT,L("<S-NAME> smile(s) and pet(s) <T-NAME>."));
				else
				if(CMLib.flags().canSpeak(mob))
					R.show(mob, PA, CMMsg.MSG_NOISE,L("<S-NAME> speak(s) quietly with <T-NAME>."));
			}
		}
		return true;
	}
}