forked from teamnwah/openmw-tes3coop
265 lines
7 KiB
C++
265 lines
7 KiB
C++
//
|
|
// Created by koncord on 01.01.16.
|
|
//
|
|
|
|
#include "Main.hpp"
|
|
#include <apps/openmw/mwworld/manualref.hpp>
|
|
#include <apps/openmw/mwmechanics/aitravel.hpp>
|
|
#include <components/esm/esmwriter.hpp>
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwstate/statemanagerimp.hpp"
|
|
#include "../mwinput/inputmanagerimp.hpp"
|
|
#include "../mwscript/scriptmanagerimp.hpp"
|
|
#include "../mwgui/windowmanagerimp.hpp"
|
|
#include "../mwworld/worldimp.hpp"
|
|
#include "../mwworld/ptr.hpp"
|
|
#include "../mwworld/player.hpp"
|
|
#include "../mwworld/customdata.hpp"
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
#include "../mwclass/npc.hpp"
|
|
#include "../mwclass/creature.hpp"
|
|
#include "../mwmechanics/mechanicsmanagerimp.hpp"
|
|
#include "../mwworld/cellstore.hpp"
|
|
#include <apps/openmw/mwdialogue/dialoguemanagerimp.hpp>
|
|
#include <apps/openmw/mwworld/inventorystore.hpp>
|
|
#include <apps/openmw/mwmechanics/spellcasting.hpp>
|
|
#include <components/openmw-mp/Log.hpp>
|
|
#include <cstdlib>
|
|
|
|
#include "DedicatedPlayer.hpp"
|
|
#include "LocalPlayer.hpp"
|
|
|
|
using namespace mwmp;
|
|
using namespace std;
|
|
|
|
Main *Main::pMain = 0;
|
|
std::string Main::addr = "";
|
|
|
|
std::string loadSettings (Settings::Manager & settings)
|
|
{
|
|
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()
|
|
{
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "%s", "TES3MP started");
|
|
mNetworking = new Networking();
|
|
mLocalPlayer = new LocalPlayer();
|
|
mGUIController = new GUIController();
|
|
//mLocalPlayer->CharGen(0, 4);
|
|
|
|
server = "mp.tes3mp.com";
|
|
port = 25565;
|
|
}
|
|
|
|
Main::~Main()
|
|
{
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "%s", "TES3MP stopped");
|
|
delete mNetworking;
|
|
delete mLocalPlayer;
|
|
Players::CleanUp();
|
|
}
|
|
|
|
void Main::OptionsDesc(boost::program_options::options_description *desc)
|
|
{
|
|
namespace bpo = boost::program_options;
|
|
desc->add_options()("connect", bpo::value<std::string>()->default_value(""),
|
|
"connect to server (e.g. --connect=127.0.0.1:25565)");
|
|
}
|
|
|
|
void Main::Configure(const boost::program_options::variables_map &variables)
|
|
{
|
|
Main::addr = variables["connect"].as<string>();
|
|
}
|
|
|
|
static Settings::CategorySettingValueMap saveUserSettings;
|
|
static Settings::CategorySettingValueMap saveDefaultSettings;
|
|
static Settings::CategorySettingVector saveChangedSettings;
|
|
|
|
void InitMgr(Settings::Manager &mgr)
|
|
{
|
|
saveUserSettings = mgr.mUserSettings;
|
|
saveDefaultSettings = mgr.mDefaultSettings;
|
|
saveChangedSettings = mgr.mChangedSettings;
|
|
mgr.mUserSettings.clear();
|
|
mgr.mDefaultSettings.clear();
|
|
mgr.mChangedSettings.clear();
|
|
loadSettings(mgr);
|
|
}
|
|
|
|
void RestoreMgr(Settings::Manager &mgr)
|
|
{
|
|
mgr.mUserSettings = saveUserSettings;
|
|
mgr.mDefaultSettings = saveDefaultSettings;
|
|
mgr.mChangedSettings = saveChangedSettings;
|
|
}
|
|
|
|
void Main::Init(std::vector<std::string> &content)
|
|
{
|
|
assert(!pMain);
|
|
pMain = new Main();
|
|
|
|
Settings::Manager mgr;
|
|
InitMgr(mgr);
|
|
|
|
int logLevel = mgr.getInt("loglevel", "General");
|
|
Log::SetLevel(logLevel);
|
|
if (addr.empty())
|
|
{
|
|
pMain->server = mgr.getString("server", "General");
|
|
pMain->port = (unsigned short) mgr.getInt("port", "General");
|
|
}
|
|
else
|
|
{
|
|
size_t delim_pos = addr.find(':');
|
|
pMain->server = addr.substr(0, delim_pos);
|
|
pMain->port = atoi(addr.substr(delim_pos + 1).c_str());
|
|
|
|
}
|
|
|
|
pMain->mNetworking->Connect(pMain->server, pMain->port);
|
|
RestoreMgr(mgr);
|
|
}
|
|
|
|
void Main::PostInit()
|
|
{
|
|
Settings::Manager mgr;
|
|
InitMgr(mgr);
|
|
|
|
pMain->mGUIController->setupChat(mgr);
|
|
RestoreMgr(mgr);
|
|
|
|
const MWBase::Environment &environment = MWBase::Environment::get();
|
|
environment.getStateManager()->newGame(true);
|
|
MWBase::Environment::get().getMechanicsManager()->toggleAI();
|
|
}
|
|
|
|
void Main::Destroy()
|
|
{
|
|
assert(pMain);
|
|
|
|
delete pMain;
|
|
pMain = 0;
|
|
}
|
|
|
|
void Main::Frame(float dt)
|
|
{
|
|
const MWBase::Environment &environment = MWBase::Environment::get();
|
|
if (environment.getWindowManager()->containsMode(MWGui::GM_MainMenu))
|
|
{
|
|
//environment.getWindowManager()->exitCurrentGuiMode();
|
|
}
|
|
|
|
get().getNetworking()->Update();
|
|
|
|
Players::Update(dt);
|
|
get().UpdateWorld(dt);
|
|
|
|
get().getGUIController()->update(dt);
|
|
|
|
}
|
|
|
|
void Main::UpdateWorld(float dt) const
|
|
{
|
|
|
|
if (!mLocalPlayer->charGenThread())
|
|
return;
|
|
|
|
static bool init = true;
|
|
if (init)
|
|
{
|
|
init = false;
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "%s", "Sending ID_GAME_BASE_INFO to server");
|
|
|
|
mNetworking->GetPlayerPacket(ID_GAME_BASE_INFO)->Send(getLocalPlayer());
|
|
mNetworking->GetPlayerPacket(ID_LOADED)->Send(getLocalPlayer());
|
|
mLocalPlayer->updateDynamicStats(true);
|
|
get().getGUIController()->setChatVisible(true);
|
|
}
|
|
else
|
|
mLocalPlayer->Update();
|
|
}
|
|
|
|
const Main &Main::get()
|
|
{
|
|
return *pMain;
|
|
}
|
|
|
|
Networking *Main::getNetworking() const
|
|
{
|
|
return mNetworking;
|
|
}
|
|
|
|
LocalPlayer *Main::getLocalPlayer() const
|
|
{
|
|
return mLocalPlayer;
|
|
}
|
|
|
|
|
|
GUIController *Main::getGUIController() const
|
|
{
|
|
return mGUIController;
|
|
}
|
|
|
|
void Main::PressedKey(int key)
|
|
{
|
|
if (pMain == nullptr) return;
|
|
if (get().getGUIController()->pressedKey(key))
|
|
return; // if any gui bind pressed
|
|
}
|
|
|
|
// When sending packets with ingame script values, certain packets
|
|
// should be ignored because of their potential for spam
|
|
bool Main::isValidPacketScript(std::string script)
|
|
{
|
|
static const int invalidPacketScriptsCount = 17;
|
|
static const std::string invalidPacketScripts[invalidPacketScriptsCount] = {
|
|
// Spammy shorts
|
|
"OutsideBanner",
|
|
"sleeperScript",
|
|
"dreamer_talkerEnable",
|
|
"drenSlaveOwners",
|
|
"ahnassiScript",
|
|
"hlormarScript",
|
|
// Spammy floats
|
|
"Float",
|
|
"SignRotate",
|
|
"FaluraScript",
|
|
"jsaddhaScript",
|
|
// Spammy globals
|
|
"wraithguardScript",
|
|
// Spammy globals leading to crashes
|
|
"LegionUniform",
|
|
"OrdinatorUniform",
|
|
"LorkhanHeart",
|
|
"ouch_keening",
|
|
"ouch_sunder",
|
|
"ouch_wraithguard"
|
|
};
|
|
|
|
for (int i = 0; i < invalidPacketScriptsCount; i++)
|
|
{
|
|
if (Misc::StringUtils::ciEqual(script, invalidPacketScripts[i]))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|