1
Fork 0
mirror of https://github.com/SapphireServer/Sapphire.git synced 2025-04-27 14:57:44 +00:00
sapphire/src/servers/sapphire_zone/Territory/Land.cpp

355 lines
9.5 KiB
C++
Raw Normal View History

2018-11-01 23:56:43 +01:00
#include <set>
#include <Common.h>
#include <Logging/Logger.h>
#include <Util/Util.h>
#include <Util/UtilMath.h>
#include <Exd/ExdDataGenerated.h>
#include <Database/DatabaseDef.h>
#include <MySqlBase.h>
#include <Connection.h>
#include <Network/GamePacketNew.h>
#include <Network/PacketDef/Zone/ServerZoneDef.h>
#include "Actor/Player.h"
#include "Inventory/ItemContainer.h"
#include "Inventory/Item.h"
#include "Forwards.h"
#include "Land.h"
#include "Framework.h"
2018-11-25 01:55:53 +11:00
#include "House.h"
2018-11-01 23:56:43 +01:00
extern Sapphire::Framework g_fw;
2018-11-01 23:56:43 +01:00
using namespace Sapphire::Common;
2018-11-01 23:56:43 +01:00
Sapphire::Land::Land( uint16_t territoryTypeId, uint8_t wardNum, uint8_t landId, uint32_t landSetId,
Sapphire::Data::HousingLandSetPtr info ) :
2018-11-25 01:55:53 +11:00
m_territoryTypeId( territoryTypeId ),
2018-11-01 23:56:43 +01:00
m_wardNum( wardNum ),
m_landId( landId ),
m_currentPrice( 0 ),
2018-11-09 10:53:11 +01:00
m_minPrice( 0 ),
2018-11-10 23:47:19 +01:00
m_nextDrop( static_cast< uint32_t >( Util::getTimeSeconds() ) + 21600 ),
2018-11-10 19:00:13 +01:00
m_ownerPlayerId( 0 ),
2018-11-04 23:47:10 +01:00
m_landSetId( landSetId ),
m_landInfo( info ),
m_type( Common::LandType::none ),
2018-11-25 01:55:53 +11:00
m_fcIcon( 0 ),
m_fcIconColor( 0 ),
m_fcId( 0 ),
m_iconAddIcon( 0 )
2018-11-01 23:56:43 +01:00
{
2018-11-10 19:00:13 +01:00
memset( &m_tag, 0x00, 3 );
2018-11-11 14:27:39 +01:00
2018-11-01 23:56:43 +01:00
load();
}
Sapphire::Land::~Land()
2018-11-01 23:56:43 +01:00
{
}
void Sapphire::Land::load()
2018-11-01 23:56:43 +01:00
{
2018-11-04 23:47:10 +01:00
auto pDb = g_fw.get< Db::DbWorkerPool< Db::ZoneDbConnection > >();
2018-11-10 19:00:13 +01:00
auto res = pDb->query( "SELECT * FROM land WHERE LandSetId = " + std::to_string( m_landSetId ) + " "
"AND LandId = " + std::to_string( m_landId ) );
2018-11-04 23:47:10 +01:00
if( !res->next() )
{
pDb->directExecute( "INSERT INTO land ( landsetid, landid, type, size, status, landprice, UpdateTime, OwnerId, HouseId ) "
2018-11-04 23:47:10 +01:00
"VALUES ( " + std::to_string( m_landSetId ) + "," + std::to_string( m_landId ) + ","
+ std::to_string( static_cast< uint8_t >( m_type ) ) + ","
+ std::to_string( m_landInfo->plotSize[ m_landId ] ) + ","
+ " 1, " + std::to_string( m_landInfo->initialPrice[ m_landId ] ) + ", 0, 0, 0 );" );
2018-11-04 23:47:10 +01:00
m_currentPrice = m_landInfo->initialPrice[ m_landId ];
m_minPrice = m_landInfo->minPrice[ m_landId ];
m_size = m_landInfo->plotSize[ m_landId ];
m_state = HouseState::forSale;
2018-11-04 23:47:10 +01:00
}
else
{
m_type = static_cast< Common::LandType >( res->getUInt( "Type" ) );
m_size = res->getUInt( "Size" );
m_state = res->getUInt( "Status" );
2018-11-10 19:00:13 +01:00
m_currentPrice = res->getUInt( "LandPrice" );
m_ownerPlayerId = res->getUInt( "OwnerId" );
m_minPrice = m_landInfo->minPrice[ m_landId ];
m_maxPrice = m_landInfo->initialPrice[ m_landId ];
2018-11-26 23:32:22 +11:00
auto houseId = res->getUInt( "HouseId" );
// fetch the house if we have one for this land
if( houseId > 0 )
m_pHouse = make_House( houseId, m_landSetId, m_landId, m_wardNum, m_territoryTypeId );
2018-11-04 23:47:10 +01:00
}
auto pExdData = g_fw.get< Data::ExdDataGenerated >();
auto info = pExdData->get< Sapphire::Data::HousingMapMarkerInfo >( getTerritoryTypeId(), getLandId() );
if( info )
{
m_mapMarkerPosition.x = info->x;
m_mapMarkerPosition.y = info->y;
m_mapMarkerPosition.z = info->z;
}
2018-11-04 23:47:10 +01:00
init();
2018-11-01 23:56:43 +01:00
}
uint32_t Sapphire::Land::convertItemIdToHousingItemId( uint32_t itemId )
2018-11-01 23:56:43 +01:00
{
auto pExdData = g_fw.get< Data::ExdDataGenerated >();
auto info = pExdData->get< Sapphire::Data::Item >( itemId );
2018-11-01 23:56:43 +01:00
return info->additionalData;
}
uint32_t Sapphire::Land::getCurrentPrice() const
2018-11-04 23:47:10 +01:00
{
return m_currentPrice;
}
uint32_t Sapphire::Land::getMaxPrice() const
2018-11-17 01:16:44 +01:00
{
return m_maxPrice;
}
2018-11-01 23:56:43 +01:00
//Primary State
void Sapphire::Land::setSize( uint8_t size )
2018-11-01 23:56:43 +01:00
{
m_size = size;
2018-11-01 23:56:43 +01:00
}
void Sapphire::Land::setState( uint8_t state )
2018-11-01 23:56:43 +01:00
{
m_state = state;
2018-11-01 23:56:43 +01:00
}
void Sapphire::Land::setSharing( uint8_t state )
2018-11-01 23:56:43 +01:00
{
m_iconAddIcon = state;
2018-11-01 23:56:43 +01:00
}
void Sapphire::Land::setLandType( Common::LandType type )
{
m_type = type;
}
uint8_t Sapphire::Land::getSize() const
2018-11-01 23:56:43 +01:00
{
return m_size;
2018-11-01 23:56:43 +01:00
}
uint8_t Sapphire::Land::getState() const
2018-11-01 23:56:43 +01:00
{
return m_state;
2018-11-01 23:56:43 +01:00
}
uint8_t Sapphire::Land::getSharing() const
2018-11-01 23:56:43 +01:00
{
return m_iconAddIcon;
2018-11-01 23:56:43 +01:00
}
uint32_t Sapphire::Land::getLandSetId() const
2018-11-10 19:00:13 +01:00
{
return m_landSetId;
}
uint8_t Sapphire::Land::getWardNum() const
2018-11-10 19:00:13 +01:00
{
return m_wardNum;
}
uint8_t Sapphire::Land::getLandId() const
2018-11-10 19:00:13 +01:00
{
return m_landId;
}
uint16_t Sapphire::Land::getTerritoryTypeId() const
2018-11-10 19:00:13 +01:00
{
2018-11-25 01:55:53 +11:00
return m_territoryTypeId;
2018-11-10 19:00:13 +01:00
}
Sapphire::HousePtr Sapphire::Land::getHouse() const
2018-11-19 09:40:44 +01:00
{
2018-11-19 11:55:29 +01:00
return m_pHouse;
2018-11-19 09:40:44 +01:00
}
FFXIVARR_POSITION3 Sapphire::Land::getMapMarkerPosition()
{
return m_mapMarkerPosition;
}
Sapphire::Common::LandType Sapphire::Land::getLandType() const
{
return m_type;
}
2018-11-01 23:56:43 +01:00
//Free Comapny
void Sapphire::Land::setFreeCompany( uint32_t id, uint32_t icon, uint32_t color )
2018-11-01 23:56:43 +01:00
{
m_fcId = id;
m_fcIcon = icon;
m_fcIconColor = color; //RGBA
2018-11-01 23:56:43 +01:00
}
uint32_t Sapphire::Land::getFcId()
2018-11-01 23:56:43 +01:00
{
return m_fcIcon;
2018-11-01 23:56:43 +01:00
}
uint32_t Sapphire::Land::getFcIcon()
2018-11-01 23:56:43 +01:00
{
return m_fcIcon;
2018-11-01 23:56:43 +01:00
}
uint32_t Sapphire::Land::getFcColor()
2018-11-01 23:56:43 +01:00
{
return m_fcIconColor;
2018-11-01 23:56:43 +01:00
}
//Player
void Sapphire::Land::setPlayerOwner( uint32_t id )
2018-11-01 23:56:43 +01:00
{
m_ownerPlayerId = id;
}
uint32_t Sapphire::Land::getPlayerOwner()
2018-11-01 23:56:43 +01:00
{
return m_ownerPlayerId;
}
uint32_t Sapphire::Land::getMaxItems()
2018-11-01 23:56:43 +01:00
{
return m_maxItems;
}
uint32_t Sapphire::Land::getDevaluationTime()
2018-11-07 11:08:07 +01:00
{
2018-11-10 23:47:19 +01:00
return m_nextDrop - static_cast< uint32_t >( Util::getTimeSeconds() );
2018-11-07 11:08:07 +01:00
}
void Sapphire::Land::setCurrentPrice( uint32_t currentPrice )
2018-11-17 01:16:44 +01:00
{
m_currentPrice = currentPrice;
}
void Sapphire::Land::setLandTag( uint8_t slot, uint8_t tag )
2018-11-10 19:00:13 +01:00
{
m_tag[ slot ] = tag;
}
uint8_t Sapphire::Land::getLandTag( uint8_t slot )
2018-11-10 19:00:13 +01:00
{
return m_tag[ slot ];
}
void Sapphire::Land::init()
2018-11-01 23:56:43 +01:00
{
2018-11-14 10:00:16 +01:00
switch( m_size )
2018-11-01 23:56:43 +01:00
{
2018-11-03 23:44:43 +01:00
case HouseSize::small:
2018-11-01 23:56:43 +01:00
m_maxItems = 20;
break;
2018-11-03 23:44:43 +01:00
case HouseSize::medium:
2018-11-01 23:56:43 +01:00
m_maxItems = 30;
break;
2018-11-03 23:44:43 +01:00
case HouseSize::big:
2018-11-01 23:56:43 +01:00
m_maxItems = 40;
break;
default:
break;
}
}
void Sapphire::Land::updateLandDb()
2018-11-01 23:56:43 +01:00
{
2018-11-25 01:55:53 +11:00
uint32_t houseId = 0;
if( getHouse() )
houseId = getHouse()->getHouseId();
2018-11-26 23:32:22 +11:00
// todo: change to prepared statement
2018-11-10 19:00:13 +01:00
auto pDb = g_fw.get< Db::DbWorkerPool< Db::ZoneDbConnection > >();
pDb->directExecute( "UPDATE land SET status = " + std::to_string( m_state )
2018-11-10 19:00:13 +01:00
+ ", LandPrice = " + std::to_string( getCurrentPrice() )
+ ", UpdateTime = " + std::to_string( getDevaluationTime() )
2018-11-24 15:17:18 +11:00
+ ", OwnerId = " + std::to_string( getPlayerOwner() )
2018-11-25 01:55:53 +11:00
+ ", HouseId = " + std::to_string( houseId )
+ ", Type = " + std::to_string( static_cast< uint32_t >( m_type ) ) //TODO: add house id
2018-11-10 19:00:13 +01:00
+ " WHERE LandSetId = " + std::to_string( m_landSetId )
+ " AND LandId = " + std::to_string( m_landId ) + ";" );
2018-11-26 23:32:22 +11:00
if( auto house = getHouse() )
house->updateHouseDb();
2018-11-01 23:56:43 +01:00
}
void Sapphire::Land::update( uint32_t currTime )
2018-11-01 23:56:43 +01:00
{
2018-11-07 11:08:07 +01:00
if( getState() == HouseState::forSale )
2018-11-01 23:56:43 +01:00
{
2018-11-09 10:00:36 +01:00
if( m_nextDrop < currTime && m_minPrice < m_currentPrice )
2018-11-07 11:08:07 +01:00
{
2018-11-09 10:00:36 +01:00
m_nextDrop = currTime + 21600;
2018-12-02 02:01:41 +01:00
m_currentPrice = static_cast< uint32_t >( ( m_currentPrice / 100 ) * 99.58f );
updateLandDb();
2018-11-07 11:08:07 +01:00
}
2018-11-01 23:56:43 +01:00
}
}
2018-11-24 15:17:18 +11:00
uint32_t Sapphire::Land::getNextHouseId()
2018-11-25 01:55:53 +11:00
{
auto pDb = g_fw.get< Db::DbWorkerPool< Db::ZoneDbConnection > >();
auto pQR = pDb->query( "SELECT MAX( HouseId ) FROM house" );
if( !pQR->next() )
return 0;
return pQR->getUInt( 1 ) + 1;
}
bool Sapphire::Land::setPreset( uint32_t itemId )
2018-11-24 15:17:18 +11:00
{
auto housingItemId = convertItemIdToHousingItemId( itemId );
auto exdData = g_fw.get< Sapphire::Data::ExdDataGenerated >();
2018-11-24 15:17:18 +11:00
if( !exdData )
return false;
auto housingPreset = exdData->get< Sapphire::Data::HousingPreset >( housingItemId );
2018-11-25 01:55:53 +11:00
if( !housingPreset )
return false;
if( !getHouse() )
{
// todo: i guess we'd create a house here?
auto newId = getNextHouseId();
m_pHouse = make_House( newId, getLandSetId(), getLandId(), getWardNum(), getTerritoryTypeId() );
}
2018-12-02 14:59:24 +11:00
getHouse()->setHousePart( Common::HousePartSlot::ExteriorRoof, convertItemIdToHousingItemId( housingPreset->exteriorRoof ) );
getHouse()->setHousePart( Common::HousePartSlot::ExteriorWall, convertItemIdToHousingItemId( housingPreset->exteriorWall ) );
getHouse()->setHousePart( Common::HousePartSlot::ExteriorWindow, convertItemIdToHousingItemId( housingPreset->exteriorWindow ) );
getHouse()->setHousePart( Common::HousePartSlot::ExteriorDoor, convertItemIdToHousingItemId( housingPreset->exteriorDoor ) );
2018-11-24 15:17:18 +11:00
2018-12-02 14:59:24 +11:00
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorWall, convertItemIdToHousingItemId( housingPreset->interiorWall ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorFloor, convertItemIdToHousingItemId( housingPreset->interiorFlooring ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorLight, convertItemIdToHousingItemId( housingPreset->interiorLighting ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorWall_Attic, convertItemIdToHousingItemId( housingPreset->otherFloorWall ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorFloor_Attic, convertItemIdToHousingItemId( housingPreset->otherFloorFlooring ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorLight_Attic, convertItemIdToHousingItemId( housingPreset->otherFloorLighting ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorWall_Basement, convertItemIdToHousingItemId( housingPreset->basementWall ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorFloor_Basement, convertItemIdToHousingItemId( housingPreset->basementFlooring ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorLight_Basement, convertItemIdToHousingItemId( housingPreset->basementLighting ) );
getHouse()->setHouseInteriorPart( Common::HousingInteriorSlot::InteriorLight_Mansion, convertItemIdToHousingItemId( housingPreset->mansionLighting ) );
2018-12-01 21:40:30 +01:00
2018-11-25 01:55:53 +11:00
return true;
2018-11-24 15:17:18 +11:00
}