2017-04-18 15:51:40 +00:00
|
|
|
|
#include <cstdlib>
|
|
|
|
|
|
2019-09-09 07:28:35 +00:00
|
|
|
|
#include <components/openmw-mp/Utils.hpp>
|
2019-08-19 18:39:33 +00:00
|
|
|
|
#include <components/openmw-mp/TimedLog.hpp>
|
2017-04-18 15:51:40 +00:00
|
|
|
|
#include <components/openmw-mp/Version.hpp>
|
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
|
#include <components/esm/esmwriter.hpp>
|
2020-06-06 19:38:28 +00:00
|
|
|
|
#include <components/files/configurationmanager.hpp>
|
2017-05-28 10:26:48 +00:00
|
|
|
|
#include <components/files/escape.hpp>
|
2017-04-18 15:51:40 +00:00
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
|
#include "../mwbase/environment.hpp"
|
2017-04-18 15:51:40 +00:00
|
|
|
|
|
|
|
|
|
#include "../mwclass/creature.hpp"
|
|
|
|
|
#include "../mwclass/npc.hpp"
|
|
|
|
|
|
|
|
|
|
#include "../mwdialogue/dialoguemanagerimp.hpp"
|
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
|
#include "../mwgui/windowmanagerimp.hpp"
|
2017-04-18 15:51:40 +00:00
|
|
|
|
|
|
|
|
|
#include "../mwinput/inputmanagerimp.hpp"
|
|
|
|
|
|
|
|
|
|
#include "../mwmechanics/aitravel.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
|
#include "../mwmechanics/mechanicsmanagerimp.hpp"
|
2017-04-18 15:51:40 +00:00
|
|
|
|
#include "../mwmechanics/spellcasting.hpp"
|
|
|
|
|
|
|
|
|
|
#include "../mwscript/scriptmanagerimp.hpp"
|
|
|
|
|
|
|
|
|
|
#include "../mwstate/statemanagerimp.hpp"
|
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
|
#include "../mwworld/cellstore.hpp"
|
2017-04-18 15:51:40 +00:00
|
|
|
|
#include "../mwworld/customdata.hpp"
|
2016-11-17 15:16:25 +00:00
|
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2017-04-18 15:51:40 +00:00
|
|
|
|
#include "../mwworld/manualref.hpp"
|
|
|
|
|
#include "../mwworld/player.hpp"
|
|
|
|
|
#include "../mwworld/ptr.hpp"
|
|
|
|
|
#include "../mwworld/worldimp.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2017-04-18 15:51:40 +00:00
|
|
|
|
#include "Main.hpp"
|
2016-12-16 08:59:15 +00:00
|
|
|
|
#include "Networking.hpp"
|
2019-11-09 03:12:00 +00:00
|
|
|
|
#include "LocalSystem.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
#include "LocalPlayer.hpp"
|
2016-12-16 08:59:15 +00:00
|
|
|
|
#include "DedicatedPlayer.hpp"
|
2017-04-30 11:57:43 +00:00
|
|
|
|
#include "PlayerList.hpp"
|
2016-12-16 08:59:15 +00:00
|
|
|
|
#include "GUIController.hpp"
|
2017-04-05 09:00:21 +00:00
|
|
|
|
#include "CellController.hpp"
|
2017-04-17 11:37:19 +00:00
|
|
|
|
#include "MechanicsHelper.hpp"
|
2020-08-06 14:25:12 +00:00
|
|
|
|
#include "RecordHelper.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
|
|
using namespace mwmp;
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
|
|
Main *Main::pMain = 0;
|
2018-12-30 15:23:12 +00:00
|
|
|
|
std::string Main::address = "";
|
|
|
|
|
std::string Main::serverPassword = TES3MP_DEFAULT_PASSW;
|
2017-05-28 10:26:48 +00:00
|
|
|
|
std::string Main::resourceDir = "";
|
|
|
|
|
|
|
|
|
|
std::string Main::getResDir()
|
|
|
|
|
{
|
|
|
|
|
return resourceDir;
|
|
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2019-04-25 01:29:54 +00:00
|
|
|
|
std::string loadSettings(Settings::Manager& settings)
|
2016-01-12 03:41:44 +00:00
|
|
|
|
{
|
|
|
|
|
Files::ConfigurationManager mCfgMgr;
|
|
|
|
|
// Create the settings manager and load default settings file
|
|
|
|
|
const std::string localdefault = (mCfgMgr.getLocalPath() / "tes3mp-client-default.cfg").string();
|
|
|
|
|
const std::string globaldefault = (mCfgMgr.getGlobalPath() / "tes3mp-client-default.cfg").string();
|
|
|
|
|
|
|
|
|
|
// prefer local
|
|
|
|
|
if (boost::filesystem::exists(localdefault))
|
|
|
|
|
settings.loadDefault(localdefault);
|
|
|
|
|
else if (boost::filesystem::exists(globaldefault))
|
|
|
|
|
settings.loadDefault(globaldefault);
|
|
|
|
|
else
|
|
|
|
|
throw std::runtime_error ("No default settings file found! Make sure the file \"tes3mp-client-default.cfg\" was properly installed.");
|
|
|
|
|
|
|
|
|
|
// load user settings if they exist
|
|
|
|
|
const std::string settingspath = (mCfgMgr.getUserConfigPath() / "tes3mp-client.cfg").string();
|
|
|
|
|
if (boost::filesystem::exists(settingspath))
|
|
|
|
|
settings.loadUser(settingspath);
|
|
|
|
|
|
|
|
|
|
return settingspath;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Main::Main()
|
|
|
|
|
{
|
2019-08-19 18:39:33 +00:00
|
|
|
|
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "tes3mp started");
|
2016-01-12 03:41:44 +00:00
|
|
|
|
mNetworking = new Networking();
|
2019-11-09 03:12:00 +00:00
|
|
|
|
mLocalSystem = new LocalSystem();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
mLocalPlayer = new LocalPlayer();
|
2016-07-23 14:02:06 +00:00
|
|
|
|
mGUIController = new GUIController();
|
2017-04-05 09:00:21 +00:00
|
|
|
|
mCellController = new CellController();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
|
|
server = "mp.tes3mp.com";
|
|
|
|
|
port = 25565;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Main::~Main()
|
|
|
|
|
{
|
2019-08-19 18:39:33 +00:00
|
|
|
|
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "tes3mp stopped");
|
2016-01-12 03:41:44 +00:00
|
|
|
|
delete mNetworking;
|
2019-11-09 03:12:00 +00:00
|
|
|
|
delete mLocalSystem;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
delete mLocalPlayer;
|
2017-04-30 05:56:30 +00:00
|
|
|
|
delete mCellController;
|
|
|
|
|
delete mGUIController;
|
2017-04-17 16:10:33 +00:00
|
|
|
|
PlayerList::cleanUp();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
|
void Main::optionsDesc(boost::program_options::options_description *desc)
|
2016-08-24 08:15:34 +00:00
|
|
|
|
{
|
|
|
|
|
namespace bpo = boost::program_options;
|
2017-02-20 14:02:52 +00:00
|
|
|
|
desc->add_options()
|
|
|
|
|
("connect", bpo::value<std::string>()->default_value(""),
|
|
|
|
|
"connect to server (e.g. --connect=127.0.0.1:25565)")
|
|
|
|
|
("password", bpo::value<std::string>()->default_value(TES3MP_DEFAULT_PASSW),
|
|
|
|
|
"сonnect to a secured server. (e.g. --password=AnyPassword");
|
2016-08-24 08:15:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
|
void Main::configure(const boost::program_options::variables_map &variables)
|
2016-08-24 08:15:34 +00:00
|
|
|
|
{
|
2018-12-30 15:23:12 +00:00
|
|
|
|
Main::address = variables["connect"].as<string>();
|
|
|
|
|
Main::serverPassword = variables["password"].as<string>();
|
2020-11-01 21:17:59 +00:00
|
|
|
|
resourceDir = variables["resources"].as<Files::EscapePath>().mPath.string();
|
2016-08-24 08:15:34 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-03-04 05:11:46 +00:00
|
|
|
|
bool Main::init(std::vector<std::string> &content, Files::Collections &collections)
|
2016-09-18 03:54:45 +00:00
|
|
|
|
{
|
|
|
|
|
assert(!pMain);
|
|
|
|
|
pMain = new Main();
|
|
|
|
|
|
2019-04-25 01:29:54 +00:00
|
|
|
|
Settings::Manager manager;
|
2019-08-28 03:37:33 +00:00
|
|
|
|
loadSettings(manager);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2019-04-25 01:29:54 +00:00
|
|
|
|
int logLevel = manager.getInt("logLevel", "General");
|
2019-08-19 18:39:33 +00:00
|
|
|
|
TimedLog::SetLevel(logLevel);
|
2018-12-30 15:23:12 +00:00
|
|
|
|
if (address.empty())
|
2016-08-24 08:15:34 +00:00
|
|
|
|
{
|
2019-04-25 01:29:54 +00:00
|
|
|
|
pMain->server = manager.getString("destinationAddress", "General");
|
|
|
|
|
pMain->port = (unsigned short) manager.getInt("port", "General");
|
2017-02-20 14:02:52 +00:00
|
|
|
|
|
2019-04-25 01:29:54 +00:00
|
|
|
|
serverPassword = manager.getString("password", "General");
|
2018-12-30 15:23:12 +00:00
|
|
|
|
if (serverPassword.empty())
|
|
|
|
|
serverPassword = TES3MP_DEFAULT_PASSW;
|
2016-08-24 08:15:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-12-30 15:23:12 +00:00
|
|
|
|
size_t delimPos = address.find(':');
|
|
|
|
|
pMain->server = address.substr(0, delimPos);
|
|
|
|
|
pMain->port = atoi(address.substr(delimPos + 1).c_str());
|
2016-08-24 08:15:34 +00:00
|
|
|
|
}
|
2019-11-09 03:12:00 +00:00
|
|
|
|
get().mLocalSystem->serverPassword = serverPassword;
|
2017-03-04 06:55:35 +00:00
|
|
|
|
|
2017-03-04 05:23:26 +00:00
|
|
|
|
pMain->mNetworking->connect(pMain->server, pMain->port, content, collections);
|
2019-08-28 03:37:33 +00:00
|
|
|
|
|
2016-12-03 15:36:53 +00:00
|
|
|
|
return pMain->mNetworking->isConnected();
|
2016-09-18 03:54:45 +00:00
|
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
|
void Main::postInit()
|
2016-09-18 03:54:45 +00:00
|
|
|
|
{
|
2019-08-28 03:37:33 +00:00
|
|
|
|
pMain->mGUIController->setupChat();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
|
|
const MWBase::Environment &environment = MWBase::Environment::get();
|
|
|
|
|
environment.getStateManager()->newGame(true);
|
2016-09-18 03:54:45 +00:00
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->toggleAI();
|
2020-08-06 14:25:12 +00:00
|
|
|
|
RecordHelper::createPlaceholderInteriorCell();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-04 06:39:33 +00:00
|
|
|
|
bool Main::isInitialized()
|
|
|
|
|
{
|
|
|
|
|
return pMain != nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
|
void Main::destroy()
|
2016-01-12 03:41:44 +00:00
|
|
|
|
{
|
|
|
|
|
assert(pMain);
|
|
|
|
|
|
|
|
|
|
delete pMain;
|
|
|
|
|
pMain = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
|
void Main::frame(float dt)
|
2016-01-12 03:41:44 +00:00
|
|
|
|
{
|
2016-11-15 19:54:06 +00:00
|
|
|
|
get().getNetworking()->update();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2017-04-17 16:10:33 +00:00
|
|
|
|
PlayerList::update(dt);
|
2017-04-07 12:51:34 +00:00
|
|
|
|
get().getCellController()->updateDedicated(dt);
|
2016-11-15 19:54:06 +00:00
|
|
|
|
get().updateWorld(dt);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2016-08-24 01:48:35 +00:00
|
|
|
|
get().getGUIController()->update(dt);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
|
void Main::updateWorld(float dt) const
|
2016-01-12 03:41:44 +00:00
|
|
|
|
{
|
|
|
|
|
|
2018-04-19 10:25:29 +00:00
|
|
|
|
if (!mLocalPlayer->processCharGen())
|
2016-01-12 03:41:44 +00:00
|
|
|
|
return;
|
|
|
|
|
|
2016-09-18 03:54:45 +00:00
|
|
|
|
static bool init = true;
|
|
|
|
|
if (init)
|
2016-01-12 03:41:44 +00:00
|
|
|
|
{
|
2016-09-18 03:54:45 +00:00
|
|
|
|
init = false;
|
2019-08-19 18:39:33 +00:00
|
|
|
|
LOG_MESSAGE_SIMPLE(TimedLog::LOG_INFO, "Sending ID_PLAYER_BASEINFO to server");
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
2017-03-06 09:44:08 +00:00
|
|
|
|
mNetworking->getPlayerPacket(ID_PLAYER_BASEINFO)->setPlayer(getLocalPlayer());
|
|
|
|
|
mNetworking->getPlayerPacket(ID_LOADED)->setPlayer(getLocalPlayer());
|
|
|
|
|
mNetworking->getPlayerPacket(ID_PLAYER_BASEINFO)->Send();
|
|
|
|
|
mNetworking->getPlayerPacket(ID_LOADED)->Send();
|
2017-04-16 11:11:55 +00:00
|
|
|
|
mLocalPlayer->updateStatsDynamic(true);
|
2016-08-24 01:48:35 +00:00
|
|
|
|
get().getGUIController()->setChatVisible(true);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2017-04-05 09:00:21 +00:00
|
|
|
|
{
|
2016-11-15 19:54:06 +00:00
|
|
|
|
mLocalPlayer->update();
|
2017-04-10 14:10:18 +00:00
|
|
|
|
mCellController->updateLocal(false);
|
2017-04-05 09:00:21 +00:00
|
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const Main &Main::get()
|
|
|
|
|
{
|
|
|
|
|
return *pMain;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Networking *Main::getNetworking() const
|
|
|
|
|
{
|
|
|
|
|
return mNetworking;
|
|
|
|
|
}
|
|
|
|
|
|
2019-11-09 03:12:00 +00:00
|
|
|
|
LocalSystem *Main::getLocalSystem() const
|
|
|
|
|
{
|
|
|
|
|
return mLocalSystem;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
|
LocalPlayer *Main::getLocalPlayer() const
|
|
|
|
|
{
|
|
|
|
|
return mLocalPlayer;
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-24 01:48:35 +00:00
|
|
|
|
GUIController *Main::getGUIController() const
|
2016-01-12 03:41:44 +00:00
|
|
|
|
{
|
2016-07-23 14:02:06 +00:00
|
|
|
|
return mGUIController;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
}
|
|
|
|
|
|
2017-04-05 09:00:21 +00:00
|
|
|
|
CellController *Main::getCellController() const
|
2016-11-15 16:42:52 +00:00
|
|
|
|
{
|
2017-04-05 09:00:21 +00:00
|
|
|
|
return mCellController;
|
2016-11-15 16:42:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-09-14 06:37:19 +00:00
|
|
|
|
bool Main::isValidPacketScript(std::string scriptId)
|
2016-10-25 19:44:15 +00:00
|
|
|
|
{
|
2019-09-09 07:28:35 +00:00
|
|
|
|
mwmp::BaseWorldstate *worldstate = get().getNetworking()->getWorldstate();
|
2016-10-30 19:00:54 +00:00
|
|
|
|
|
2019-09-14 06:37:19 +00:00
|
|
|
|
if (Utils::vectorContains(worldstate->synchronizedClientScriptIds, scriptId))
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Main::isValidPacketGlobal(std::string globalId)
|
|
|
|
|
{
|
|
|
|
|
mwmp::BaseWorldstate *worldstate = get().getNetworking()->getWorldstate();
|
|
|
|
|
|
|
|
|
|
if (Utils::vectorContains(worldstate->synchronizedClientGlobalIds, globalId))
|
2019-09-09 07:28:35 +00:00
|
|
|
|
return true;
|
2016-10-30 19:00:54 +00:00
|
|
|
|
|
2019-09-09 07:28:35 +00:00
|
|
|
|
return false;
|
2016-10-25 19:44:15 +00:00
|
|
|
|
}
|