/
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.Abilities.Properties;
import com.planet_ink.coffee_mud.core.interfaces.*;
import com.planet_ink.coffee_mud.Libraries.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 Prop_RoomForSale extends Property implements LandTitle
{
	public String ID() { return "Prop_RoomForSale"; }
	public String name(){ return "Putting a room up for sale";}
	protected int canAffectCode(){return Ability.CAN_ROOMS;}

    public final static String SALESTR=" This lot is for sale (look id).";
    public final static String RENTSTR=" This lot (look id) is for rent on a monthly basis.";
	protected int lastItemNums=-1;
	protected int lastDayDone=-1;
    protected boolean scheduleReset=false;

	public String accountForYourself()
	{ return "For Sale";	}

	public int landPrice()
	{
		if(text().length()==0)
		    return 100000;
		String s=text();
		int index=s.length();
		while((--index)>=0)
		{
			if((!Character.isDigit(s.charAt(index)))
			&&(!Character.isWhitespace(s.charAt(index))))
			    break;
		}
		int price=CMath.s_int(s.substring(index+1).trim());
			    
		if(price<=0) price=100000;
		return price;
	}
	
	public void setLandPrice(int price)
	{   
	    setMiscText(landOwner()+"/"
	        +(rentalProperty()?"RENTAL ":"")
	        +((backTaxes()!=0)?"TAX"+backTaxes()+"X ":"")
	        +price);
	}
	
	public String landOwner()
	{
		if(text().indexOf("/")<0) return "";
		return text().substring(0,text().indexOf("/"));
	}

	public void setLandOwner(String owner)
	{   
        if((owner.length()==0)&&(landOwner().length()>0))
            scheduleReset=true;
	    setMiscText(owner+"/"
		        +(rentalProperty()?"RENTAL ":"")
		        +((backTaxes()!=0)?"TAX"+backTaxes()+"X ":"")
		        +landPrice());
    }

	public int backTaxes()
	{
		if(text().indexOf("/")<0) return 0;
		int x=text().indexOf("TAX",text().indexOf("/"));
		if(x<0) return 0;
		String s=(String)CMParms.parse(text().substring(x+3)).firstElement();
		return CMath.s_int(s.substring(0,s.length()-1));
    }
	public void setBackTaxes(int tax)
	{	
	    setMiscText(landOwner()+"/"
		        +(rentalProperty()?"RENTAL ":"")
		        +((tax!=0)?"TAX"+tax+"X ":"")
		        +landPrice());
	}
	
	public boolean rentalProperty()
	{
		if(text().indexOf("/")<0) return text().indexOf("RENTAL")>=0;
	    return text().indexOf("RENTAL",text().indexOf("/"))>0;
    }
	public void setRentalProperty(boolean truefalse)
	{	
	    setMiscText(landOwner()+"/"
		        +(truefalse?"RENTAL ":"")
		        +((backTaxes()!=0)?"TAX"+backTaxes()+"X ":"")
		        +landPrice());
	}

	// update title, since it may affect clusters, worries about ALL involved
	public void updateTitle()
	{
		if(affected instanceof Room)
			CMLib.database().DBUpdateRoom((Room)affected);
		else
		{
			Room R=CMLib.map().getRoom(landPropertyID());
			if(R!=null) CMLib.database().DBUpdateRoom(R);
		}
	}

	public String landPropertyID(){
		if((affected!=null)&&(affected instanceof Room))
			return CMLib.map().getExtendedRoomID(((Room)affected));
		return "";
	}

	public void setLandPropertyID(String landID){}

	public static boolean shopkeeperMobPresent(Room R)
	{
	    if(R==null) return false;
	    MOB M=null;
	    for(int i=0;i<R.numInhabitants();i++)
	    {
	        M=R.fetchInhabitant(i);
	        if((M.getStartRoom()==R)
	        &&(M.isMonster())
	        &&(CMLib.coffeeShops().getShopKeeper(M)!=null))
	            return true;
	    }
	    return false;
	}
	
	public static boolean robberyCheck(LandTitle A, CMMsg msg)
	{
		if(msg.targetMinor()==CMMsg.TYP_GET)
		{
			if((msg.target() instanceof Item)
			&&(((Item)msg.target()).owner() ==msg.source().location())
            &&((!(msg.tool() instanceof Item))||(msg.source().isMine(msg.tool())))
			&&(!CMath.bset(msg.sourceMajor(),CMMsg.MASK_ALWAYS))
			&&(A.landOwner().length()>0)
			&&(msg.source().location()!=null)
		    &&(msg.othersMessage()!=null)
		    &&(msg.othersMessage().length()>0)
		    &&(!shopkeeperMobPresent(msg.source().location()))
			&&(!CMLib.law().doesHavePriviledgesHere(msg.source(),msg.source().location())))
		    {
			    Room R=msg.source().location();
                LegalBehavior B=CMLib.law().getLegalBehavior(R);
				if(B!=null)
				{
				    for(int m=0;m<R.numInhabitants();m++)
				    {
				        MOB M=R.fetchInhabitant(m);
				        if(CMLib.law().doesHavePriviledgesHere(M,R))
				            return true;
				    }
					MOB D=null;
				    Clan C=CMLib.clans().getClan(A.landOwner());
				    if(C!=null)
				        D=C.getResponsibleMember();
				    else
				        D=CMLib.map().getLoadPlayer(A.landOwner());
				    if(D==null) return true;
                    B.accuse(CMLib.law().getLegalObject(R),msg.source(),D,CMParms.makeVector("PROPERTYROB","THIEF_ROBBERY"));
				}
		    }
			return true;
        }
		return false;
	}
	
	public void executeMsg(Environmental myHost, CMMsg msg)
	{
		super.executeMsg(myHost,msg);
		if(((msg.sourceMinor()==CMMsg.TYP_SHUTDOWN)
				||((msg.targetMinor()==CMMsg.TYP_EXPIRE)&&(msg.target()==affected))
				||(msg.sourceMinor()==CMMsg.TYP_ROOMRESET))
		&&(affected instanceof Room))
		{
			updateLot(null);
			Vector mobs=new Vector();
			Room R=(Room)affected;
			if(R!=null)
			{
				synchronized(("SYNC"+R.roomID()).intern())
				{
					R=CMLib.map().getRoom(R);
					for(int m=0;m<R.numInhabitants();m++)
					{
						MOB M=R.fetchInhabitant(m);
						if((M!=null)
						&&(M.savable())
						&&(M.getStartRoom()==R)
						&&((M.baseEnvStats().rejuv()==0)||(M.baseEnvStats().rejuv()==Integer.MAX_VALUE)))
							mobs.addElement(M);
					}
					if(!CMSecurity.isSaveFlag("NOPROPERTYMOBS"))
						CMLib.database().DBUpdateTheseMOBs(R,mobs);
				}
			}
		}
	}

	public boolean okMessage(Environmental myHost, CMMsg msg)
	{
	    if(!super.okMessage(myHost,msg)) return false;
		Prop_RoomForSale.robberyCheck(this,msg);
		return true;
	}
	
	public static void colorForSale(Room R, boolean rental, boolean reset)
	{
		synchronized(("SYNC"+R.roomID()).intern())
		{
			R=CMLib.map().getRoom(R);
		    String theStr=rental?RENTSTR:SALESTR;
		    String otherStr=rental?SALESTR:RENTSTR;
			int x=R.description().indexOf(otherStr);
			while(x>=0)
			{
				R.setDescription(R.description().substring(0,x));
				CMLib.database().DBUpdateRoom(R);
	            x=R.description().indexOf(otherStr);
			}
			if(R.description().indexOf(theStr.trim())<0)
			{
				if(reset)
				{
					R.setDisplayText("An empty plot");
					R.setDescription("");
				}
				R.setDescription(R.description()+theStr);
				CMLib.database().DBUpdateRoom(R);
	        }
			Item I=R.fetchItem(null,"$id$");
			if((I==null)||(!I.ID().equals("GenWallpaper")))
			{
				I=CMClass.getItem("GenWallpaper");
				CMLib.flags().setReadable(I,true);
				I.setName("id");
				I.setReadableText("This room is "+CMLib.map().getExtendedRoomID(R));
				I.setDescription("This room is "+CMLib.map().getExtendedRoomID(R));
				R.addItem(I);
				CMLib.database().DBUpdateItems(R);
			}
		}
	}

	public Vector getPropertyRooms()
	{
		Vector V=new Vector();
		if(affected instanceof Room)
			V.addElement(affected);
		else
		{
			Room R=CMLib.map().getRoom(landPropertyID());
			if(R!=null) V.addElement(R);
		}
		return V;
	}

	public static int updateLotWithThisData(Room R,
											LandTitle T,
											boolean resetRoomName,
                                            boolean clearAllItems,
                                            Vector optPlayerList,
											int lastNumItems)
	{
        boolean updateItems=false;
        boolean updateExits=false;
        boolean updateRoom=false;
		synchronized(("SYNC"+R.roomID()).intern())
		{
			R=CMLib.map().getRoom(R);
			if(T.landOwner().length()==0)
			{
	            Item I=null;
				for(int i=R.numItems()-1;i>=0;i--)
				{
					I=R.fetchItem(i);
	                if((I==null)||(I.Name().equalsIgnoreCase("id"))) continue;
	                if(clearAllItems)
	                {
	                    I.destroy();
	                    updateItems=true;
	                }
	                else
	                {
	    				if(I.expirationDate()==0)
	    				{
	    					long now=System.currentTimeMillis();
	    					now+=(TimeManager.MILI_HOUR*Item.REFUSE_PLAYER_DROP);
	    					I.setExpirationDate(now);
	    				}
	    				if((I.envStats().rejuv()!=Integer.MAX_VALUE)
	    				&&(I.envStats().rejuv()!=0))
	    				{
	    					I.baseEnvStats().setRejuv(Integer.MAX_VALUE);
	    					I.recoverEnvStats();
	    				}
	                }
				}
	            Ability A=null;
	            if(clearAllItems)
	            for(int a=R.numEffects();a>=0;a--)
	            {
	                A=R.fetchEffect(a);
	                if(((A!=null)&&((A.classificationCode()&Ability.ALL_ACODES)!=Ability.ACODE_PROPERTY)))
	                {
	                    A.unInvoke();
	                    R.delEffect(A);
	                    updateRoom=true;
	                }
	            }
	            for(int d=0;d<Directions.NUM_DIRECTIONS;d++)
	            {
	                Room R2=R.rawDoors()[d];
	                Exit E=R.rawExits()[d];
	                if((E!=null)&&(E.hasALock())&&(E.isGeneric()))
	                {
	                    E.setKeyName("");
	                    E.setDoorsNLocks(E.hasADoor(),E.isOpen(),E.defaultsClosed(),false,false,false);
	                    updateExits=true;
	                    if(R2!=null)
	                    {
	                        E=R2.rawExits()[Directions.getOpDirectionCode(d)];
	                        if((E!=null)&&(E.hasALock())&&(E.isGeneric()))
	                        {
	                            E.setKeyName("");
	                            E.setDoorsNLocks(E.hasADoor(),E.isOpen(),E.defaultsClosed(),false,false,false);
	                            CMLib.database().DBUpdateExits(R2);
	                            R2.getArea().fillInAreaRoom(R2);
	                        }
	                    }
	                }
	            }
	            if(updateExits)
	            {
	                CMLib.database().DBUpdateExits(R);
	                R.getArea().fillInAreaRoom(R);
	            }
	            if(updateItems)
	                CMLib.database().DBUpdateItems(R);
	            if(updateRoom)
	                CMLib.database().DBUpdateRoom(R);
				colorForSale(R,T.rentalProperty(),resetRoomName);
				return -1;
			}
            
			if((lastNumItems<0)
            &&(!CMSecurity.isDisabled("PROPERTYOWNERCHECKS"))
            &&(optPlayerList!=null))
			{
                boolean playerExists=(CMLib.map().getPlayer(T.landOwner())!=null);
                if(!playerExists) playerExists=(CMLib.clans().getClan(T.landOwner())!=null);
                if(!playerExists) playerExists=optPlayerList.contains(T.landOwner());
                if(!playerExists)
                for(int i=0;i<optPlayerList.size();i++)
                    if(((String)optPlayerList.elementAt(i)).equalsIgnoreCase(T.landOwner()))
                    { playerExists=true; break;}
                if(!playerExists)
				{
					T.setLandOwner("");
					T.updateLot(null);
					return -1;
				}
			}
	
			int x=R.description().indexOf(SALESTR);
			if(x>=0)
			{
				R.setDescription(R.description().substring(0,x));
				CMLib.database().DBUpdateRoom(R);
			}
			x=R.description().indexOf(RENTSTR);
			if(x>=0)
			{
				R.setDescription(R.description().substring(0,x));
				CMLib.database().DBUpdateRoom(R);
			}
			
			// this works on the priciple that
			// 1. if an item has ONLY been removed, the lastNumItems will be != current # items
			// 2. if an item has ONLY been added, the dispossessiontime will be != null
			// 3. if an item has been added AND removed, the dispossession time will be != null on the added
			if((lastNumItems>=0)&&(R.numItems()!=lastNumItems))
				updateItems=true;
	
			for(int i=0;i<R.numItems();i++)
			{
				Item I=R.fetchItem(i);
				if((I.expirationDate()!=0)
	            &&((I.savable())||(I.Name().equalsIgnoreCase("id")))
				&&((!(I instanceof DeadBody))||(((DeadBody)I).playerCorpse())))
				{
					I.setExpirationDate(0);
					updateItems=true;
				}
	
				if((I.envStats().rejuv()!=Integer.MAX_VALUE)
				&&(I.envStats().rejuv()!=0))
				{
					I.baseEnvStats().setRejuv(Integer.MAX_VALUE);
					I.recoverEnvStats();
					updateItems=true;
				}
			}
			lastNumItems=R.numItems();
			if((!CMSecurity.isSaveFlag("NOPROPERTYITEMS"))
			&&(updateItems))
				CMLib.database().DBUpdateItems(R);
		}
		return lastNumItems;
	}

	public static boolean doRentalProperty(Area A, String ID, String owner, int rent)
	{
	    if(!CMProps.getBoolVar(CMProps.SYSTEMB_MUDSTARTED))
	        return false;
	    int month=A.getTimeObj().getMonth();
	    int day=A.getTimeObj().getDayOfMonth();
	    int year=A.getTimeObj().getYear();
	    Object O=Resources.getResource("RENTAL INFO/"+owner);
	    Vector V=null;
	    if(O instanceof Vector)
	        V=(Vector)O;
	    else
	        V=CMLib.database().DBReadData(owner,"RENTAL INFO");
	    if(V==null)
	        V=new Vector();
	    if(V.size()==0)
	    {
		    V=new Vector();
	        V.addElement(owner);
	        V.addElement("RENTAL INFO");
	        V.addElement("RENTAL INFO/"+owner);
	        V.addElement(ID+"|~>|"+day+" "+month+" "+year+"|~;|");
	        CMLib.database().DBCreateData(owner,"RENTAL INFO","RENTAL INFO/"+owner,(String)V.lastElement());
	        Vector V2=new Vector();
	        V2.addElement(V);
	        Resources.submitResource("RENTAL INFO/"+owner,V2);
	        return false;
	    }
	    else
	    if(V.firstElement() instanceof Vector)
	    {
	        V=(Vector)V.firstElement();
	        if(V.size()>2)
	        {
		        String parse=(String)V.lastElement();
		        int x=parse.indexOf("|~;|");
		        StringBuffer reparse=new StringBuffer("");
		        boolean changesMade=false;
                boolean needsToPay=false;
		        while(x>=0)
		        {
		            String thisOne=parse.substring(0,x);
		            if(thisOne.startsWith(ID+"|~>|"))
		            {
		                thisOne=thisOne.substring((ID+"|~>|").length());
		                V=CMParms.parse(thisOne);
		                if(V.size()==3)
		                {
		                    int lastYear=CMath.s_int((String)V.lastElement());
		                    int lastMonth=CMath.s_int((String)V.elementAt(1));
		                    int lastDay=CMath.s_int((String)V.firstElement());
		                    while(!needsToPay)
		                    {
			                    if(lastYear<year) 
			                        needsToPay=true;
			                    else
			                    if((lastYear==year)&&(lastMonth<month)&&(day>=lastDay)) 
			                        needsToPay=true;
			                    if(needsToPay)
			                    {
			                        if(CMLib.beanCounter().modifyLocalBankGold(A,
			                                owner,
			                                CMLib.utensils().getFormattedDate(A)+":Withdrawl of "+rent+": Rent for "+ID,
			                                CMLib.beanCounter().getCurrency(A),
			                                new Integer(-rent).doubleValue()))
			                        {
			                            lastMonth++;
			                            if(lastMonth>A.getTimeObj().getMonthsInYear())
			                            {
			                                lastMonth=1;
			                                lastYear++;
			                            }
			                            changesMade=true;
					                    needsToPay=false;
			                        }
			                    }
			                    else
			                        break;
		                    }
		                    if(changesMade)
		                        reparse.append(ID+"|~>|"+lastDay+" "+lastMonth+" "+lastYear+"|~;|");
		                    if(needsToPay&&(!changesMade))
		                        return true;
		                }
		            }
		            else
		                reparse.append(thisOne+"|~;|");
		            parse=parse.substring(x+4);
		            x=parse.indexOf("|~;|");
		        }
		        if(changesMade)
		        {
			        CMLib.database().DBReCreateData(owner,"RENTAL INFO","RENTAL INFO/"+owner,reparse.toString());
				    V=new Vector();
			        V.addElement(owner);
			        V.addElement("RENTAL INFO");
			        V.addElement("RENTAL INFO/"+owner);
			        V.addElement(reparse.toString());
			        Vector V2=new Vector();
			        V2.addElement(V);
			        Resources.updateResource("RENTAL INFO/"+owner,V2);
		        }
			    return needsToPay;
	        }
		    return false;
	    }
	    return false;
	}
	
	// update lot, since its called by the savethread, ONLY worries about itself
	public void updateLot(Vector optPlayerList)
	{
		if(affected instanceof Room)
		{
            Room R=(Room)affected;
    		synchronized(("SYNC"+R.roomID()).intern())
    		{
    			R=CMLib.map().getRoom(R);
				lastItemNums=updateLotWithThisData(R,this,false,scheduleReset,optPlayerList,lastItemNums);
				if((lastDayDone!=R.getArea().getTimeObj().getDayOfMonth())
				&&(CMProps.getBoolVar(CMProps.SYSTEMB_MUDSTARTED)))
				{
				    lastDayDone=R.getArea().getTimeObj().getDayOfMonth();
				    if((landOwner().length()>0)&&rentalProperty()&&(R.roomID().length()>0))
				        if(doRentalProperty(R.getArea(),R.roomID(),landOwner(),landPrice()))
				        {
				            setLandOwner("");
							CMLib.database().DBUpdateRoom(R);
							lastItemNums=updateLotWithThisData(R,this,false,scheduleReset,optPlayerList,lastItemNums);
				        }
				}
	            scheduleReset=false;
    		}
		}
	}
}