package com.planet_ink.coffee_mud.Abilities.Properties;
import com.planet_ink.coffee_mud.core.interfaces.*;
import com.planet_ink.coffee_mud.core.*;
import com.planet_ink.coffee_mud.core.collections.*;
import com.planet_ink.coffee_mud.Abilities.interfaces.*;
import com.planet_ink.coffee_mud.Areas.interfaces.*;
import com.planet_ink.coffee_mud.Behaviors.interfaces.*;
import com.planet_ink.coffee_mud.CharClasses.interfaces.*;
import com.planet_ink.coffee_mud.Commands.interfaces.*;
import com.planet_ink.coffee_mud.Common.interfaces.*;
import com.planet_ink.coffee_mud.Exits.interfaces.*;
import com.planet_ink.coffee_mud.Items.interfaces.*;
import com.planet_ink.coffee_mud.Libraries.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 2003-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 Prop_LotsForSale extends Prop_RoomForSale
{
@Override
public String ID()
{
return "Prop_LotsForSale";
}
@Override
public String name()
{
return "Putting many rooms up for sale";
}
protected String uniqueLotID = null;
@Override
public boolean allowsExpansionConstruction()
{
return true;
}
protected void fillCluster(final Room R, final List<Room> V)
{
V.add(R);
for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
{
final Room R2=R.getRoomInDir(d);
if((R2!=null)&&(R2.roomID().length()>0)&&(!V.contains(R2)))
{
final Ability A=R2.fetchEffect(ID());
if((R2.getArea()==R.getArea())&&(A!=null))
fillCluster(R2,V);
else
{
V.remove(R); // purpose here is to put the "front" door up front.
V.add(0,R);
}
}
}
}
@Override
public List<Room> getConnectedPropertyRooms()
{
final List<Room> V=new ArrayList<Room>();
Room R=null;
if(affected instanceof Room)
R=(Room)affected;
else
R=CMLib.map().getRoom(landPropertyID());
if(R!=null)
{
fillCluster(R,V);
String uniqueID="LOTS_PROPERTY_"+this;
if(V.size()>0)
uniqueID="LOTS_PROPERTY_"+CMLib.map().getExtendedRoomID(V.get(0));
for(final Iterator<Room> r=V.iterator();r.hasNext();)
{
Ability A=null;
R=r.next();
if(R!=null)
A=R.fetchEffect(ID());
if(A instanceof Prop_LotsForSale)
((Prop_LotsForSale)A).uniqueLotID=uniqueID;
}
}
else
uniqueLotID="";
return V;
}
protected boolean isRetractableLink(final Map<Room,Boolean> recurseChkRooms, final Room fromRoom, final Room theRoom)
{
if(theRoom==null)
return true;
if((theRoom.roomID().length()>0)
&&((CMLib.law().getLandTitle(theRoom)==null)
||(CMLib.law().getLandTitle(theRoom).getOwnerName().length()>0)))
{
if(recurseChkRooms != null)
recurseChkRooms.put(theRoom, Boolean.valueOf(false));
return false;
}
for(int d=Directions.NUM_DIRECTIONS()-1;d>=0;d--)
{
final Room R=theRoom.rawDoors()[d];
if(R!=null)
{
if((recurseChkRooms != null)
&&(recurseChkRooms.containsKey(R)))
return recurseChkRooms.get(theRoom).booleanValue();
if((R!=fromRoom)
&&(R.roomID().length()>0))
{
if((CMLib.law().getLandTitle(R)==null)||(CMLib.law().getLandTitle(R).getOwnerName().length()>0))
{
if(recurseChkRooms != null)
recurseChkRooms.put(theRoom, Boolean.valueOf(false));
return false;
}
if((recurseChkRooms != null)
&&(!isRetractableLink(recurseChkRooms,theRoom,R)))
{
recurseChkRooms.put(theRoom, Boolean.valueOf(false));
return false;
}
}
}
}
if(recurseChkRooms != null)
recurseChkRooms.put(theRoom, Boolean.valueOf(true));
return true;
}
@Override
public String getUniqueLotID()
{
if(uniqueLotID==null)
getConnectedPropertyRooms();
return uniqueLotID;
}
@Override
public LandTitle generateNextRoomTitle()
{
final LandTitle newTitle=(LandTitle)this.copyOf();
newTitle.setOwnerName("");
newTitle.setBackTaxes(0);
return newTitle;
}
protected boolean canGenerateAdjacentRooms(final Room R)
{
return getOwnerName().length()>0;
}
protected boolean retractRooms(final Room R, final List<Runnable> postWork)
{
boolean updateExits=false;
boolean foundOne=false;
boolean didAnything = false;
final Map<Room,Boolean> checkedRetractRooms;
if(super.gridLayout())
checkedRetractRooms = new Hashtable<Room,Boolean>();
else
checkedRetractRooms = null;
for(int d=0;d<Directions.NUM_DIRECTIONS();d++)
{
if(d==Directions.GATE)
continue;
final Room R2=R.rawDoors()[d];
if((R2!=null)&&((!R2.isSavable())||(R2.roomID().length()==0)))
continue;
Exit E=R.getRawExit(d);
if(checkedRetractRooms != null)
checkedRetractRooms.clear();
if((R2!=null)&&(R2.rawDoors()[Directions.getOpDirectionCode(d)]==R))
foundOne=true;
else
if((R2!=null)
&&(isRetractableLink(checkedRetractRooms,R,R2)))
{
R.rawDoors()[d]=null;
R.setRawExit(d,null);
updateExits=true;
postWork.add(new Runnable()
{
final Room room=R2;
@Override
public void run()
{
CMLib.map().obliterateMapRoom(room);
}
});
didAnything=true;
}
else
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.getRawExit(Directions.getOpDirectionCode(d));
if((E!=null)&&(E.hasALock())&&(E.isGeneric()))
{
E.setKeyName("");
E.setDoorsNLocks(E.hasADoor(),E.isOpen(),E.defaultsClosed(),false,false,false);
postWork.add(new Runnable()
{
final Room room=R2;
@Override
public void run()
{
CMLib.database().DBUpdateExits(room);
R2.getArea().fillInAreaRoom(room);
}
});
didAnything=true;
}
}
}
}
if(checkedRetractRooms != null)
checkedRetractRooms.clear();
if(!foundOne)
{
CMLib.map().obliterateMapRoom(R);
didAnything=true;
}
else
if(updateExits)
{
CMLib.database().DBUpdateExits(R);
R.getArea().fillInAreaRoom(R);
didAnything=true;
}
return didAnything;
}
public boolean expandRooms(final Room R, final List<Runnable> postWork)
{
int numberOfPeers = -1;//getConnectedPropertyRooms().size();
final boolean doGrid=super.gridLayout();
long roomLimit = Long.MAX_VALUE;
final Set<Room> updateExits=new HashSet<Room>();
Prop_ReqCapacity cap = null;
boolean didAnything = false;
for(int d=0;d<Directions.NUM_DIRECTIONS();d++)
{
if((d==Directions.UP)||(d==Directions.DOWN)||(d==Directions.GATE))
continue;
final Room chkR=R.getRoomInDir(d);
if((chkR==null)&&(numberOfPeers < 0))
{
final List<Room> allRooms = getConnectedPropertyRooms();
if(allRooms.size()>0)
{
cap = (Prop_ReqCapacity)allRooms.get(0).fetchEffect("Prop_ReqCapacity");
if(cap != null)
{
roomLimit = cap.roomLimit;
}
}
numberOfPeers = allRooms.size();
}
if((chkR==null)&&(numberOfPeers < roomLimit))
{
numberOfPeers++;
final Room R2=CMClass.getLocale(CMClass.classID(R));
R2.setRoomID(R.getArea().getNewRoomID(R,d));
if(R2.roomID().length()==0)
continue;
R2.setArea(R.getArea());
final LandTitle oldTitle=CMLib.law().getLandTitle(R);
final LandTitle newTitle;
if((oldTitle!=null)&&(CMLib.law().getLandTitle(R2)==null))
{
newTitle = oldTitle.generateNextRoomTitle();
R2.addNonUninvokableEffect((Ability)newTitle);
}
else
newTitle=null;
R.rawDoors()[d]=R2;
R.setRawExit(d,CMClass.getExit("Open"));
R2.rawDoors()[Directions.getOpDirectionCode(d)]=R;
R2.setRawExit(Directions.getOpDirectionCode(d),CMClass.getExit("Open"));
updateExits.add(R);
if(doGrid)
{
final PairVector<Room,int[]> rooms=CMLib.tracking().buildGridList(R2, this.getOwnerName(), 100);
for(int dir=0;dir<Directions.NUM_DIRECTIONS();dir++)
{
if(dir==Directions.GATE)
continue;
Room R3=R2.getRoomInDir(dir);
if(R3 == null)
{
R3=CMLib.tracking().getCalculatedAdjacentRoom(rooms, R3, dir);
if(R3!=null)
{
R2.rawDoors()[dir]=R3;
R3.rawDoors()[Directions.getOpDirectionCode(dir)]=R2;
updateExits.add(R3);
}
}
}
}
updateExits.add(R2);
if(CMSecurity.isDebugging(CMSecurity.DbgFlag.PROPERTY))
Log.debugOut("Lots4Sale",R2.roomID()+" created and put up for sale.");
if(cap != null)
R2.addNonUninvokableEffect((Ability)cap.copyOf());
postWork.add(new Runnable()
{
final Room room = R2;
final LandTitle title=newTitle;
@Override
public void run()
{
CMLib.database().DBCreateRoom(room);
if(title!=null)
CMLib.law().colorRoomForSale(room,title,true);
room.getArea().fillInAreaRoom(room);
final MOB mob=CMClass.getFactoryMOB("the wind",1,room);
try
{
room.executeMsg(mob, CMClass.getMsg(mob, room, CMMsg.MSG_NEWROOM, null));
}
finally
{
mob.destroy();
}
}
});
didAnything=true;
}
}
if(updateExits.size()>0)
{
didAnything=true;
R.getArea().fillInAreaRoom(R);
postWork.add(new Runnable()
{
final Set<Room> updateExits2=new SHashSet<Room>(updateExits);
@Override
public void run()
{
for(final Room xR : updateExits2)
{
CMLib.database().DBUpdateExits(xR);
}
}
});
}
return didAnything;
}
@Override
public void updateLot(final Set<String> optPlayerList)
{
final Environmental EV=affected;
if(!(EV instanceof Room))
return;
Room R=(Room)EV;
boolean didAnything=false;
try
{
final List<Runnable> postWork=new ArrayList<Runnable>();
synchronized(("SYNC"+R.roomID()).intern())
{
R=CMLib.map().getRoom(R);
final int[] data=updateLotWithThisData(R,this,true,scheduleReset,optPlayerList,lastItemNums,daysWithNoChange);
lastItemNums=data[0];
daysWithNoChange=data[1];
if(getOwnerName().length()==0)
{
didAnything = retractRooms(R,postWork) || didAnything;
}
else
if(canGenerateAdjacentRooms(R))
{
didAnything = expandRooms(R,postWork) || didAnything;
}
}
for(final Runnable run : postWork)
run.run();
scheduleReset=false;
}
finally
{
if(didAnything)
getConnectedPropertyRooms(); // recalculates the unique id for this lot of rooms
}
}
}