/
com/planet_ink/coffee_mud/Abilities/
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/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/Spells/
com/planet_ink/coffee_mud/Abilities/Thief/
com/planet_ink/coffee_mud/Abilities/Traps/
com/planet_ink/coffee_mud/Areas/interfaces/
com/planet_ink/coffee_mud/Behaviors/
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/Exits/interfaces/
com/planet_ink/coffee_mud/Items/Armor/
com/planet_ink/coffee_mud/Items/Basic/
com/planet_ink/coffee_mud/Items/MiscMagic/
com/planet_ink/coffee_mud/Items/Software/
com/planet_ink/coffee_mud/Items/Weapons/
com/planet_ink/coffee_mud/Libraries/interfaces/
com/planet_ink/coffee_mud/Locales/
com/planet_ink/coffee_mud/Locales/interfaces/
com/planet_ink/coffee_mud/MOBS/
com/planet_ink/coffee_mud/MOBS/interfaces/
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/application/
com/planet_ink/coffee_mud/core/smtp/
com/planet_ink/siplet/applet/
lib/
resources/examples/
resources/fakedb/
resources/quests/delivery/
resources/quests/diseased/
resources/quests/drowning/
resources/quests/gobwar/
resources/quests/holidays/
resources/quests/robbed/
resources/quests/smurfocide/
resources/quests/stolen/
resources/quests/templates/
resources/quests/treasurehunt/
resources/quests/vengeance/
web/
web/admin.templates/
web/admin/images/
web/pub.templates/
web/pub/images/mxp/
web/pub/sounds/
package com.planet_ink.coffee_mud.MOBS;
import com.planet_ink.coffee_mud.core.interfaces.*;
import com.planet_ink.coffee_mud.core.*;
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.Locales.interfaces.*;
import com.planet_ink.coffee_mud.MOBS.interfaces.*;
import com.planet_ink.coffee_mud.Races.interfaces.*;
import java.util.*;



/* 
   Copyright 2000-2006 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 StdPostman extends StdShopKeeper implements PostOffice
{
    public String ID(){return "StdPostman";}

    protected double minimumPostage=1.0;
    protected double postagePerPound=1.0;
    protected double holdFeePerPound=1.0;
    protected double feeForNewBox=50.0;
    protected int maxMudMonthsHeld=12;
    protected static Hashtable postalTimes=new Hashtable();
    private long postalWaitTime=-1;

    public StdPostman()
    {
        super();
        Username="a postman";
        setDescription("He\\`s making a speedy delivery!");
        setDisplayText("The local postman is waiting to serve you.");
        CMLib.factions().setAlignment(this,Faction.ALIGN_GOOD);
        setMoney(0);
        whatISell=ShopKeeper.DEAL_POSTMAN;
        baseEnvStats.setWeight(150);
        setWimpHitPoint(0);

        baseCharStats().setStat(CharStats.STAT_INTELLIGENCE,16);
        baseCharStats().setStat(CharStats.STAT_CHARISMA,25);

        baseEnvStats().setArmor(0);

        baseState.setHitPoints(1000);

        recoverMaxState();
        resetToMaxState();
        recoverEnvStats();
        recoverCharStats();
    }



    public double minimumPostage(){return minimumPostage;}
    public void setMinimumPostage(double d){minimumPostage=d;}
    public double postagePerPound(){return postagePerPound;}
    public void setPostagePerPound(double d){postagePerPound=d;}
    public double holdFeePerPound(){return holdFeePerPound;}
    public void setHoldFeePerPound(double d){holdFeePerPound=d;}
    public double feeForNewBox(){return feeForNewBox;}
    public void setFeeForNewBox(double d){feeForNewBox=d;}
    public int maxMudMonthsHeld(){return maxMudMonthsHeld;}
    public void setMaxMudMonthsHeld(int months){maxMudMonthsHeld=months;}
    
    public void destroy()
    {
        super.destroy();
        CMLib.map().delPostOffice(this);
    }
    public void bringToLife(Room newLocation, boolean resetStats)
    {
        super.bringToLife(newLocation,resetStats);
        CMLib.map().addPostOffice(this);
    }
    
    public int whatIsSold(){return whatISell;}
    public void setWhatIsSold(int newSellCode){
        if(newSellCode!=ShopKeeper.DEAL_CLANPOSTMAN)
            whatISell=ShopKeeper.DEAL_POSTMAN;
        else
            whatISell=ShopKeeper.DEAL_CLANPOSTMAN;
    }

    public String postalChain(){return text();}
    public void setPostalChain(String name){setMiscText(name);}
    public String postalBranch(){return CMLib.map().getExtendedRoomID(getStartRoom());}

    public void addToBox(String mob, Item thisThang, String from, String to, long holdTime, double COD)
    {
        String name=thisThang.ID();
        CMLib.database().DBCreateData(mob,
                postalChain(),
                postalBranch()+";"+thisThang+Math.random(),
                from+";"
                +to+";"
                +holdTime+";"
                +COD+";"
                +name+";"
                +CMLib.coffeeMaker().getPropertiesStr(thisThang,true));
    }

    public void addToBox(MOB mob, Item thisThang, String from, String to, long holdTime, double COD)
    {
        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
        {
            if(mob.getClanID().length()==0)
                return;
            addToBox(mob.getClanID(),thisThang,from,to,holdTime,COD);
        }
        else
            addToBox(mob.Name(),thisThang,from,to,holdTime,COD);
    }

    public boolean delFromBox(MOB mob, Item thisThang)
    {
        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
        {
            if(mob.getClanID().length()>0)
                return delFromBox(mob.getClanID(),thisThang);
        }
        else
            return delFromBox(mob.Name(),thisThang);
        return false;
    }

    public boolean delFromBox(String mob, Item thisThang)
    {
        Vector V=getBoxRowData(mob);
        boolean found=false;
        for(int v=V.size()-1;v>=0;v--)
        {
            Vector V2=(Vector)V.elementAt(v);
            if((V2.size()>3)
            &&(((String)V2.elementAt(DATA_KEY)).startsWith(postalBranch()+";")))
            {
                Item I=makeItem(parsePostalItemData((String)V2.elementAt(DATA_DATA)));
                if(I==null) continue;
                if(thisThang.sameAs(I))
                {
                    found=true;
                    CMLib.database().DBDeleteData(
                            ((String)V2.elementAt(DATA_USERID)),
                            ((String)V2.elementAt(DATA_CHAIN)),
                            ((String)V2.elementAt(DATA_KEY)));
                    break;
                }
            }
        }
        return found;
    };
    public void emptyBox(String mob)
    {
        CMLib.database().DBDeleteData(mob,postalChain());
    };
    public Hashtable getOurOpenBoxes(String mob)
    {
        Hashtable branches=new Hashtable();
        Vector V=CMLib.database().DBReadData(mob,postalChain());
        if(V==null) return branches;
        for(int v=0;v<V.size();v++)
        {
            Vector V2=(Vector)V.elementAt(v);
            if(V2.size()>3)
            {
                String key=(String)V2.elementAt(DATA_KEY);
                int x=key.indexOf("/");
                if(x>0)
                    branches.put(key.substring(0,x),key.substring(x+1));
            }
        }
        return branches;
    }
    public void createBoxHere(String mob, String forward)
    {
        if(!getOurOpenBoxes(mob).containsKey(postalBranch()))
        {
            CMLib.database().DBCreateData(mob,
                    postalChain(),
                    postalBranch()+"/"+forward,
                    "50");
        }
    }
    public void deleteBoxHere(String mob)
    {
        Vector V=getBoxRowData(mob);
        if(V==null) return;
        for(int v=0;v<V.size();v++)
        {
            Vector V2=(Vector)V.elementAt(v);
            if((V2.size()>3)
            &&(((String)V2.elementAt(DATA_KEY)).startsWith(postalBranch()+"/")))
            {
                CMLib.database().DBDeleteData(
                ((String)V2.elementAt(DATA_USERID)),
                ((String)V2.elementAt(DATA_CHAIN)),
                ((String)V2.elementAt(DATA_KEY)));
            }
        }
    }
    public Vector getAllLocalBoxVectors(String mob)
    {
        Vector V=getBoxRowData(mob);
        Vector mine=new Vector();
        for(int v=0;v<V.size();v++)
        {
            Vector V2=(Vector)V.elementAt(v);
            if((V2.size()>3)
            &&(((String)V2.elementAt(DATA_KEY)).startsWith(postalBranch()+";")))
            {
                mine.addElement(V2);
            }
        }
        return mine;
    }
    public Vector getBoxRowData(String mob)
    {
        return CMLib.database().DBReadData(mob,postalChain());
    };
    public Item findBoxContents(MOB mob, String likeThis)
    {
        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
        {
            if(mob.getClanID().length()==0) return null;
            return findBoxContents(mob.getClanID(),likeThis);
        }
        return findBoxContents(mob.Name(),likeThis);
    }

    public Item findBoxContents(String mob, String likeThis)
    {
        Vector V=getBoxRowData(mob);
        for(int v=0;v<V.size();v++)
        {
            Vector V2=(Vector)V.elementAt(v);
            if((V2.size()>3)
            &&(((String)V2.elementAt(DATA_KEY)).startsWith(postalBranch()+";")))
            {
                Item I=makeItem(parsePostalItemData((String)V2.elementAt(DATA_DATA)));
                if(I==null) continue;
                if(CMLib.english().containsString(I.Name(),likeThis))
                    return I;
            }
        }
        return null;
    };

    public Vector findExactBoxData(String mob, Item likeThis)
    {
        Vector V=getBoxRowData(mob);
        for(int v=0;v<V.size();v++)
        {
            Vector V2=(Vector)V.elementAt(v);
            if((V2.size()>3)
            &&(((String)V2.elementAt(DATA_KEY)).startsWith(postalBranch()+";")))
            {
                Item I=makeItem(parsePostalItemData((String)V2.elementAt(DATA_DATA)));
                if(I==null) continue;
                if(I.sameAs(likeThis))
                    return parsePostalItemData((String)V2.elementAt(DATA_DATA));
            }
        }
        return null;
    };

    public Vector parsePostalItemData(String data)
    {
        Vector V=new Vector();
        for(int i=0;i<5;i++)
        {
            int x=data.indexOf(";");
            if(x<0) break;
            V.addElement(data.substring(0,x));
            data=data.substring(x+1);
        }
        V.addElement(data);
        if(V.size()<NUM_PIECES)
        {
        	Log.errOut("StdPostman","Man formed postal data: "+data);
        	return new Vector();
        }
        return V;
    }
    
    protected Item makeItem(Vector data)
    {
        if(data.size()<NUM_PIECES)
            return null;
        Item I=CMClass.getItem((String)data.elementAt(PIECE_CLASSID));
        if(I!=null)
        {
            CMLib.coffeeMaker().setPropertiesStr(I,(String)data.elementAt(PIECE_MISCDATA),true);
            I.recoverEnvStats();
            I.text();
            return I;
        }
        return null;
    }
    
    protected int getChargeableWeight(Item I)
    {
        if(I==null) return 0;
        int chargeableWeight=0;
        if(I.envStats().weight()>0)
            chargeableWeight=(I.envStats().weight()-1);
        return chargeableWeight;
    }
    
    protected double getSimplePostage(int chargeableWeight)
    {
        if(getStartRoom()==null) return 0.0;
        return minimumPostage()+CMath.mul(postagePerPound(),chargeableWeight);
    }
    
    protected double getHoldingCost(Vector data, int chargeableWeight)
    {
        if(data.size()<NUM_PIECES)
            return 0.0;
        if(getStartRoom()==null) return 0.0;
        double amt=0.0;
        TimeClock TC=(getStartRoom()==null)?CMClass.globalClock():getStartRoom().getArea().getTimeObj();
        long time=System.currentTimeMillis()-CMath.s_long((String)data.elementAt(PIECE_TIME));
        long millisPerMudMonth=TC.getDaysInMonth()*Tickable.TIME_MILIS_PER_MUDHOUR*TC.getHoursInDay();
        if(time<=0) return amt;
        amt+=CMath.mul(CMath.mul(Math.floor(CMath.div(time,millisPerMudMonth)),holdFeePerPound()),chargeableWeight);
        return amt;
    }
    
    
    protected double getCODChargeForPiece(Vector data)
    {
        if(data.size()<NUM_PIECES)
            return 0.0;
        int chargeableWeight=getChargeableWeight(makeItem(data));
        double COD=CMath.s_double((String)data.elementAt(PIECE_COD));
        double amt=0.0;
        if(COD>0.0) 
            amt=getSimplePostage(chargeableWeight)+COD;
        return amt+getHoldingCost(data,chargeableWeight);
    }
    
    protected String getBranchPostableTo(String toWhom, String branch, Hashtable allBranchBoxes)
    {
        String forward=(String)allBranchBoxes.get(branch);
        if(forward==null) return null;
        if(forward.equalsIgnoreCase(toWhom))
            return branch;
        PostOffice P=CMLib.map().getPostOffice(postalChain(),forward);
        if(P!=null)
        {
            forward=(String)allBranchBoxes.get(P.postalBranch());
            if((forward!=null)&&forward.equalsIgnoreCase(toWhom))
                return P.postalBranch();
        }
        return null;
    }
    
    public String findProperBranch(String toWhom)
    {
        if(CMLib.map().getLoadPlayer(toWhom)!=null)
        {
            MOB M=CMLib.map().getLoadPlayer(toWhom);
            if(M.getStartRoom()!=null)
            {
                Hashtable allBranchBoxes=getOurOpenBoxes(toWhom);
                PostOffice P=CMLib.map().getPostOffice(postalChain(),M.getStartRoom().getArea().Name());
                String branch=null;
                if(P!=null)
                {
                    branch=getBranchPostableTo(toWhom,P.postalBranch(),allBranchBoxes);
                    if(branch!=null) return branch;
                    if((branch==null)&&(allBranchBoxes.size()==0))
                    {
                        P.createBoxHere(toWhom,toWhom);
                        return P.postalBranch();
                    }
                }
                branch=getBranchPostableTo(toWhom,postalBranch(),allBranchBoxes);
                if(branch!=null) return branch;
                for(Enumeration e=allBranchBoxes.keys();e.hasMoreElements();)
                {
                    String tryBranch=(String)e.nextElement();
                    branch=getBranchPostableTo(toWhom,tryBranch,allBranchBoxes);
                    if(branch!=null) return branch;
                }
                if(P!=null)
                {
                    P.deleteBoxHere(toWhom);
                    P.createBoxHere(toWhom,toWhom);
                    return P.postalBranch();
                }
            }
        }
        else
        if(CMLib.clans().getClan(toWhom)!=null)
        {
            Hashtable allBranchBoxes=getOurOpenBoxes(toWhom);
            String branch=getBranchPostableTo(toWhom,postalBranch(),allBranchBoxes);
            if(branch!=null) return branch;
            for(Enumeration e=allBranchBoxes.keys();e.hasMoreElements();)
            {
                String tryBranch=(String)e.nextElement();
                branch=getBranchPostableTo(toWhom,tryBranch,allBranchBoxes);
                if(branch!=null) return branch;
            }
        }
        return null;
    }

    public long postalWaitTime()
    {
        if((postalWaitTime<0)&&(getStartRoom()!=null))
        	postalWaitTime=10038;
        else
        if((postalWaitTime==10038)&&(getStartRoom()!=null))
            postalWaitTime=new Long((getStartRoom().getArea().getTimeObj().getHoursInDay())*Tickable.TIME_MILIS_PER_MUDHOUR).longValue();
        return postalWaitTime;
    }
    
    public boolean tick(Tickable ticking, int tickID)
    {
        if(!super.tick(ticking,tickID))
            return false;
		if(!CMProps.getBoolVar(CMProps.SYSTEMB_MUDSTARTED)) return true;
		
        if((tickID==Tickable.TICKID_MOB)&&(getStartRoom()!=null))
        {
            boolean proceed=false;
            // handle interest by watching the days go by...
            // each chain is handled by one branch, 
            // since pending mail all looks the same.
            Long L=(Long)postalTimes.get(postalChain()+"/"+postalBranch());
            if((L==null)||(L.longValue()<System.currentTimeMillis()))
            {
                proceed=(L!=null);
                L=new Long(System.currentTimeMillis()+postalWaitTime());
                postalTimes.remove(postalChain()+"/"+postalBranch());
                postalTimes.put(postalChain()+"/"+postalBranch(),L);
            }
            if(proceed)
            {
                Vector V=getBoxRowData(postalChain());
                // first parse all the pending mail,
                // and remove it from the sorter
                Vector parsed=new Vector();
                if(V==null) V=new Vector();
                for(int v=0;v<V.size();v++)
                {
                    Vector V2=(Vector)V.elementAt(v);
                    parsed.addElement(parsePostalItemData((String)V2.elementAt(DATA_DATA)));
                    CMLib.database().DBDeleteData(
                    ((String)V2.elementAt(DATA_USERID)),
                    ((String)V2.elementAt(DATA_CHAIN)),
                    ((String)V2.elementAt(DATA_KEY)));
                }
                PostOffice P=null;
                for(int v=0;v<parsed.size();v++)
                {
                    Vector V2=(Vector)parsed.elementAt(v);
                    String toWhom=(String)V2.elementAt(PIECE_TO);
                    String deliveryBranch=findProperBranch(toWhom);
                    if(deliveryBranch!=null)
                    {
                        P=CMLib.map().getPostOffice(postalChain(),deliveryBranch);
                        Item I=makeItem(V2);
                        if((P!=null)&&(I!=null))
                        {
                            P.addToBox(toWhom,I,(String)V2.elementAt(PIECE_FROM),(String)V2.elementAt(PIECE_TO),CMath.s_long((String)V2.elementAt(PIECE_TIME)),CMath.s_double((String)V2.elementAt(PIECE_COD)));
                            continue;
                        }
                    }
                    String fromWhom=(String)V2.elementAt(PIECE_FROM);
                    deliveryBranch=findProperBranch(fromWhom);
                    if(deliveryBranch!=null)
                    {
                        P=CMLib.map().getPostOffice(postalChain(),deliveryBranch);
                        Item I=makeItem(V2);
                        if((P!=null)&&(I!=null))
                            P.addToBox(fromWhom,I,(String)V2.elementAt(PIECE_TO),"POSTMASTER",System.currentTimeMillis(),0.0);
                    }
                }
                V=CMLib.database().DBReadData(postalChain());
                TimeClock TC=null;
                if(getStartRoom()!=null) TC=getStartRoom().getArea().getTimeObj();
                if((TC!=null)&&(maxMudMonthsHeld()>0))
                for(int v=0;v<V.size();v++)
                {
                    Vector V2=(Vector)V.elementAt(v);
                    if((V2.size()>3)
                    &&(((String)V2.elementAt(DATA_KEY)).startsWith(postalBranch()+";")))
                    {
                        Vector data=parsePostalItemData(((String)V2.elementAt(DATA_DATA)));
                        if((data!=null)&&(data.size()>1)&&(getStartRoom()!=null))
                        {
                            long time=System.currentTimeMillis()-CMath.s_long((String)data.elementAt(PIECE_TIME));
                            long millisPerMudMonth=TC.getDaysInMonth()*Tickable.TIME_MILIS_PER_MUDHOUR*TC.getHoursInDay();
                            if(time>0)
                            {
                                int months=(int)Math.round(Math.floor(CMath.div(time,millisPerMudMonth)));
                                if(months>maxMudMonthsHeld())
                                {
                                    Item I=makeItem(V2);
                                    CMLib.database().DBDeleteData(
                                            ((String)V2.elementAt(DATA_USERID)),
                                            ((String)V2.elementAt(DATA_CHAIN)),
                                            ((String)V2.elementAt(DATA_KEY)));
                                    if(I!=null)
                                        getShop().addStoreInventory(I,this);
                                }
                            }
                        }
                    }
                }
            }
        }
        return true;
    }

    public void autoGive(MOB src, MOB tgt, Item I)
    {
        CMMsg msg2=CMClass.getMsg(src,I,null,CMMsg.MSG_DROP,null,CMMsg.MSG_DROP,"GIVE",CMMsg.MSG_DROP,null);
        location().send(this,msg2);
        msg2=CMClass.getMsg(tgt,I,null,CMMsg.MSG_GET,null,CMMsg.MSG_GET,"GIVE",CMMsg.MSG_GET,null);
        location().send(this,msg2);
    }
    
    public void executeMsg(Environmental myHost, CMMsg msg)
    {
        MOB mob=msg.source();
        if(msg.amITarget(this))
        {
            switch(msg.targetMinor())
            {
            case CMMsg.TYP_GIVE:
            case CMMsg.TYP_DEPOSIT:
				if(CMLib.flags().aliveAwakeMobileUnbound(mob,true))
                {
                    if(msg.tool() instanceof Container)
                        ((Container)msg.tool()).emptyPlease();
                    Session S=msg.source().session();
                    if((!msg.source().isMonster())&&(S!=null)&&(msg.tool() instanceof Item))
                    {
                        autoGive(msg.source(),this,(Item)msg.tool());
                        if(isMine(msg.tool()))
                        {
                            try
                            {
                                String toWhom=S.prompt("Address this to whom? ","");
                                if((toWhom!=null)&&(toWhom.length()>0)
                                &&((CMLib.map().getLoadPlayer(toWhom)!=null)||(CMLib.clans().findClan(toWhom)!=null)))
                                {
                                    String fromWhom=msg.source().Name();
                                    if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                                        fromWhom=msg.source().getClanID();
                                    if(CMLib.map().getLoadPlayer(toWhom)!=null)
                                        toWhom=CMLib.map().getLoadPlayer(toWhom).Name();
                                    else
                                        toWhom=CMLib.clans().findClan(toWhom).name();
                                    double amt=getSimplePostage(getChargeableWeight((Item)msg.tool()));
                                    double COD=0.0;
                                    boolean deliver=true;
                                    String choice=S.choose("Postage on this will be "+CMLib.beanCounter().nameCurrencyShort(this,amt)+".\n\rWould you like to P)ay this now, or be C)harged on delivery (c/P)?","CP\n","P").trim().toUpperCase();
                                    if(choice.startsWith("C"))
                                    {
                                        String CODstr=S.prompt("Enter COD amount ("+CMLib.beanCounter().getDenominationName(CMLib.beanCounter().getCurrency(this),CMLib.beanCounter().getLowestDenomination(CMLib.beanCounter().getCurrency(this)))+"): ");
                                        if((CODstr.length()==0)||(!CMath.isNumber(CODstr))||(CMath.s_double(CODstr)<=0.0))
                                        {
                                            CMLib.commands().postSay(this,mob,"That is not a valid amount.",true,false);
                                            autoGive(this,msg.source(),(Item)msg.tool());
                                            deliver=false;
                                        }
                                        else
                                        {
                                            Coins currency=CMLib.beanCounter().makeBestCurrency(CMLib.beanCounter().getCurrency(this),CMLib.beanCounter().getLowestDenomination(CMLib.beanCounter().getCurrency(this))*CMath.s_double(CODstr));
                                            COD=currency.getTotalValue();
                                            amt=0.0;
                                        }
                                    }
                                    else
                                    if((amt>0.0)&&(CMLib.beanCounter().getTotalAbsoluteShopKeepersValue(msg.source(),this)<amt))
                                    {
                                        CMLib.commands().postSay(this,mob,"You can't afford postage.",true,false);
                                        autoGive(this,msg.source(),(Item)msg.tool());
                                        deliver=false;
                                    }
                                    else
                                        CMLib.beanCounter().subtractMoney(mob,CMLib.beanCounter().getCurrency(this),amt);
                                    if(deliver)
                                    {
                                        addToBox(postalChain(),(Item)msg.tool(),fromWhom,toWhom,System.currentTimeMillis(),COD);
                                        CMLib.commands().postSay(this,mob,"I'll deliver that for ya right away!",true,false);
                                        ((Item)msg.tool()).destroy();
                                    }
                                }
                                else
                                {
                                    CMLib.commands().postSay(this,mob,"That is not a valid player or clan name.",true,false);
                                    autoGive(this,msg.source(),(Item)msg.tool());
                                }
                            }
                            catch(Exception e)
                            {
                                CMLib.commands().postDrop(mob,msg.tool(),false,false);
                            }
                        }
                        else
                            CMLib.commands().postSay(this,mob,"Ugh, I can't seem to deliver "+msg.tool().name()+".",true,false);
                    }
                }
                return;
            case CMMsg.TYP_WITHDRAW:
				if(CMLib.flags().aliveAwakeMobileUnbound(mob,true))
                {
                    String thename=msg.source().Name();
                    if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                        thename=msg.source().getClanID();
                    Item old=(Item)msg.tool();
                    Vector data=findExactBoxData(thename,(Item)msg.tool());
                    if((data==null)
                    &&(whatISell!=ShopKeeper.DEAL_CLANPOSTMAN)
                    &&(msg.source().isMarriedToLiege()))
                        data=findExactBoxData(msg.source().getLiegeID(),(Item)msg.tool());
                    if((data==null)||(data.size()<NUM_PIECES))
                        CMLib.commands().postSay(this,mob,"You want WHAT? Try LIST.",true,false);
                    else
                    {
                        if((!delFromBox(msg.source(),old))
                        &&(whatISell!=ShopKeeper.DEAL_CLANPOSTMAN)
                        &&(msg.source().isMarriedToLiege()))
                            delFromBox(msg.source().getLiegeID(),old);
                        double totalCharge=getCODChargeForPiece(data);
                        if((totalCharge>0.0)&&(CMLib.beanCounter().getTotalAbsoluteShopKeepersValue(msg.source(),this)>=totalCharge))
                        {
                            
                            CMLib.beanCounter().subtractMoney(msg.source(),totalCharge);
                            double COD=CMath.s_double((String)data.elementAt(PIECE_COD));
                            Coins returnMoney=null;
                            if(COD>0.0)
                                returnMoney=CMLib.beanCounter().makeBestCurrency(this,COD);
                            if(returnMoney!=null)
                            {
                                CMLib.commands().postSay(this,mob,"The COD amount of "+returnMoney.Name()+" has been sent back to "+((String)data.elementAt(PIECE_FROM))+".",true,false);
                                addToBox(postalChain(),returnMoney,(String)data.elementAt(PIECE_TO),(String)data.elementAt(PIECE_FROM),System.currentTimeMillis(),0.0);
                                CMLib.commands().postSay(this,mob,"The total charge on that was a COD charge of "+returnMoney.Name()+" plus "+CMLib.beanCounter().nameCurrencyShort(this,totalCharge-COD)+" postage and holding fees.",true,false);
                            }
                            else
                                CMLib.commands().postSay(this,mob,"The total charge on that was "+CMLib.beanCounter().nameCurrencyShort(this,totalCharge)+" in holding/storage fees.",true,false);
                        }
                        CMLib.commands().postSay(this,mob,"There ya go!",true,false);
                        if(location()!=null)
                            location().addItemRefuse(old,Item.REFUSE_PLAYER_DROP);
                        CMMsg msg2=CMClass.getMsg(mob,old,this,CMMsg.MSG_GET,null);
                        if(location().okMessage(mob,msg2))
                            location().send(mob,msg2);
                    }
                }
                return;
            case CMMsg.TYP_VALUE:
            case CMMsg.TYP_SELL:
            case CMMsg.TYP_VIEW:
                super.executeMsg(myHost,msg);
                return;
            case CMMsg.TYP_BUY:
                super.executeMsg(myHost,msg);
                return;
            case CMMsg.TYP_SPEAK:
            {
                super.executeMsg(myHost,msg);
                String str=CMStrings.getSayFromMessage(msg.targetMessage());
                String theName=msg.source().Name();
                if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                    theName=msg.source().getClanID();
                if((str!=null)&&(str.trim().equalsIgnoreCase("open")))
                {
                    if((whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                    &&((theName.length()==0)
                      ||(CMLib.clans().getClan(theName)==null)))
                        CMLib.commands().postSay(this,mob,"I'm sorry, I only do business with Clans, and you aren't part of one.",true,false);
                    else
                    if(getOurOpenBoxes(theName).containsKey(postalBranch()))
                    {
                        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                            CMLib.commands().postSay(this,mob,"Your clan already has a box open here!",true,false);
                        else
                            CMLib.commands().postSay(this,mob,"You already have a box open here!",true,false);
                    }
                    else
                    if(feeForNewBox()>0.0)
                    {
                        if(CMLib.beanCounter().getTotalAbsoluteShopKeepersValue(msg.source(),this)<feeForNewBox())
                        {
                            CMLib.commands().postSay(this,mob,"Too bad you can't afford it.",true,false);
                            return;
                        }
                        CMLib.beanCounter().subtractMoney(msg.source(),CMLib.beanCounter().getCurrency(this),feeForNewBox());
                        createBoxHere(theName,theName);
                        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                            CMLib.commands().postSay(this,mob,"A box has been opened for your clan.",true,false);
                        else
                            CMLib.commands().postSay(this,mob,"A box has been opened for you.",true,false);
                    }
                    else
                    {
                        createBoxHere(theName,theName);
                        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                            CMLib.commands().postSay(this,mob,"A box has been opened for your clan.",true,false);
                        else
                            CMLib.commands().postSay(this,mob,"A box has been opened for you.",true,false);
                    }
                }
                else
                if((str!=null)&&(str.trim().equalsIgnoreCase("close")))
                {
                    if(!getOurOpenBoxes(theName).containsKey(postalBranch()))
                    {
                        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                            CMLib.commands().postSay(this,mob,"Your clan does not have a box here!",true,false);
                        else
                            CMLib.commands().postSay(this,mob,"You don't have a box open here!",true,false);
                    }
                    else
                    if(getAllLocalBoxVectors(theName).size()>0)
                        CMLib.commands().postSay(this,mob,"That box has pending items which must be removed first.",true,false);
                    else
                    {
                        deleteBoxHere(theName);
                        CMLib.commands().postSay(this,mob,"That box is now closed.",true,false);
                    }
                }
                else
                if((str!=null)&&(str.toUpperCase().trim().startsWith("FORWARD")))
                {
                    str=str.trim().substring(7).trim();
                    if(!getOurOpenBoxes(theName).containsKey(postalBranch()))
                    {
                        if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                            CMLib.commands().postSay(this,mob,"Your clan does not have a box here!",true,false);
                        else
                            CMLib.commands().postSay(this,mob,"You don't have a box open here!",true,false);
                    }
                    else
                    {
                        Area A=CMLib.map().findArea(str);
                        if(A==null)
                            CMLib.commands().postSay(this,mob,"I don't know of an area called '"+str+"'.",true,false);
                        else
                        {
                            PostOffice P=CMLib.map().getPostOffice(postalChain(),A.Name());
                            if(P==null)
                                CMLib.commands().postSay(this,mob,"I'm sorry, we don't have a branch in "+A.name()+".",true,false);
                            else
                            if(!P.getOurOpenBoxes(theName).containsKey(P.postalBranch()))
                            {
                                if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                                    CMLib.commands().postSay(this,mob,"I'm sorry, your clan does not have a box at our branch in "+A.name()+".",true,false);
                                else
                                    CMLib.commands().postSay(this,mob,"I'm sorry, you don't have a box at our branch in "+A.name()+".",true,false);
                            }
                            else
                            {
                                deleteBoxHere(theName);
                                createBoxHere(theName,P.postalBranch());
                                CMLib.commands().postSay(this,mob,"Ok, mail will now be forwarded to our branch in "+A.name()+".",true,false);
                            }
                        }
                    }
                }
                return;
            }
            case CMMsg.TYP_LIST:
            {
                super.executeMsg(myHost,msg);
    			if(CMLib.flags().aliveAwakeMobileUnbound(mob,true))
    			{
	                Vector V=null;
	                if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
	                    V=getAllLocalBoxVectors(mob.getClanID());
	                else
	                {
	                    V=getAllLocalBoxVectors(mob.Name());
	                    if(mob.isMarriedToLiege())
	                    {
	                        Vector V2=getAllLocalBoxVectors(mob.getLiegeID());
	                        if((V2!=null)&&(V2.size()>0))
	                            CMParms.addToVector(V2,V);
	                    }
	                }
	
	                TimeClock C=CMClass.globalClock();
	                if(getStartRoom()!=null) C=getStartRoom().getArea().getTimeObj();
	                boolean codCharge=false;
	                if(V.size()==0)
	                    mob.tell("Your postal box is presently empty.");
	                else
	                {
	                    StringBuffer str=new StringBuffer("");
	                    str.append("\n\rItems in your postal box here:\n\r");
	                    str.append("^x[COD     ][From           ][Sent           ][Item                        ]^.^N");
	                    mob.tell(str.toString());
	                    for(int i=0;i<V.size();i++)
	                    {
	                        Vector V2=(Vector)V.elementAt(i);
	                        Vector pieces=parsePostalItemData((String)V2.elementAt(DATA_DATA));
	                        Item I=makeItem(pieces);
	                        if(I==null) continue;
	                        str=new StringBuffer("^N");
	                        if(getCODChargeForPiece(pieces)>0.0)
	                        {
	                            codCharge=true;
	                            str.append("["+CMStrings.padRight(""+CMLib.beanCounter().abbreviatedPrice(this,getCODChargeForPiece(pieces)),8)+"]");
	                        }
	                        else
	                            str.append("[        ]");
	                        str.append("["+CMStrings.padRight((String)pieces.elementAt(PIECE_FROM),15)+"]");
	                        TimeClock C2=C.deriveClock(CMath.s_long((String)pieces.elementAt(PIECE_TIME)));
	                        str.append("["+CMStrings.padRight(C2.getShortestTimeDescription(),15)+"]");
	                        str.append("["+CMStrings.padRight(I.Name(),28)+"]");
	                        mob.tell(str.toString()+"^T");
	                    }
	                }
	                StringBuffer str=new StringBuffer("\n\r^N");
	                if(codCharge)
	                    str.append("* COD charges above include all shipping costs.\n\r");
	                str.append("* This branch charges minimum "+CMLib.beanCounter().nameCurrencyShort(this,minimumPostage())+" postage for first pound.\n\r");
	                str.append("* An additional "+CMLib.beanCounter().nameCurrencyShort(this,postagePerPound())+" per pound is charged for packages.\n\r");
	                str.append("* A charge of "+CMLib.beanCounter().nameCurrencyShort(this,holdFeePerPound())+" per pound per month is charged for holding.\n\r");
	                str.append("* To forward your mail, 'say \""+name()+"\" \"forward <areaname>\"'.\n\r");
	                str.append("* To close your box, 'say \""+name()+"\" close'.\n\r");
	                mob.tell(str.toString());
    			}
                return;
            }
            default:
                break;
            }
        }
        else
        if(msg.sourceMinor()==CMMsg.TYP_RETIRE)
            emptyBox(msg.source().Name());
        super.executeMsg(myHost,msg);
    }

    public boolean okMessage(Environmental myHost, CMMsg msg)
    {
        MOB mob=msg.source();
        if((msg.targetMinor()==CMMsg.TYP_EXPIRE)
        &&(msg.target()==location())
        &&(CMLib.flags().isInTheGame(this,true)))
        	return false;
        else
        if(msg.amITarget(this))
        {
            switch(msg.targetMinor())
            {
            case CMMsg.TYP_GIVE:
            case CMMsg.TYP_DEPOSIT:
                {
                    if(!CMLib.coffeeShops().ignoreIfNecessary(msg.source(),finalIgnoreMask(),this)) 
                        return false;
                    if(msg.tool()==null) return false;
                    if((whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                    &&((msg.source().getClanID().length()==0)
                      ||(CMLib.clans().getClan(msg.source().getClanID())==null)))
                    {
                        CMLib.commands().postSay(this,mob,"I'm sorry, I only do business with Clans, and you aren't part of one.",true,false);
                        return false;
                    }
                    if(!(msg.tool() instanceof Item))
                    {
                        mob.tell(mob.charStats().HeShe()+" doesn't look interested.");
                        return false;
                    }
                }
                return true;
            case CMMsg.TYP_WITHDRAW:
                {
                    if(!CMLib.coffeeShops().ignoreIfNecessary(msg.source(),finalIgnoreMask(),this)) 
                        return false;
                    String thename=msg.source().Name();
                    if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                    {
                        thename=msg.source().getClanID();
                        Clan C=CMLib.clans().getClan(msg.source().getClanID());
                        if((msg.source().getClanID().length()==0)
                          ||(C==null))
                        {
                            CMLib.commands().postSay(this,mob,"I'm sorry, I only do business with Clans, and you aren't part of one.",true,false);
                            return false;
                        }

                        if(C.allowedToDoThis(msg.source(),Clan.FUNC_CLANWITHDRAW)<0)
                        {
                            CMLib.commands().postSay(this,mob,"I'm sorry, you aren't authorized by your clan to do that.",true,false);
                            return false;
                        }
                    }
                    if((msg.tool()==null)||(!(msg.tool() instanceof Item)))
                    {
                        CMLib.commands().postSay(this,mob,"What do you want? I'm busy!",true,false);
                        return false;
                    }
                    if((msg.tool()!=null)&&(!msg.tool().okMessage(myHost,msg)))
                        return false;
                    Vector data=findExactBoxData(thename,(Item)msg.tool());
                    if((data==null)
                    &&(whatISell!=ShopKeeper.DEAL_CLANPOSTMAN)
                    &&(msg.source().isMarriedToLiege()))
                        data=findExactBoxData(msg.source().getLiegeID(),(Item)msg.tool());
                    if((data==null)||(data.size()<NUM_PIECES))
                    {
                        CMLib.commands().postSay(this,mob,"You want WHAT? Try LIST.",true,false);
                        return false;
                    }
                    double totalCharge=getCODChargeForPiece(data);
                    if(CMLib.beanCounter().getTotalAbsoluteShopKeepersValue(msg.source(),this)<totalCharge)
                    {
                        CMLib.commands().postSay(this,mob,"The total charge to receive that item is "+CMLib.beanCounter().nameCurrencyShort(this,totalCharge)+". You don't have enough." ,true,false);
                        return false;
                    }
                }
                return true;
            case CMMsg.TYP_VALUE:
            case CMMsg.TYP_SELL:
            case CMMsg.TYP_VIEW:
                return super.okMessage(myHost,msg);
            case CMMsg.TYP_BUY:
                return super.okMessage(myHost,msg);
            case CMMsg.TYP_LIST:
            {
                if(!CMLib.coffeeShops().ignoreIfNecessary(msg.source(),finalIgnoreMask(),this)) 
                    return false;
                String thename=msg.source().Name();
                if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                {
                    thename=msg.source().getClanID();
                    Clan C=CMLib.clans().getClan(msg.source().getClanID());
                    if((msg.source().getClanID().length()==0)
                      ||(C==null))
                    {
                        CMLib.commands().postSay(this,mob,"I'm sorry, I only do business with Clans, and you aren't part of one.",true,false);
                        return false;
                    }
                    if(C.allowedToDoThis(msg.source(),Clan.FUNC_CLANDEPOSITLIST)<0)
                    {
                        CMLib.commands().postSay(this,mob,"I'm sorry, you aren't authorized by your clan to do that.",true,false);
                        return false;
                    }
                }
                else
                if((!getOurOpenBoxes(thename).containsKey(postalBranch()))
                &&((whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                   ||(!msg.source().isMarriedToLiege())
                   ||(!getOurOpenBoxes(msg.source().getLiegeID()).containsKey(postalBranch()))))
                {
                    if((whatISell!=ShopKeeper.DEAL_CLANPOSTMAN)
                    &&(msg.source().getStartRoom().getArea()==getStartRoom().getArea()))
                    {
                        createBoxHere(msg.source().Name(),msg.source().Name());
                        return true;
                    }
                    StringBuffer str=new StringBuffer("");
                    if(whatISell==ShopKeeper.DEAL_CLANPOSTMAN)
                        str.append("The Clan "+thename+" does not have a postal box at this branch, I'm afraid.");
                    else
                        str.append("You don't have a postal box at this branch, I'm afraid.");
                    if(postalChain().length()>0)
                        str.append("\n\rThis branch is part of the "+postalChain()+" postal chain.");
                    CMLib.commands().postSay(this,mob,str.toString()+"^T",true,false);
                    mob.tell("Use 'say \""+name()+"\" open' to open a box here"+((feeForNewBox()<=0.0)?".":(" for "+CMLib.beanCounter().nameCurrencyShort(this,feeForNewBox())+".")));
                    return false;
                }
                else
                    return true;
            }
            default:
                break;
            }
        }
        return super.okMessage(myHost,msg);
    }
}