openmw-tes3coop/apps/openmw-mp/Cell.cpp

289 lines
6.6 KiB
C++
Raw Normal View History

2017-02-19 05:26:42 +00:00
//
// Created by koncord on 18.02.17.
//
#include "Cell.hpp"
#include <iostream>
#include "Player.hpp"
using namespace std;
2017-02-19 08:07:44 +00:00
void Cell::addPlayer(Player *player)
2017-02-19 05:26:42 +00:00
{
auto it = find(player->cells.begin(), player->cells.end(), this);
if (it == player->cells.end())
{
LOG_APPEND(Log::LOG_INFO, "- Adding %s to Player %s",
getDescription().c_str(),
player->npc.mName.c_str());
2017-02-19 05:26:42 +00:00
player->cells.push_back(this);
}
LOG_APPEND(Log::LOG_INFO, "- Adding %s to Cell %s",
player->npc.mName.c_str(),
getDescription().c_str());
2017-02-19 05:26:42 +00:00
players.push_back(player);
}
2017-02-19 08:07:44 +00:00
void Cell::removePlayer(Player *player)
2017-02-19 05:26:42 +00:00
{
for (Iterator it = begin(); it != end(); it++)
2017-02-19 05:26:42 +00:00
{
if (*it == player)
2017-02-19 05:26:42 +00:00
{
auto it2 = find(player->cells.begin(), player->cells.end(), this);
if (it2 != player->cells.end())
{
LOG_APPEND(Log::LOG_INFO, "- Removing %s from Player %s",
getDescription().c_str(),
player->npc.mName.c_str());
2017-02-19 05:26:42 +00:00
player->cells.erase(it2);
}
LOG_APPEND(Log::LOG_INFO, "- Removing %s from Cell %s",
player->npc.mName.c_str(),
getDescription().c_str());
2017-02-19 05:26:42 +00:00
players.erase(it);
return;
}
}
}
Cell::TPlayers Cell::getPlayers() const
2017-02-19 05:26:42 +00:00
{
return players;
}
void Cell::sendToLoaded(mwmp::WorldPacket *worldPacket, mwmp::BaseEvent *baseEvent) const
{
if (players.empty())
return;
std::list <Player*> plList;
for (auto pl : players)
plList.push_back(pl);
plList.sort();
plList.unique();
for (auto pl : plList)
{
if (pl->guid == baseEvent->guid) continue;
worldPacket->Send(baseEvent, pl->guid);
}
}
std::string Cell::getDescription() const
{
return cell.getDescription();
}
2017-02-19 05:26:42 +00:00
CellController::CellController()
{
}
CellController::~CellController()
{
for (auto cell : cells)
{
delete cell;
}
2017-02-19 05:26:42 +00:00
}
CellController *CellController::sThis = nullptr;
2017-02-19 08:07:44 +00:00
void CellController::create()
2017-02-19 05:26:42 +00:00
{
assert(!sThis);
2017-02-19 05:26:42 +00:00
sThis = new CellController;
}
2017-02-19 08:07:44 +00:00
void CellController::destroy()
2017-02-19 05:26:42 +00:00
{
assert(sThis);
delete sThis;
sThis = nullptr;
}
2017-02-19 08:07:44 +00:00
CellController *CellController::get()
2017-02-19 05:26:42 +00:00
{
assert(sThis);
2017-02-19 05:26:42 +00:00
return sThis;
}
Cell *CellController::getCell(ESM::Cell *esmCell)
{
if (esmCell->isExterior())
return getCellByXY(esmCell->mData.mX, esmCell->mData.mY);
else
return getCellByName(esmCell->mName);
}
2017-02-19 08:07:44 +00:00
Cell *CellController::getCellByXY(int x, int y)
2017-02-19 05:26:42 +00:00
{
auto it = find_if(cells.begin(), cells.end(), [x, y](const Cell *c)
{
2017-02-19 05:26:42 +00:00
return c->cell.mData.mX == x && c->cell.mData.mY == y;
});
if (it == cells.end())
{
LOG_APPEND(Log::LOG_INFO, "- Attempt to get Cell at %i, %i failed!", x, y);
2017-02-19 05:26:42 +00:00
return nullptr;
}
2017-02-19 05:26:42 +00:00
return *it;
}
Cell *CellController::getCellByName(std::string cellName)
2017-02-19 05:26:42 +00:00
{
auto it = find_if(cells.begin(), cells.end(), [cellName](const Cell *c)
{
return c->cell.mName == cellName;
2017-02-19 05:26:42 +00:00
});
if (it == cells.end())
{
LOG_APPEND(Log::LOG_INFO, "- Attempt to get Cell at %s failed!", cellName.c_str());
2017-02-19 05:26:42 +00:00
return nullptr;
}
2017-02-19 05:26:42 +00:00
return *it;
}
2017-02-19 08:07:44 +00:00
Cell *CellController::addCell(ESM::Cell cellData)
2017-02-19 05:26:42 +00:00
{
LOG_MESSAGE_SIMPLE(Log::LOG_VERBOSE, "Loaded cells: %d", cells.size());
2017-02-19 05:26:42 +00:00
auto it = find_if(cells.begin(), cells.end(), [cellData](const Cell *c) {
//return c->cell.sRecordId == cellData.sRecordId; // Currently we cannot compare because plugin lists can be loaded in different order
return c->cell.isExterior() ? (c->cell.mData.mX == cellData.mData.mX && c->cell.mData.mY == cellData.mData.mY) :
(c->cell.mName == cellData.mName);
2017-02-19 05:26:42 +00:00
});
2017-02-19 05:26:42 +00:00
Cell *cell;
if (it == cells.end())
2017-02-19 05:26:42 +00:00
{
LOG_APPEND(Log::LOG_INFO, "- Adding %s to CellController",
cellData.getDescription().c_str());
2017-02-19 05:26:42 +00:00
cell = new Cell(cellData);
cells.push_back(cell);
}
else
{
LOG_APPEND(Log::LOG_INFO, "- Found %s in CellController",
cellData.getDescription().c_str());
2017-02-19 05:26:42 +00:00
cell = *it;
}
2017-02-19 05:26:42 +00:00
return cell;
}
2017-02-19 08:07:44 +00:00
void CellController::removeCell(Cell *cell)
2017-02-19 05:26:42 +00:00
{
if (cell == nullptr)
2017-02-19 05:26:42 +00:00
return;
2017-02-19 05:26:42 +00:00
for (auto it = cells.begin(); it != cells.end();)
{
if (*it != nullptr && *it == cell)
2017-02-19 05:26:42 +00:00
{
LOG_APPEND(Log::LOG_INFO, "- Removing %s from CellController",
cell->getDescription().c_str());
2017-02-19 05:26:42 +00:00
delete *it;
it = cells.erase(it);
}
else
++it;
}
}
2017-02-19 08:07:44 +00:00
void CellController::removePlayer(Cell *cell, Player *player)
2017-02-19 05:26:42 +00:00
{
2017-02-19 08:07:44 +00:00
cell->removePlayer(player);
2017-02-19 05:26:42 +00:00
if (cell->players.empty())
2017-02-19 05:26:42 +00:00
{
2017-02-21 10:14:02 +00:00
LOG_MESSAGE_SIMPLE(Log::LOG_VERBOSE, "Deleting empty cell from memory: %s", player->npc.mName.c_str(),
player->getId(), cell->cell.getDescription().c_str());
2017-02-19 05:26:42 +00:00
auto it = find(cells.begin(), cells.end(), cell);
delete *it;
cells.erase(it);
}
}
void CellController::deletePlayer(Player *player)
{
LOG_APPEND(Log::LOG_INFO, "- Iterating through Cells from Player %s",
player->npc.mName.c_str());
for_each(player->getCells()->begin(), player->getCells()->end(), [&player](Cell *cell)
{
LOG_APPEND(Log::LOG_INFO, "-- Found Cell %s",
cell->getDescription().c_str());
for (auto it = cell->begin(); it != cell->end(); ++it)
{
if (*it == player)
{
LOG_APPEND(Log::LOG_INFO, "-- Deleting %s from Cell %s",
player->npc.mName.c_str(),
cell->getDescription().c_str());
cell->players.erase(it);
break;
}
}
});
}
2017-02-19 05:26:42 +00:00
void CellController::update(Player *player)
{
for (auto cell : player->cellStateChanges.cellStates)
2017-02-19 05:26:42 +00:00
{
if (cell.type == mwmp::CellState::LOAD)
2017-02-19 05:26:42 +00:00
{
2017-02-19 08:07:44 +00:00
Cell *c = addCell(cell.cell);
c->addPlayer(player);
2017-02-19 05:26:42 +00:00
}
else
{
2017-02-20 11:45:35 +00:00
LOG_MESSAGE_SIMPLE(Log::LOG_VERBOSE, "Player %s (%d) unloaded cell: %s", player->npc.mName.c_str(), player->getId(), cell.cell.getDescription().c_str());
2017-02-19 05:26:42 +00:00
Cell *c;
if (!cell.cell.isExterior())
c = getCellByName(cell.cell.mName);
2017-02-19 05:26:42 +00:00
else
2017-02-19 08:07:44 +00:00
c = getCellByXY(cell.cell.getGridX(), cell.cell.getGridY());
2017-02-19 05:26:42 +00:00
if (c != nullptr)
2017-02-19 08:07:44 +00:00
removePlayer(c, player);
2017-02-19 05:26:42 +00:00
}
}
}
Cell::Cell(ESM::Cell cell): cell(cell)
{
}
2017-02-22 04:03:03 +00:00
Cell::Iterator Cell::begin() const
{
return players.begin();
}
2017-02-22 04:03:03 +00:00
Cell::Iterator Cell::end() const
{
return players.end();
}