forked from MagicBane/Server
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1381 lines
42 KiB
1381 lines
42 KiB
3 years ago
|
package engine.net.client.handlers;
|
||
|
|
||
|
import engine.Enum;
|
||
|
import engine.Enum.*;
|
||
|
import engine.InterestManagement.HeightMap;
|
||
|
import engine.InterestManagement.RealmMap;
|
||
|
import engine.InterestManagement.WorldGrid;
|
||
|
import engine.db.archive.CityRecord;
|
||
|
import engine.db.archive.DataWarehouse;
|
||
|
import engine.exception.MsgSendException;
|
||
|
import engine.gameManager.*;
|
||
|
import engine.math.Bounds;
|
||
|
import engine.math.Vector3fImmutable;
|
||
|
import engine.net.Dispatch;
|
||
|
import engine.net.DispatchMessage;
|
||
|
import engine.net.client.ClientConnection;
|
||
|
import engine.net.client.msg.CityZoneMsg;
|
||
|
import engine.net.client.msg.ClientNetMsg;
|
||
|
import engine.net.client.msg.ErrorPopupMsg;
|
||
|
import engine.net.client.msg.PlaceAssetMsg;
|
||
|
import engine.net.client.msg.PlaceAssetMsg.PlacementInfo;
|
||
|
import engine.objects.*;
|
||
|
import engine.server.MBServerStatics;
|
||
|
import org.joda.time.DateTime;
|
||
|
import org.pmw.tinylog.Logger;
|
||
|
|
||
|
import java.time.LocalDateTime;
|
||
|
import java.util.ArrayList;
|
||
|
import java.util.HashSet;
|
||
|
import java.util.concurrent.locks.ReentrantReadWriteLock;
|
||
|
|
||
|
/*
|
||
|
* @Summary: Processes application protocol message which requests
|
||
|
* creation of new city / buildings from seeds/deeds in inventory.
|
||
|
*/
|
||
|
public class PlaceAssetMsgHandler extends AbstractClientMsgHandler {
|
||
|
|
||
|
// Useful constants
|
||
|
// ActionType 1 = client request
|
||
|
// 2 = Server confirms open window
|
||
|
// 3 = Request to place asset
|
||
|
// 4 = Server confirms/close window
|
||
|
private static final int CLIENTREQ_UNKNOWN = 1;
|
||
|
private static final int SERVER_OPENWINDOW = 2;
|
||
|
private static final int CLIENTREQ_NEWBUILDING = 3; // Request to place asset
|
||
|
private static final int SERVER_CLOSEWINDOW = 4;
|
||
|
|
||
|
private final ReentrantReadWriteLock lock = new ReentrantReadWriteLock();
|
||
|
|
||
|
public PlaceAssetMsgHandler() {
|
||
|
|
||
|
super(PlaceAssetMsg.class);
|
||
|
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
protected boolean _handleNetMsg(ClientNetMsg baseMsg, ClientConnection origin) throws MsgSendException {
|
||
|
|
||
|
// Member variable declaration
|
||
|
|
||
|
PlaceAssetMsg msg;
|
||
|
Boolean buildingCreated;
|
||
|
|
||
|
// Character location and session
|
||
|
|
||
|
PlayerCharacter playerCharacter;
|
||
|
PlacementInfo buildingList;
|
||
|
Blueprint buildingBlueprint;
|
||
|
|
||
|
// Tell compiler it's ok to trust us and parse
|
||
|
// what we need from the message structure
|
||
|
|
||
|
msg = (PlaceAssetMsg) baseMsg;
|
||
|
|
||
|
// Action type 3 is a client requesting to place an object
|
||
|
// For all other action types let's just early exit
|
||
|
|
||
|
if (msg.getActionType() != CLIENTREQ_NEWBUILDING)
|
||
|
return true;
|
||
|
|
||
|
// assign our character
|
||
|
|
||
|
playerCharacter = SessionManager.getPlayerCharacter(origin);
|
||
|
|
||
|
// We need to figure out what exactly the player is attempting
|
||
|
// to place, as some objects like tol/bane/walls are edge cases.
|
||
|
// So let's get the first item in their list.
|
||
|
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
// Early exit if null building list.
|
||
|
|
||
|
if (buildingList == null) {
|
||
|
Logger.error("Player " + playerCharacter.getCombinedName()
|
||
|
+ " null building list on deed use");
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
closePlaceAssetWindow(origin);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
Item contract = null;
|
||
|
|
||
|
for (Item inventoryItem : playerCharacter.getInventory()) {
|
||
|
if (inventoryItem.getItemBase().getUseID() == buildingList.getBlueprintUUID()) {
|
||
|
contract = inventoryItem;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Grab the blueprint from the uuid in the message
|
||
|
|
||
|
buildingBlueprint = Blueprint.getBlueprint(buildingList.getBlueprintUUID());
|
||
|
|
||
|
// Early exit if blueprint can't be retrieved for the object.
|
||
|
|
||
|
if (buildingBlueprint == null) {
|
||
|
Logger.error("Player " + playerCharacter.getCombinedName()
|
||
|
+ " null blueprint UUID: " + buildingList.getBlueprintUUID() + " on deed use");
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
closePlaceAssetWindow(origin);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Let's now attempt to place the building
|
||
|
|
||
|
buildingCreated = false;
|
||
|
|
||
|
// Many buildings have particular validation and
|
||
|
// post-creation cleanup requirements.
|
||
|
|
||
|
boolean close = true;
|
||
|
lock.writeLock().lock();
|
||
|
|
||
|
try {
|
||
|
switch (buildingBlueprint.getBuildingGroup()) {
|
||
|
|
||
|
case TOL:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
buildingCreated = placeTreeOfLife(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
case WAREHOUSE:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
if (!playerCharacter.getCharItemManager().doesCharOwnThisItem(contract.getObjectUUID()))
|
||
|
break;
|
||
|
buildingCreated = placeWarehouse(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
case SIEGETENT:
|
||
|
case BULWARK:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
if (!playerCharacter.getCharItemManager().doesCharOwnThisItem(contract.getObjectUUID()))
|
||
|
break;
|
||
|
buildingCreated = placeSiegeEquip(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
case SPIRE:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
if (!playerCharacter.getCharItemManager().doesCharOwnThisItem(contract.getObjectUUID()))
|
||
|
break;
|
||
|
buildingCreated = placeSpire(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
case SHRINE:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
if (!playerCharacter.getCharItemManager().doesCharOwnThisItem(contract.getObjectUUID()))
|
||
|
break;
|
||
|
buildingCreated = placeShrine(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
case BARRACK:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
if (!playerCharacter.getCharItemManager().doesCharOwnThisItem(contract.getObjectUUID()))
|
||
|
break;
|
||
|
buildingCreated = placeBarrack(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
case WALLSTRAIGHT:
|
||
|
case WALLCORNER:
|
||
|
case SMALLGATE:
|
||
|
case ARTYTOWER:
|
||
|
case WALLSTAIRS:
|
||
|
buildingCreated = placeCityWalls(playerCharacter, origin, msg);
|
||
|
close = false;
|
||
|
break;
|
||
|
default:
|
||
|
if (contract == null)
|
||
|
break;
|
||
|
if (!playerCharacter.getCharItemManager().doesCharOwnThisItem(contract.getObjectUUID()))
|
||
|
break;
|
||
|
buildingCreated = placeSingleBuilding(playerCharacter, origin, msg);
|
||
|
break;
|
||
|
}
|
||
|
} catch (Exception e) {
|
||
|
Logger.error("PlaceAssetHandler", e.getMessage());
|
||
|
e.printStackTrace();
|
||
|
} finally {
|
||
|
lock.writeLock().unlock();
|
||
|
}
|
||
|
|
||
|
// Update the player's last contract (What is this used for?)
|
||
|
|
||
|
playerCharacter.setLastContract(msg.getContractID());
|
||
|
|
||
|
// Remove the appropiate deed.
|
||
|
if (buildingCreated == true)
|
||
|
if (contract != null) {
|
||
|
playerCharacter.getCharItemManager().delete(contract);
|
||
|
playerCharacter.getCharItemManager().updateInventory();
|
||
|
}
|
||
|
|
||
|
// Close the window. We're done!
|
||
|
//DONT CLOSE THE WINDOW IF WALL KTHANX
|
||
|
|
||
|
if (close)
|
||
|
closePlaceAssetWindow(origin);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Default method: Validates and places all buildings that do not
|
||
|
// require special treatment in some fashion.
|
||
|
|
||
|
private boolean placeSingleBuilding(PlayerCharacter playerCharacter, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
PlacementInfo buildingList;
|
||
|
Zone serverZone;
|
||
|
|
||
|
// Retrieve the building details we're placing
|
||
|
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(buildingList.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
// with locating the current or zone
|
||
|
|
||
|
if (serverZone == null) {
|
||
|
Logger.error("Null zone in placeSingleBuilding");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// buildings. Player must be on a city grid, must be
|
||
|
// inner council of the city's guild, etc.
|
||
|
|
||
|
if (validateBuildingPlacement(serverZone, msg, origin, playerCharacter, buildingList) == false)
|
||
|
return false; // Close window here?
|
||
|
|
||
|
// Place the building
|
||
|
if (createStructure(playerCharacter, buildingList, serverZone) == null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private boolean placeWarehouse(PlayerCharacter player, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
Zone serverZone;
|
||
|
City cityObject;
|
||
|
PlacementInfo buildingList;
|
||
|
|
||
|
// Retrieve the building details we're placing
|
||
|
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(buildingList.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
|
||
|
if (serverZone == null)
|
||
|
return false;
|
||
|
|
||
|
cityObject = City.getCity(serverZone.getPlayerCityUUID());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
|
||
|
if (cityObject == null)
|
||
|
return false;
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// buildings. Player must be on a city grid, must be
|
||
|
// inner council of the city's guild, etc.
|
||
|
|
||
|
if (validateCityBuildingPlacement(serverZone, msg, origin, player, buildingList) == false)
|
||
|
return false;
|
||
|
|
||
|
if (cityObject.getWarehouse() != null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 50, ""); //"You can only have one warehouse"
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Create the warehouse object and it's entry in the database
|
||
|
|
||
|
if (createWarehouse(player, msg.getFirstPlacementInfo(), serverZone) == false) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private boolean placeSiegeEquip(PlayerCharacter player, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
Zone serverZone;
|
||
|
Building siegeBuilding;
|
||
|
PlacementInfo buildingList;
|
||
|
City serverCity;
|
||
|
int numAttackerBuildings = 0;
|
||
|
int numDefenderBuildings = 0;
|
||
|
|
||
|
// Retrieve the building details we're placing
|
||
|
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(buildingList.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
// with locating the current city and/or zone
|
||
|
|
||
|
if (serverZone == null) {
|
||
|
Logger.error("Error obtaining reference to zone");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Checks validation conditions arising when placing
|
||
|
// generic structures.
|
||
|
|
||
|
if (validateBuildingPlacement(serverZone, msg, origin, player, buildingList) == false)
|
||
|
return false;
|
||
|
|
||
|
// If there is a bane placed, only the attackers and defenders can
|
||
|
// place siege assets
|
||
|
|
||
|
serverCity = ZoneManager.getCityAtLocation(buildingList.getLoc());
|
||
|
|
||
|
//no city found
|
||
|
//check if attacker city.
|
||
|
if (serverCity == null){
|
||
|
Bane bane = Bane.getBaneByAttackerGuild(player.getGuild());
|
||
|
City attackerCity = null;
|
||
|
if (bane != null)
|
||
|
attackerCity = bane.getCity();
|
||
|
|
||
|
if (attackerCity != null)
|
||
|
if (buildingList.getLoc().isInsideCircle(attackerCity.getLoc(), Enum.CityBoundsType.SIEGE.extents))
|
||
|
serverCity = attackerCity;
|
||
|
}
|
||
|
//no city found for attacker city,
|
||
|
//check if defender city
|
||
|
|
||
|
if (serverCity == null){
|
||
|
if (player.getGuild().getOwnedCity() != null)
|
||
|
if (buildingList.getLoc().isInsideCircle(player.getGuild().getOwnedCity().getLoc(), Enum.CityBoundsType.SIEGE.extents))
|
||
|
serverCity = player.getGuild().getOwnedCity();
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
if ((serverCity != null) &&
|
||
|
(serverCity.getBane() != null)) {
|
||
|
|
||
|
// Set the server zone to the city zone in order to account for being inside
|
||
|
// the siege bounds buffer area
|
||
|
|
||
|
serverZone = serverCity.getParent();
|
||
|
|
||
|
if ((player.getGuild().equals(serverCity.getBane().getOwner().getGuild()) == false)
|
||
|
&& (player.getGuild().equals(serverCity.getGuild()) == false)) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 54, ""); // Must belong to attacker or defender
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// cant place siege equipment off city zone.
|
||
|
|
||
|
|
||
|
// Create the siege Building
|
||
|
|
||
|
siegeBuilding = createStructure(player, msg.getFirstPlacementInfo(), serverZone);
|
||
|
if (serverCity == null)
|
||
|
return true;
|
||
|
// Oops something went really wrong
|
||
|
|
||
|
if (siegeBuilding == null)
|
||
|
return false;
|
||
|
|
||
|
|
||
|
if (serverCity.getBane() == null)
|
||
|
return true;
|
||
|
|
||
|
// If there is an bane placed, we protect 2x the stone rank's worth of attacker assets
|
||
|
// and 1x the tree rank's worth of assets automatically
|
||
|
|
||
|
HashSet<AbstractWorldObject> awoList = WorldGrid.getObjectsInRangePartial(serverCity, 1000, MBServerStatics.MASK_BUILDING);
|
||
|
|
||
|
|
||
|
|
||
|
for (AbstractWorldObject awo : awoList) {
|
||
|
Building building = (Building)awo;
|
||
|
|
||
|
if (building.getBlueprint() != null)
|
||
|
if (!building.getBlueprint().isSiegeEquip())
|
||
|
continue;
|
||
|
|
||
|
if (!building.getLoc().isInsideCircle(serverCity.getLoc(), Enum.CityBoundsType.SIEGE.extents))
|
||
|
continue;
|
||
|
|
||
|
if (building.getGuild() == null)
|
||
|
continue;
|
||
|
|
||
|
if (building.getGuild().isErrant())
|
||
|
continue;
|
||
|
|
||
|
|
||
|
if (!building.getGuild().equals(serverCity.getGuild()) && !building.getGuild().equals(serverCity.getBane().getOwner().getGuild()))
|
||
|
continue;
|
||
|
|
||
|
// Only count auto protected buildings
|
||
|
if (building.getProtectionState() != ProtectionState.PROTECTED)
|
||
|
continue;
|
||
|
|
||
|
if (building.getGuild().equals(serverCity.getGuild()))
|
||
|
numDefenderBuildings++;
|
||
|
else
|
||
|
if (building.getGuild().equals(serverCity.getBane().getOwner().getGuild()))
|
||
|
numAttackerBuildings++;
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
// Validate bane limits on siege assets
|
||
|
|
||
|
if (serverCity.getBane() != null)
|
||
|
if ((player.getGuild().equals(serverCity.getBane().getOwner().getGuild())) &&
|
||
|
(numAttackerBuildings >= serverCity.getBane().getStone().getRank() * 2)) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if ((player.getGuild().equals(serverCity.getGuild())) &&
|
||
|
(numDefenderBuildings >= serverCity.getTOL().getRank())) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
// passes validation: can assign auto-protection to war asset
|
||
|
|
||
|
if (serverCity.getBane() != null)
|
||
|
if (serverCity.isLocationOnCityGrid(siegeBuilding.getBounds()))
|
||
|
if (player.getGuild().equals(serverCity.getBane().getOwner().getGuild()))
|
||
|
return true;
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
siegeBuilding.setProtectionState(ProtectionState.PROTECTED);
|
||
|
// No bane placed. We're done!
|
||
|
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private boolean placeTreeOfLife(PlayerCharacter playerCharacter, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
Realm serverRealm;
|
||
|
Zone serverZone;
|
||
|
ArrayList<AbstractGameObject> cityObjects; // MySql result set
|
||
|
PlacementInfo treeInfo;
|
||
|
Building treeObject = null;
|
||
|
City cityObject = null;
|
||
|
Zone cityZone = null;
|
||
|
Guild playerNation;
|
||
|
PlacementInfo treePlacement = msg.getFirstPlacementInfo();
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
|
||
|
serverRealm = RealmMap.getRealmAtLocation(treePlacement.getLoc());
|
||
|
serverZone = ZoneManager.findSmallestZone(treePlacement.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
// with locating the current realm and/or zone
|
||
|
|
||
|
if (serverRealm == null || serverZone == null)
|
||
|
return false;
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// trees
|
||
|
|
||
|
if (validateTreeOfLifePlacement(playerCharacter, serverRealm, serverZone, origin, msg) == false)
|
||
|
return false;
|
||
|
|
||
|
// Retrieve tree info for the w value it's passing.
|
||
|
|
||
|
treeInfo = msg.getFirstPlacementInfo();
|
||
|
|
||
|
if (treeInfo == null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
Vector3fImmutable plantLoc = new Vector3fImmutable(treeInfo.getLoc().x,
|
||
|
serverZone.getHeightMap().getInterpolatedTerrainHeight(treeInfo.getLoc()),
|
||
|
treeInfo.getLoc().z);
|
||
|
|
||
|
cityObjects = DbManager.CityQueries.CREATE_CITY(playerCharacter.getObjectUUID(), serverZone.getObjectUUID(),
|
||
|
serverRealm.getRealmID(),
|
||
|
plantLoc.x - serverZone.getAbsX(), plantLoc.y,
|
||
|
plantLoc.z - serverZone.getAbsZ(), treeInfo.getRot().y, treeInfo.getW(), playerCharacter.getGuild().getName(), LocalDateTime.now());
|
||
|
|
||
|
// Uh oh!
|
||
|
if (cityObjects == null || cityObjects.isEmpty()) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Assign our worker variables after figuring out what
|
||
|
// is what in the result set.
|
||
|
|
||
|
for (AbstractGameObject gameObject : cityObjects) {
|
||
|
|
||
|
switch (gameObject.getObjectType()) {
|
||
|
case Building:
|
||
|
treeObject = (Building) gameObject;
|
||
|
treeObject.runAfterLoad();
|
||
|
break;
|
||
|
case City:
|
||
|
cityObject = (City) gameObject;
|
||
|
break;
|
||
|
case Zone:
|
||
|
cityZone = (Zone) gameObject;
|
||
|
break;
|
||
|
default:
|
||
|
// log some error here? *** Refactor
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//?? your not allowed to plant a tree if ur not an errant guild.
|
||
|
// Desub from any previous nation.
|
||
|
// This should be done automatically in a method inside Guild *** Refactor
|
||
|
// Player is now a Soverign guild, configure them as such.
|
||
|
|
||
|
playerCharacter.getGuild().setNation(playerCharacter.getGuild());
|
||
|
playerNation = playerCharacter.getGuild();
|
||
|
playerNation.setGuildState(GuildState.Sovereign);
|
||
|
|
||
|
// Link the zone with the city and then add
|
||
|
// to the appropritae hash tables and cache
|
||
|
|
||
|
cityZone.setPlayerCity(true);
|
||
|
|
||
|
if (cityZone.getParent() != null)
|
||
|
cityZone.getParent().addNode(cityZone); //add as child to parent
|
||
|
|
||
|
ZoneManager.addZone(cityZone.getObjectUUID(), cityZone);
|
||
|
ZoneManager.addPlayerCityZone(cityZone);
|
||
|
serverZone.addNode(cityZone);
|
||
|
|
||
|
cityZone.generateWorldAltitude();
|
||
|
|
||
|
cityObject.setParent(cityZone);
|
||
|
cityObject.setObjectTypeMask(MBServerStatics.MASK_CITY); // *** Refactor : should have it already
|
||
|
//Link the tree of life with the new zone
|
||
|
|
||
|
treeObject.setObjectTypeMask(MBServerStatics.MASK_BUILDING);
|
||
|
treeObject.setParentZone(cityZone);
|
||
|
MaintenanceManager.setMaintDateTime(treeObject, LocalDateTime.now().plusDays(7));
|
||
|
|
||
|
// Update guild binds and tags
|
||
|
//load the new city on the clients
|
||
|
|
||
|
CityZoneMsg czm = new CityZoneMsg(1, treeObject.getLoc().x, treeObject.getLoc().y, treeObject.getLoc().z, cityObject.getCityName(), cityZone, Enum.CityBoundsType.ZONE.extents, Enum.CityBoundsType.ZONE.extents);
|
||
|
DispatchMessage.dispatchMsgToAll(czm);
|
||
|
|
||
|
GuildManager.updateAllGuildBinds(playerNation, cityObject);
|
||
|
GuildManager.updateAllGuildTags(playerNation);
|
||
|
|
||
|
// Send all the cities to the clients?
|
||
|
// *** Refactor : figure out how to send like, one?
|
||
|
|
||
|
City.lastCityUpdate = System.currentTimeMillis();
|
||
|
WorldGrid.addObject(treeObject, playerCharacter);
|
||
|
|
||
|
serverRealm.addCity(cityObject.getObjectUUID());
|
||
|
playerNation.setCityUUID(cityObject.getObjectUUID());
|
||
|
|
||
|
// Bypass warehouse entry if we're an admin
|
||
|
|
||
|
if (playerCharacter.getAccount().status.equals(AccountStatus.ADMIN))
|
||
|
return true;
|
||
|
|
||
|
// Push this event to the data warehouse
|
||
|
|
||
|
CityRecord cityRecord = CityRecord.borrow(cityObject, RecordEventType.CREATE);
|
||
|
DataWarehouse.pushToWarehouse(cityRecord);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private boolean placeSpire(PlayerCharacter playerCharacter, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
Zone serverZone;
|
||
|
Building spireBuilding;
|
||
|
Blueprint blueprint;
|
||
|
City cityObject;
|
||
|
PlacementInfo buildingList;
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(buildingList.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
// with locating the current realm and/or city
|
||
|
|
||
|
if (serverZone == null)
|
||
|
return false;
|
||
|
|
||
|
cityObject = City.getCity(serverZone.getPlayerCityUUID());
|
||
|
|
||
|
if (cityObject == null)
|
||
|
return false;
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// buildings. Player must be on a city grid, must be
|
||
|
// inner council of the city's guild, etc.
|
||
|
|
||
|
if (validateCityBuildingPlacement(serverZone, msg, origin, playerCharacter, buildingList) == false)
|
||
|
return false;
|
||
|
|
||
|
// Loop through all buildings in this city looking for a spire of the.
|
||
|
// same type we are placing. There can be only one of each type
|
||
|
|
||
|
int spireCount = 0;
|
||
|
|
||
|
blueprint = Blueprint.getBlueprint(msg.getFirstPlacementInfo().getBlueprintUUID());
|
||
|
|
||
|
for (Building building : serverZone.zoneBuildingSet) {
|
||
|
|
||
|
if (building.getBlueprint().getBuildingGroup() == BuildingGroup.SPIRE) {
|
||
|
|
||
|
if (building.getBlueprintUUID() == blueprint.getMeshForRank(0)) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 46, ""); // "Spire of that type exists"
|
||
|
return false;
|
||
|
}
|
||
|
spireCount++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Too many spires for this tree's rank?
|
||
|
|
||
|
if (spireCount >= Blueprint.getMaxShrines(cityObject.getTOL().getRank())) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 45, ""); //Tree cannot support anymore spires
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Create the spire
|
||
|
|
||
|
spireBuilding = createStructure(playerCharacter, msg.getFirstPlacementInfo(), serverZone);
|
||
|
return spireBuilding != null;
|
||
|
}
|
||
|
|
||
|
private boolean placeShrine(PlayerCharacter playerCharacter, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
Zone serverZone;
|
||
|
Blueprint blueprint;
|
||
|
City cityObject;
|
||
|
PlacementInfo buildingList;
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(buildingList.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
// with locating the current realm and/or zone
|
||
|
if (serverZone == null)
|
||
|
return false;
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// buildings. Player must be on a city grid, must be
|
||
|
// inner council of the city's guild, etc.
|
||
|
|
||
|
if (validateCityBuildingPlacement(serverZone, msg, origin, playerCharacter, buildingList) == false)
|
||
|
return false;
|
||
|
|
||
|
// Loop through all buildings in this city looking for a shrine of the.
|
||
|
// same type we are placing. There can be only one of each type
|
||
|
|
||
|
int shrineCount = 0;
|
||
|
|
||
|
cityObject = City.getCity(serverZone.getPlayerCityUUID());
|
||
|
|
||
|
// Cannot place shrine in abanadoned city. Shrines must be owned
|
||
|
// by the tol owner not the person placing them.
|
||
|
|
||
|
if (cityObject.getTOL().getOwnerUUID() == 0) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 42, ""); //Tree cannot support anymore shrines
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
blueprint = Blueprint.getBlueprint(msg.getFirstPlacementInfo().getBlueprintUUID());
|
||
|
|
||
|
if (blueprint == null){
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
for (Building building : serverZone.zoneBuildingSet) {
|
||
|
if (building.getBlueprint() == null)
|
||
|
continue;
|
||
|
|
||
|
if (building.getBlueprint().getBuildingGroup() == BuildingGroup.SHRINE) {
|
||
|
if (building.getBlueprintUUID() == blueprint.getMeshForRank(0)) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 43, ""); // "shrine of that type exists"
|
||
|
return false;
|
||
|
}
|
||
|
shrineCount++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Too many shrines for this tree's rank?
|
||
|
|
||
|
if (shrineCount >= Blueprint.getMaxShrines(cityObject.getTOL().getRank())) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 42, ""); //Tree cannot support anymore shrines
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Create the shrine
|
||
|
|
||
|
return createShrine((PlayerCharacter)cityObject.getTOL().getOwner(), msg.getFirstPlacementInfo(), serverZone);
|
||
|
}
|
||
|
|
||
|
private boolean placeBarrack(PlayerCharacter playerCharacter, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
Zone serverZone;
|
||
|
City cityObject;
|
||
|
PlacementInfo buildingList;
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
buildingList = msg.getFirstPlacementInfo();
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(buildingList.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
// with locating the current realm and/or zone
|
||
|
|
||
|
if (serverZone == null)
|
||
|
return false;
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// buildings. Player must be on a city grid, must be
|
||
|
// inner council of the city's guild, etc.
|
||
|
|
||
|
if (validateCityBuildingPlacement(serverZone, msg, origin, playerCharacter, buildingList) == false)
|
||
|
return false;
|
||
|
|
||
|
// Loop through all buildings in this city counting barracks .
|
||
|
|
||
|
int barracksCount = 0;
|
||
|
|
||
|
cityObject = City.getCity(serverZone.getPlayerCityUUID());
|
||
|
|
||
|
// Cannot place barracks in abanadoned city.
|
||
|
|
||
|
if (cityObject.getTOL().getOwnerUUID() == 0) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 42, ""); //Tree cannot support anymore shrines
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
for (Building building : serverZone.zoneBuildingSet) {
|
||
|
if (building.getBlueprint().getBuildingGroup() == BuildingGroup.BARRACK)
|
||
|
barracksCount++;
|
||
|
}
|
||
|
|
||
|
// Too many shrines for this tree's rank?
|
||
|
|
||
|
if (barracksCount >= cityObject.getTOL().getRank()) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 47, ""); //Tree cannot support anymore shrines
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Create the shrine
|
||
|
|
||
|
return createBarracks((PlayerCharacter)cityObject.getTOL().getOwner(), msg.getFirstPlacementInfo(), serverZone);
|
||
|
}
|
||
|
|
||
|
private boolean placeCityWalls(PlayerCharacter player, ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
// Member variables
|
||
|
|
||
|
Zone serverZone;
|
||
|
City cityObject;
|
||
|
int placementCost = 0;
|
||
|
CharacterItemManager itemMan;
|
||
|
Item goldItem;
|
||
|
Building wallPiece;
|
||
|
|
||
|
// Setup working variables we'll need
|
||
|
|
||
|
serverZone = ZoneManager.findSmallestZone(player.getLoc());
|
||
|
|
||
|
// Early exit if something went horribly wrong
|
||
|
|
||
|
if (serverZone == null)
|
||
|
return false;
|
||
|
|
||
|
|
||
|
if (player.getCharItemManager().getGoldTrading() > 0){
|
||
|
ErrorPopupMsg.sendErrorPopup(player, 195);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
|
||
|
// Method checks validation conditions arising when placing
|
||
|
// buildings. Player must be on a city grid, must be
|
||
|
// inner council of the city's guild, etc.
|
||
|
|
||
|
if (validateCityBuildingPlacement(serverZone, msg, origin, player, msg.getFirstPlacementInfo()) == false)
|
||
|
return false;
|
||
|
|
||
|
cityObject = City.getCity(serverZone.getPlayerCityUUID());
|
||
|
|
||
|
// We need to be able to access how much gold a character is carrying
|
||
|
|
||
|
itemMan = player.getCharItemManager();
|
||
|
|
||
|
if (itemMan == null)
|
||
|
|
||
|
return false;
|
||
|
|
||
|
goldItem = itemMan.getGoldInventory();
|
||
|
|
||
|
// Grab list of walls we're placing
|
||
|
|
||
|
ArrayList<PlacementInfo> walls = msg.getPlacementInfo();
|
||
|
|
||
|
// Character must be able to afford walls
|
||
|
|
||
|
for (PlacementInfo wall : walls) {
|
||
|
placementCost += PlaceAssetMsg.getWallCost(wall.getBlueprintUUID());
|
||
|
}
|
||
|
|
||
|
// Early exit if not enough gold in character's inventory to place walls
|
||
|
|
||
|
if (placementCost > goldItem.getNumOfItems()) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 28, ""); // Not enough gold
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
placementCost = 0; // reset placement cost for fix bug with wall pieces somethings not taking gold out if forced an error.
|
||
|
|
||
|
|
||
|
// Overlap check and wall deed verifications
|
||
|
for (PlacementInfo wall : walls) {
|
||
|
|
||
|
if (Blueprint.isMeshWallPiece(wall.getBlueprintUUID()) == false) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 48, ""); //"Assets (except walls) must be placed one at a time"
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Ignore wall pieces not on the city grid
|
||
|
if (cityObject.isLocationOnCityGrid(wall.getLoc()) == false) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "Asset " + cityObject.getName() + " not on citygrid");
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// Does this wall collide with any other building?
|
||
|
|
||
|
for (Building building : serverZone.zoneBuildingSet) {
|
||
|
|
||
|
|
||
|
//TODO Clean up collision with placementInfo. don't need to create the same placementinfo bounds for collision checks on each building.
|
||
|
if ((building.getBlueprintUUID() != 0) && (Bounds.collide(wall, building) == true)) {
|
||
|
|
||
|
if (building.getRank() == -1) {
|
||
|
building.removeFromCache();
|
||
|
WorldGrid.RemoveWorldObject(building);
|
||
|
WorldGrid.removeObject(building);
|
||
|
building.getParentZone().getParent().zoneBuildingSet.remove(building);
|
||
|
continue;
|
||
|
}
|
||
|
// remove gold from walls already placed before returning.
|
||
|
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 3, building.getName()); //"Conflict between assets"
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
placementCost = PlaceAssetMsg.getWallCost(wall.getBlueprintUUID());
|
||
|
|
||
|
if (!itemMan.modifyInventoryGold(-placementCost)){
|
||
|
ChatManager.chatSystemInfo(player, player.getFirstName() + " can't has free moneys! no for real.. Thor.. seriously... I didnt fix it because you getting laid isnt important enough for me.");
|
||
|
return false;
|
||
|
}
|
||
|
// Attempt to place wall piece
|
||
|
|
||
|
wallPiece = createStructure(player, wall, serverZone);
|
||
|
|
||
|
if (wallPiece == null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// walls are auto protected
|
||
|
wallPiece.setProtectionState(ProtectionState.PROTECTED);
|
||
|
PlaceAssetMsg.sendPlaceAssetConfirmWall(origin,serverZone);
|
||
|
|
||
|
}
|
||
|
|
||
|
// Deduct gold from character's inventory
|
||
|
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private static void closePlaceAssetWindow(ClientConnection origin) {
|
||
|
|
||
|
// Action type 4 is the server telling the client to
|
||
|
// close the asset placement window.
|
||
|
// This is believed to be a confirmation message to the client
|
||
|
PlaceAssetMsg pam = new PlaceAssetMsg();
|
||
|
pam.setActionType(4);
|
||
|
Dispatch dispatch = Dispatch.borrow(origin.getPlayerCharacter(), pam);
|
||
|
DispatchMessage.dispatchMsgDispatch(dispatch, DispatchChannel.SECONDARY);
|
||
|
}
|
||
|
|
||
|
// Method deletes one item from the player's inventory
|
||
|
// based on the mesh UUID the deed/seed spawns
|
||
|
|
||
|
private static void removeDeedByMeshUUID(PlayerCharacter player, int meshUUID) {
|
||
|
|
||
|
CharacterItemManager inventoryManager;
|
||
|
ArrayList<Item> itemList;
|
||
|
|
||
|
inventoryManager = player.getCharItemManager();
|
||
|
itemList = player.getInventory();
|
||
|
|
||
|
for (Item inventoryItem : itemList) {
|
||
|
if (inventoryItem.getItemBase().getUseID() == meshUUID) {
|
||
|
inventoryManager.delete(inventoryItem);
|
||
|
|
||
|
inventoryManager.updateInventory();
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Method validates the location we have selected for our new city
|
||
|
|
||
|
private static boolean validateTreeOfLifePlacement(PlayerCharacter playerCharacter, Realm serverRealm, Zone serverZone,
|
||
|
ClientConnection origin, PlaceAssetMsg msg) {
|
||
|
|
||
|
PlacementInfo placementInfo = msg.getFirstPlacementInfo();
|
||
|
|
||
|
// Your guild already owns a tree
|
||
|
|
||
|
if (playerCharacter.getGuild().getOwnedCity() != null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "Your guild already owns a tree!");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Validate that the player is the leader of a guild
|
||
|
|
||
|
if (GuildStatusController.isGuildLeader(playerCharacter.getGuildStatus()) == false) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 10, ""); // Must be a guild leader
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Validate that the player is the leader of a guild
|
||
|
// that is not currently Sovereign *** BUG? Doesn't look right. isGuildLeader()?
|
||
|
|
||
|
if ((playerCharacter.getGuild().getGuildState() != GuildState.Sworn
|
||
|
|| playerCharacter.getGuild().getGuildState() != GuildState.Errant) == false) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 17, ""); // Your is not an errant or soverign guild
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// All trees must be placed within a continent.
|
||
|
|
||
|
if (!serverZone.isContininent()) {
|
||
|
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 69, ""); // Tree must be within a territory
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
RealmType realmType = RealmType.getRealmTypeByUUID(serverRealm.getRealmID());
|
||
|
|
||
|
if (
|
||
|
(realmType.equals(RealmType.MAELSTROM)) ||
|
||
|
(realmType.equals(RealmType.OBLIVION))) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 57, playerCharacter.getName()); // No building may be placed within this territory
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Cannot place a tree underwater
|
||
|
|
||
|
if (HeightMap.isLocUnderwater(placementInfo.getLoc())) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 6, ""); // Cannot place underwater
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
//Test city not too close to any other zone
|
||
|
|
||
|
if (!ZoneManager.validTreePlacementLoc(serverZone, placementInfo.getLoc().x, placementInfo.getLoc().z)) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 39, ""); // Too close to another tree
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Validate that Realm is not at it's city limit
|
||
|
|
||
|
if (serverRealm.isRealmFull() == true) {
|
||
|
int numCities;
|
||
|
numCities = serverRealm.getNumCities();
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 58, Integer.toString(numCities)); // This territory is full
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private Building createStructure(PlayerCharacter playerCharacter, PlacementInfo buildingInfo, Zone currentZone) {
|
||
|
|
||
|
Blueprint blueprint;
|
||
|
Building newMesh;
|
||
|
DateTime completionDate;
|
||
|
float vendorRotation;
|
||
|
float buildingRotation;
|
||
|
|
||
|
blueprint = Blueprint.getBlueprint(buildingInfo.getBlueprintUUID());
|
||
|
|
||
|
if (blueprint == null) {
|
||
|
Logger.error("CreateStucture: DB returned null blueprint.");
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
// All seige buildings build in 15 minutes
|
||
|
if ((blueprint.getBuildingGroup().equals(BuildingGroup.SIEGETENT))
|
||
|
|| (blueprint.getBuildingGroup().equals(BuildingGroup.BULWARK)))
|
||
|
completionDate = DateTime.now().plusMinutes(15);
|
||
|
else
|
||
|
completionDate = DateTime.now().plusHours(blueprint.getRankTime(1));
|
||
|
|
||
|
Vector3fImmutable localLoc = new Vector3fImmutable(ZoneManager.worldToLocal(buildingInfo.getLoc(), currentZone));
|
||
|
|
||
|
buildingRotation = buildingInfo.getRot().y;
|
||
|
vendorRotation = buildingInfo.getW();
|
||
|
|
||
|
// if W return is negative, this is a -90 rotation not a 90?
|
||
|
|
||
|
newMesh = DbManager.BuildingQueries.CREATE_BUILDING(
|
||
|
currentZone.getObjectUUID(), playerCharacter.getObjectUUID(), blueprint.getName(), blueprint.getMeshForRank(0),
|
||
|
localLoc, 1.0f, blueprint.getMaxHealth(0), ProtectionState.NONE, 0, 0,
|
||
|
completionDate, blueprint.getMeshForRank(0), vendorRotation, buildingRotation);
|
||
|
|
||
|
// Make sure we have a valid mesh
|
||
|
if (newMesh == null) {
|
||
|
Logger.error("CreateStucture: DB returned null object.");
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
newMesh.setObjectTypeMask(MBServerStatics.MASK_BUILDING);
|
||
|
MaintenanceManager.setMaintDateTime(newMesh, LocalDateTime.now().plusDays(7));
|
||
|
|
||
|
WorldGrid.addObject(newMesh, playerCharacter);
|
||
|
return newMesh;
|
||
|
|
||
|
}
|
||
|
|
||
|
private boolean createShrine(PlayerCharacter player, PlacementInfo buildingInfo, Zone currentZone) {
|
||
|
|
||
|
Blueprint blueprint;
|
||
|
Building newMesh;
|
||
|
Shrine newShrine;
|
||
|
City city;
|
||
|
ShrineType shrineType;
|
||
|
|
||
|
if (player == null)
|
||
|
return false;
|
||
|
|
||
|
blueprint = Blueprint.getBlueprint(buildingInfo.getBlueprintUUID());
|
||
|
|
||
|
if (blueprint == null) {
|
||
|
Logger.error("CreateShrine: DB returned null blueprint.");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
shrineType = Shrine.getShrineTypeByBlueprintUUID(blueprint.getBlueprintUUID());
|
||
|
|
||
|
city = City.getCity(currentZone.getPlayerCityUUID());
|
||
|
|
||
|
if (city == null)
|
||
|
return false;
|
||
|
|
||
|
if (!city.isLocationOnCityGrid(buildingInfo.getLoc()))
|
||
|
return false;
|
||
|
|
||
|
Vector3fImmutable localLoc = new Vector3fImmutable(ZoneManager.worldToLocal(buildingInfo.getLoc(), currentZone));
|
||
|
|
||
|
float buildingRotation = buildingInfo.getRot().y;
|
||
|
float vendorRotation = buildingInfo.getW();
|
||
|
|
||
|
|
||
|
ArrayList<AbstractGameObject> shrineObjects = DbManager.ShrineQueries.CREATE_SHRINE(
|
||
|
currentZone.getObjectUUID(), player.getObjectUUID(), blueprint.getName(), blueprint.getMeshForRank(0),
|
||
|
localLoc, 1.0f, blueprint.getMaxHealth(0), ProtectionState.PROTECTED, 0, 0,
|
||
|
DateTime.now().plusHours(blueprint.getRankTime(1)), blueprint.getMeshForRank(0), vendorRotation, buildingRotation, shrineType.name());
|
||
|
|
||
|
if (shrineObjects == null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(player.getClientConnection(), 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
for (AbstractGameObject ago : shrineObjects) {
|
||
|
|
||
|
switch (ago.getObjectType()) {
|
||
|
case Building:
|
||
|
newMesh = (Building) ago;
|
||
|
newMesh.runAfterLoad();
|
||
|
newMesh.setObjectTypeMask(MBServerStatics.MASK_BUILDING);
|
||
|
MaintenanceManager.setMaintDateTime(newMesh, LocalDateTime.now().plusDays(7));
|
||
|
WorldGrid.addObject(newMesh, player);
|
||
|
break;
|
||
|
case Shrine:
|
||
|
newShrine = (Shrine) ago;
|
||
|
newShrine.getShrineType().addShrineToServerList(newShrine);
|
||
|
break;
|
||
|
default:
|
||
|
PlaceAssetMsg.sendPlaceAssetError(player.getClientConnection(), 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private boolean createBarracks(PlayerCharacter player, PlacementInfo buildingInfo, Zone currentZone) {
|
||
|
|
||
|
Blueprint blueprint;
|
||
|
Building newMesh;
|
||
|
Shrine newShrine;
|
||
|
City city;
|
||
|
|
||
|
if (player == null)
|
||
|
return false;
|
||
|
|
||
|
blueprint = Blueprint.getBlueprint(buildingInfo.getBlueprintUUID());
|
||
|
|
||
|
if (blueprint == null) {
|
||
|
Logger.error("CreateShrine: DB returned null blueprint.");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
city = City.getCity(currentZone.getPlayerCityUUID());
|
||
|
|
||
|
if (city == null)
|
||
|
return false;
|
||
|
|
||
|
if (!city.isLocationOnCityGrid(buildingInfo.getLoc()))
|
||
|
return false;
|
||
|
|
||
|
Vector3fImmutable localLoc = new Vector3fImmutable(ZoneManager.worldToLocal(buildingInfo.getLoc(), currentZone));
|
||
|
|
||
|
float buildingRotation = buildingInfo.getRot().y;
|
||
|
float vendorRotation = buildingInfo.getW();
|
||
|
DateTime completionDate = DateTime.now().plusHours(blueprint.getRankTime(1));
|
||
|
|
||
|
|
||
|
newMesh = DbManager.BuildingQueries.CREATE_BUILDING(
|
||
|
currentZone.getObjectUUID(), player.getObjectUUID(), blueprint.getName(), blueprint.getMeshForRank(0),
|
||
|
localLoc, 1.0f, blueprint.getMaxHealth(0), ProtectionState.PROTECTED, 0, 0,
|
||
|
completionDate, blueprint.getMeshForRank(0), vendorRotation, buildingRotation);
|
||
|
|
||
|
// Make sure we have a valid mesh
|
||
|
if (newMesh == null) {
|
||
|
Logger.error("CreateStucture: DB returned null object.");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
newMesh.setObjectTypeMask(MBServerStatics.MASK_BUILDING);
|
||
|
MaintenanceManager.setMaintDateTime(newMesh, LocalDateTime.now().plusDays(7));
|
||
|
WorldGrid.addObject(newMesh, player);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private boolean createWarehouse(PlayerCharacter player, PlacementInfo buildingInfo, Zone currentZone) {
|
||
|
|
||
|
Blueprint blueprint;
|
||
|
Building newMesh = null;
|
||
|
ArrayList<AbstractGameObject> warehouseObjects;
|
||
|
|
||
|
blueprint = Blueprint.getBlueprint(buildingInfo.getBlueprintUUID());
|
||
|
|
||
|
if (blueprint == null) {
|
||
|
Logger.error("CreateWarehouse: DB returned null blueprint.");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
Vector3fImmutable localLoc = new Vector3fImmutable(ZoneManager.worldToLocal(buildingInfo.getLoc(), currentZone));
|
||
|
|
||
|
float buildingRotation = buildingInfo.getRot().y;
|
||
|
float vendorRotation = buildingInfo.getW();
|
||
|
|
||
|
warehouseObjects = DbManager.WarehouseQueries.CREATE_WAREHOUSE(
|
||
|
currentZone.getObjectUUID(), player.getObjectUUID(), blueprint.getName(), blueprint.getMeshForRank(0),
|
||
|
localLoc, 1.0f, blueprint.getMaxHealth(0), ProtectionState.NONE, 0, 0,
|
||
|
DateTime.now().plusHours(blueprint.getRankTime(1)), blueprint.getMeshForRank(0), vendorRotation, buildingRotation);
|
||
|
|
||
|
if (warehouseObjects == null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(player.getClientConnection(), 1, "A Serious error has occurred. Please post details for to ensure transaction integrity");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Load the building into the simulation
|
||
|
|
||
|
for (AbstractGameObject ago : warehouseObjects) {
|
||
|
|
||
|
if (ago.getObjectType() == GameObjectType.Building) {
|
||
|
newMesh = (Building) ago;
|
||
|
newMesh.setObjectTypeMask(MBServerStatics.MASK_BUILDING);
|
||
|
MaintenanceManager.setMaintDateTime(newMesh, LocalDateTime.now().plusDays(7));
|
||
|
WorldGrid.addObject(newMesh, player);
|
||
|
newMesh.runAfterLoad();
|
||
|
}
|
||
|
else if (ago.getObjectType() == GameObjectType.Warehouse) {
|
||
|
Warehouse warehouse = (Warehouse) ago;
|
||
|
City city = City.getCity(currentZone.getPlayerCityUUID());
|
||
|
if (city == null)
|
||
|
return true;
|
||
|
city.setWarehouseBuildingID(newMesh.getObjectUUID());
|
||
|
Warehouse.warehouseByBuildingUUID.put(newMesh.getObjectUUID(), warehouse);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
// Validates that player is able to place buildings
|
||
|
|
||
|
private static boolean validateBuildingPlacement(Zone serverZone, PlaceAssetMsg msg, ClientConnection origin, PlayerCharacter player, PlacementInfo placementInfo) {
|
||
|
|
||
|
RealmType currentRealm;
|
||
|
|
||
|
// Retrieve the building details we're placing
|
||
|
|
||
|
if (serverZone.isNPCCity() == true) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 15, ""); // Cannot place in a peace zone
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Errant guilds cannot place assets
|
||
|
|
||
|
if (player.getGuild().getGuildState() == GuildState.Errant) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "Only soverign or sworn guilds may place assets.");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Cannot place a building underwater
|
||
|
|
||
|
if (HeightMap.isLocUnderwater(placementInfo.getLoc())) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 6, ""); // Cannot place underwater
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Players cannot place buildings in mob zones.
|
||
|
|
||
|
if ((serverZone.isMacroZone() == true)
|
||
|
|| (serverZone.getParent().isMacroZone() == true)) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 57, player.getName()); // No building may be placed within this territory
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
currentRealm = RealmType.getRealmTypeByUUID(RealmMap.getRealmIDAtLocation(player.getLoc()));
|
||
|
|
||
|
if (
|
||
|
(currentRealm.equals(RealmType.MAELSTROM)) ||
|
||
|
(currentRealm.equals(RealmType.OBLIVION))) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 57, player.getName()); // No building may be placed within this territory
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Cannot place assets on a dead tree
|
||
|
|
||
|
if ((serverZone.isPlayerCity())
|
||
|
&& (City.getCity(serverZone.getPlayerCityUUID()).getTOL().getRank() == -1)){
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "Cannot place asset on dead tree until world heals");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Overlap check
|
||
|
|
||
|
for (Building building : serverZone.zoneBuildingSet) {
|
||
|
|
||
|
if ((building.getBlueprintUUID() != 0) && (Bounds.collide(placementInfo, building) == true)) {
|
||
|
|
||
|
// Ignore and remove from simulation if we are placing over rubble
|
||
|
|
||
|
if (building.getRank() == -1) {
|
||
|
|
||
|
if ((building.getBlueprintUUID() != 0)
|
||
|
&& (building.getBlueprint().getBuildingGroup() == BuildingGroup.SHRINE)){
|
||
|
Shrine.RemoveShrineFromCacheByBuilding(building);
|
||
|
if (building.getCity() != null){
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
building.removeFromCache();
|
||
|
WorldGrid.RemoveWorldObject(building);
|
||
|
WorldGrid.removeObject(building);
|
||
|
building.getParentZone().zoneBuildingSet.remove(building);
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 3, ""); // Conflict between proposed assets
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private static boolean validateCityBuildingPlacement(Zone serverZone, PlaceAssetMsg msg, ClientConnection origin, PlayerCharacter player, PlacementInfo buildingInfo) {
|
||
|
|
||
|
// Peform shared common validation first
|
||
|
|
||
|
if (validateBuildingPlacement(serverZone, msg, origin, player, buildingInfo) == false)
|
||
|
return false;
|
||
|
|
||
|
// Must be a player city
|
||
|
|
||
|
if (serverZone.isPlayerCity() == false) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 41, player.getName()); // Cannot place outisde a guild zone
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
//Test zone has a city object
|
||
|
|
||
|
City city = City.getCity(serverZone.getPlayerCityUUID());
|
||
|
|
||
|
if (city == null) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 52, ""); //"no city to associate asset with"
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// City assets must be placed on the city grid
|
||
|
|
||
|
if (!city.isLocationOnCityGrid(buildingInfo.getLoc())) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 1, "Assset must be placed on a City Grid");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Make sure it's not an errant tree
|
||
|
|
||
|
if ( (city.getGuild() == null || city.getGuild().isErrant() == true)) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 18, ""); //"There are no guild trees to be found"
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
//Test player is in correct guild to place buildings
|
||
|
|
||
|
if (!player.isCSR)
|
||
|
if (player.getGuild().getObjectUUID() != city.getGuild().getObjectUUID()) {
|
||
|
PlaceAssetMsg.sendPlaceAssetError(origin, 9, ""); //You must be a guild member to place this asset
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
}
|