mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-28 21:15:34 +00:00
7393e3def6
ESM::Cell's getDescription() method was modified by aa5161f99e
despite being used heavily by TES3MP. All instances of it in the TES3MP code have now been changed into the newly added getShortDescription() that is identical to the previous getDescription().
481 lines
13 KiB
C++
481 lines
13 KiB
C++
#include <components/detournavigator/navigator.hpp>
|
|
#include <components/esm/cellid.hpp>
|
|
#include <components/openmw-mp/TimedLog.hpp>
|
|
#include <components/openmw-mp/Utils.hpp>
|
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwworld/containerstore.hpp"
|
|
#include "../mwworld/class.hpp"
|
|
#include "../mwworld/worldimp.hpp"
|
|
|
|
#include "CellController.hpp"
|
|
#include "Main.hpp"
|
|
#include "LocalActor.hpp"
|
|
#include "LocalPlayer.hpp"
|
|
using namespace mwmp;
|
|
|
|
std::map<std::string, mwmp::Cell *> CellController::cellsInitialized;
|
|
std::map<std::string, std::string> CellController::localActorsToCells;
|
|
std::map<std::string, std::string> CellController::dedicatedActorsToCells;
|
|
std::map<std::string, unsigned int> CellController::queuedDeathStates;
|
|
|
|
mwmp::CellController::CellController()
|
|
{
|
|
|
|
}
|
|
|
|
CellController::~CellController()
|
|
{
|
|
|
|
}
|
|
|
|
void CellController::updateLocal(bool forceUpdate)
|
|
{
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
|
|
|
// Loop through Cells, deleting inactive ones and updating LocalActors in active ones
|
|
for (auto it = cellsInitialized.begin(); it != cellsInitialized.end();)
|
|
{
|
|
mwmp::Cell *mpCell = it->second;
|
|
|
|
if (mpCell->getCellStore() == nullptr || mpCell->getCellStore()->getCell() == nullptr || !world->isCellActive(*mpCell->getCellStore()->getCell()))
|
|
{
|
|
mpCell->uninitializeLocalActors();
|
|
mpCell->uninitializeDedicatedActors();
|
|
delete it->second;
|
|
cellsInitialized.erase(it++);
|
|
}
|
|
else
|
|
{
|
|
mpCell->updateLocal(forceUpdate);
|
|
++it;
|
|
}
|
|
}
|
|
|
|
// If there are cellsInitialized remaining, loop through them and initialize new LocalActors for eligible ones
|
|
//
|
|
//
|
|
// Note: This cannot be combined with the above loop because initializing LocalActors in a Cell before they are
|
|
// deleted from their previous one can make their records stay deleted
|
|
if (cellsInitialized.size() > 0)
|
|
{
|
|
for (auto& cell : cellsInitialized)
|
|
{
|
|
mwmp::Cell* mpCell = cell.second;
|
|
if (mpCell->shouldInitializeActors == true)
|
|
{
|
|
mpCell->shouldInitializeActors = false;
|
|
mpCell->initializeLocalActors();
|
|
}
|
|
}
|
|
}
|
|
// Otherwise, disable the DetourNavigator for advanced pathfinding for the time being
|
|
else
|
|
{
|
|
world->getNavigator()->setUpdatesEnabled(false);
|
|
}
|
|
}
|
|
|
|
void CellController::updateDedicated(float dt)
|
|
{
|
|
for (const auto &cell : cellsInitialized)
|
|
cell.second->updateDedicated(dt);
|
|
}
|
|
|
|
void CellController::initializeCell(const ESM::Cell& cell)
|
|
{
|
|
std::string mapIndex = cell.getShortDescription();
|
|
|
|
// If this key doesn't exist, create it
|
|
if (cellsInitialized.count(mapIndex) == 0)
|
|
{
|
|
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "Initializing mwmp::Cell %s", cell.getShortDescription().c_str());
|
|
|
|
MWWorld::CellStore *cellStore = getCellStore(cell);
|
|
|
|
if (!cellStore) return;
|
|
|
|
mwmp::Cell *mpCell = new mwmp::Cell(cellStore);
|
|
cellsInitialized[mapIndex] = mpCell;
|
|
|
|
LOG_APPEND(TimedLog::LOG_VERBOSE, "- Successfully initialized mwmp::Cell %s", cell.getShortDescription().c_str());
|
|
}
|
|
}
|
|
|
|
void CellController::uninitializeCell(const ESM::Cell& cell)
|
|
{
|
|
std::string mapIndex = cell.getShortDescription();
|
|
|
|
// If this key exists, erase the key-value pair from the map
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
{
|
|
mwmp::Cell* mpCell = cellsInitialized.at(mapIndex);
|
|
mpCell->uninitializeLocalActors();
|
|
mpCell->uninitializeDedicatedActors();
|
|
delete cellsInitialized.at(mapIndex);
|
|
cellsInitialized.erase(mapIndex);
|
|
}
|
|
}
|
|
|
|
void CellController::uninitializeCells()
|
|
{
|
|
if (cellsInitialized.size() > 0)
|
|
{
|
|
for (auto it = cellsInitialized.cbegin(); it != cellsInitialized.cend(); it++)
|
|
{
|
|
mwmp::Cell* mpCell = it->second;
|
|
mpCell->uninitializeLocalActors();
|
|
mpCell->uninitializeDedicatedActors();
|
|
delete it->second;
|
|
}
|
|
|
|
cellsInitialized.clear();
|
|
}
|
|
}
|
|
|
|
void CellController::readPositions(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readPositions(actorList);
|
|
}
|
|
|
|
void CellController::readAnimFlags(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readAnimFlags(actorList);
|
|
}
|
|
|
|
void CellController::readAnimPlay(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readAnimPlay(actorList);
|
|
}
|
|
|
|
void CellController::readStatsDynamic(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readStatsDynamic(actorList);
|
|
}
|
|
|
|
void CellController::readDeath(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readDeath(actorList);
|
|
}
|
|
|
|
void CellController::readEquipment(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readEquipment(actorList);
|
|
}
|
|
|
|
void CellController::readSpeech(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readSpeech(actorList);
|
|
}
|
|
|
|
void CellController::readSpellsActive(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readSpellsActive(actorList);
|
|
}
|
|
|
|
void CellController::readAi(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readAi(actorList);
|
|
}
|
|
|
|
void CellController::readAttack(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readAttack(actorList);
|
|
}
|
|
|
|
void CellController::readCast(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readCast(actorList);
|
|
}
|
|
|
|
void CellController::readCellChange(ActorList& actorList)
|
|
{
|
|
std::string mapIndex = actorList.cell.getShortDescription();
|
|
|
|
initializeCell(actorList.cell);
|
|
|
|
// If this now exists, send it the data
|
|
if (cellsInitialized.count(mapIndex) > 0)
|
|
cellsInitialized[mapIndex]->readCellChange(actorList);
|
|
}
|
|
|
|
bool CellController::hasQueuedDeathState(MWWorld::Ptr ptr)
|
|
{
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
|
|
return queuedDeathStates.count(actorIndex) > 0;
|
|
}
|
|
|
|
unsigned int CellController::getQueuedDeathState(MWWorld::Ptr ptr)
|
|
{
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
|
|
return queuedDeathStates[actorIndex];
|
|
}
|
|
|
|
void CellController::clearQueuedDeathState(MWWorld::Ptr ptr)
|
|
{
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
|
|
queuedDeathStates.erase(actorIndex);
|
|
}
|
|
|
|
void CellController::setQueuedDeathState(MWWorld::Ptr ptr, unsigned int deathState)
|
|
{
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
|
|
queuedDeathStates[actorIndex] = deathState;
|
|
}
|
|
|
|
void CellController::setLocalActorRecord(std::string actorIndex, std::string cellIndex)
|
|
{
|
|
localActorsToCells[actorIndex] = cellIndex;
|
|
}
|
|
|
|
void CellController::removeLocalActorRecord(std::string actorIndex)
|
|
{
|
|
localActorsToCells.erase(actorIndex);
|
|
}
|
|
|
|
bool CellController::isLocalActor(MWWorld::Ptr ptr)
|
|
{
|
|
if (ptr.mRef == nullptr)
|
|
return false;
|
|
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
|
|
return localActorsToCells.count(actorIndex) > 0;
|
|
}
|
|
|
|
bool CellController::isLocalActor(int refNum, int mpNum)
|
|
{
|
|
std::string actorIndex = generateMapIndex(refNum, mpNum);
|
|
|
|
return localActorsToCells.count(actorIndex) > 0;
|
|
}
|
|
|
|
LocalActor *CellController::getLocalActor(MWWorld::Ptr ptr)
|
|
{
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
std::string cellIndex = localActorsToCells.at(actorIndex);
|
|
|
|
return cellsInitialized.at(cellIndex)->getLocalActor(actorIndex);
|
|
}
|
|
|
|
LocalActor *CellController::getLocalActor(int refNum, int mpNum)
|
|
{
|
|
std::string actorIndex = generateMapIndex(refNum, mpNum);
|
|
std::string cellIndex = localActorsToCells.at(actorIndex);
|
|
|
|
return cellsInitialized.at(cellIndex)->getLocalActor(actorIndex);
|
|
}
|
|
|
|
void CellController::setDedicatedActorRecord(std::string actorIndex, std::string cellIndex)
|
|
{
|
|
dedicatedActorsToCells[actorIndex] = cellIndex;
|
|
}
|
|
|
|
void CellController::removeDedicatedActorRecord(std::string actorIndex)
|
|
{
|
|
dedicatedActorsToCells.erase(actorIndex);
|
|
}
|
|
|
|
bool CellController::isDedicatedActor(MWWorld::Ptr ptr)
|
|
{
|
|
if (ptr.mRef == nullptr)
|
|
return false;
|
|
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
|
|
return dedicatedActorsToCells.count(actorIndex) > 0;
|
|
}
|
|
|
|
bool CellController::isDedicatedActor(int refNum, int mpNum)
|
|
{
|
|
std::string actorIndex = generateMapIndex(refNum, mpNum);
|
|
|
|
return dedicatedActorsToCells.count(actorIndex) > 0;
|
|
}
|
|
|
|
DedicatedActor *CellController::getDedicatedActor(MWWorld::Ptr ptr)
|
|
{
|
|
std::string actorIndex = generateMapIndex(ptr);
|
|
std::string cellIndex = dedicatedActorsToCells.at(actorIndex);
|
|
|
|
return cellsInitialized.at(cellIndex)->getDedicatedActor(actorIndex);
|
|
}
|
|
|
|
DedicatedActor *CellController::getDedicatedActor(int refNum, int mpNum)
|
|
{
|
|
std::string actorIndex = generateMapIndex(refNum, mpNum);
|
|
std::string cellIndex = dedicatedActorsToCells.at(actorIndex);
|
|
|
|
return cellsInitialized.at(cellIndex)->getDedicatedActor(actorIndex);
|
|
}
|
|
|
|
std::string CellController::generateMapIndex(int refNum, int mpNum)
|
|
{
|
|
std::string mapIndex = "";
|
|
mapIndex = Utils::toString(refNum) + "-" + Utils::toString(mpNum);
|
|
return mapIndex;
|
|
}
|
|
|
|
std::string CellController::generateMapIndex(MWWorld::Ptr ptr)
|
|
{
|
|
return generateMapIndex(ptr.getCellRef().getRefNum().mIndex, ptr.getCellRef().getMpNum());
|
|
}
|
|
|
|
std::string CellController::generateMapIndex(BaseActor baseActor)
|
|
{
|
|
return generateMapIndex(baseActor.refNum, baseActor.mpNum);
|
|
}
|
|
|
|
bool CellController::hasLocalAuthority(const ESM::Cell& cell)
|
|
{
|
|
if (isInitializedCell(cell) && isActiveWorldCell(cell))
|
|
return getCell(cell)->hasLocalAuthority();
|
|
|
|
return false;
|
|
}
|
|
|
|
bool CellController::isInitializedCell(const std::string& cellDescription)
|
|
{
|
|
return (cellsInitialized.count(cellDescription) > 0);
|
|
}
|
|
|
|
bool CellController::isInitializedCell(const ESM::Cell& cell)
|
|
{
|
|
return isInitializedCell(cell.getShortDescription());
|
|
}
|
|
|
|
bool CellController::isActiveWorldCell(const ESM::Cell& cell)
|
|
{
|
|
return MWBase::Environment::get().getWorld()->isCellActive(cell);
|
|
}
|
|
|
|
Cell *CellController::getCell(const ESM::Cell& cell)
|
|
{
|
|
return cellsInitialized.at(cell.getShortDescription());
|
|
}
|
|
|
|
MWWorld::CellStore *CellController::getCellStore(const ESM::Cell& cell)
|
|
{
|
|
MWWorld::CellStore *cellStore;
|
|
|
|
if (cell.isExterior())
|
|
cellStore = MWBase::Environment::get().getWorld()->getExterior(cell.mData.mX, cell.mData.mY);
|
|
else
|
|
{
|
|
try
|
|
{
|
|
cellStore = MWBase::Environment::get().getWorld()->getInterior(cell.mName);
|
|
}
|
|
catch (std::exception&)
|
|
{
|
|
cellStore = nullptr;
|
|
}
|
|
}
|
|
|
|
return cellStore;
|
|
}
|
|
|
|
bool CellController::isSameCell(const ESM::Cell& cell, const ESM::Cell& otherCell)
|
|
{
|
|
if (&cell == nullptr || &otherCell == nullptr) return false;
|
|
|
|
bool isCellExterior = false;
|
|
bool isOtherCellExterior = false;
|
|
|
|
try
|
|
{
|
|
isCellExterior = cell.isExterior();
|
|
isOtherCellExterior = otherCell.isExterior();
|
|
}
|
|
catch (std::exception& e)
|
|
{
|
|
LOG_MESSAGE_SIMPLE(TimedLog::LOG_ERROR, "Failed cell comparison");
|
|
return false;
|
|
}
|
|
|
|
if (isCellExterior && isOtherCellExterior)
|
|
{
|
|
if (cell.mData.mX == otherCell.mData.mX && cell.mData.mY == otherCell.mData.mY)
|
|
return true;
|
|
}
|
|
else if (Misc::StringUtils::ciEqual(cell.mName, otherCell.mName))
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
int CellController::getCellSize() const
|
|
{
|
|
return 8192;
|
|
}
|