2016-01-12 03:41:44 +00:00
|
|
|
#include <components/esm/esmwriter.hpp>
|
2017-04-08 09:54:38 +00:00
|
|
|
#include <components/openmw-mp/Log.hpp>
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
2017-01-24 17:32:25 +00:00
|
|
|
#include "../mwbase/journal.hpp"
|
2017-10-31 13:19:14 +00:00
|
|
|
#include "../mwbase/soundmanager.hpp"
|
2017-04-08 09:54:38 +00:00
|
|
|
|
|
|
|
#include "../mwclass/creature.hpp"
|
|
|
|
#include "../mwclass/npc.hpp"
|
|
|
|
|
|
|
|
#include "../mwdialogue/dialoguemanagerimp.hpp"
|
|
|
|
|
|
|
|
#include "../mwgui/inventorywindow.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
#include "../mwgui/windowmanagerimp.hpp"
|
2017-04-08 09:54:38 +00:00
|
|
|
|
|
|
|
#include "../mwinput/inputmanagerimp.hpp"
|
|
|
|
|
2017-12-02 07:22:36 +00:00
|
|
|
#include "../mwmechanics/activespells.hpp"
|
2017-04-08 09:54:38 +00:00
|
|
|
#include "../mwmechanics/aitravel.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
#include "../mwmechanics/mechanicsmanagerimp.hpp"
|
2018-02-06 04:36:46 +00:00
|
|
|
#include "../mwmechanics/spellcasting.hpp"
|
2017-04-08 09:54:38 +00:00
|
|
|
|
|
|
|
#include "../mwscript/scriptmanagerimp.hpp"
|
|
|
|
|
|
|
|
#include "../mwstate/statemanagerimp.hpp"
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
#include "../mwworld/cellstore.hpp"
|
2017-04-08 09:54:38 +00:00
|
|
|
#include "../mwworld/customdata.hpp"
|
2016-11-17 15:16:25 +00:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2017-04-08 09:54:38 +00:00
|
|
|
#include "../mwworld/manualref.hpp"
|
|
|
|
#include "../mwworld/player.hpp"
|
|
|
|
#include "../mwworld/worldimp.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
#include "LocalPlayer.hpp"
|
2017-04-19 19:06:04 +00:00
|
|
|
#include "Main.hpp"
|
2016-12-16 08:59:15 +00:00
|
|
|
#include "Networking.hpp"
|
2017-04-05 09:00:21 +00:00
|
|
|
#include "CellController.hpp"
|
2018-01-25 22:45:39 +00:00
|
|
|
#include "GUIController.hpp"
|
2017-04-19 19:06:04 +00:00
|
|
|
#include "MechanicsHelper.hpp"
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
using namespace mwmp;
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
LocalPlayer::LocalPlayer()
|
|
|
|
{
|
2018-04-19 10:25:29 +00:00
|
|
|
charGenState.currentStage = 0;
|
|
|
|
charGenState.endStage = 1;
|
|
|
|
charGenState.isFinished = false;
|
2017-06-20 02:28:45 +00:00
|
|
|
|
|
|
|
difficulty = 0;
|
2018-02-14 03:53:44 +00:00
|
|
|
physicsFramerate = 60.0;
|
2017-11-27 05:39:02 +00:00
|
|
|
consoleAllowed = false;
|
2017-11-30 10:18:15 +00:00
|
|
|
bedRestAllowed = true;
|
|
|
|
wildernessRestAllowed = true;
|
2017-11-27 05:39:02 +00:00
|
|
|
waitAllowed = true;
|
2017-06-20 02:28:45 +00:00
|
|
|
|
2016-11-12 20:21:02 +00:00
|
|
|
ignorePosPacket = false;
|
2017-07-03 12:45:21 +00:00
|
|
|
ignoreJailTeleportation = false;
|
2017-07-12 15:24:37 +00:00
|
|
|
ignoreJailSkillIncreases = false;
|
2017-04-19 19:06:04 +00:00
|
|
|
|
|
|
|
attack.shouldSend = false;
|
2017-05-05 19:16:31 +00:00
|
|
|
|
|
|
|
deathReason = "suicide";
|
2017-06-10 08:43:40 +00:00
|
|
|
isChangingRegion = false;
|
2017-07-12 13:39:31 +00:00
|
|
|
|
2017-07-13 17:13:28 +00:00
|
|
|
jailProgressText = "";
|
|
|
|
jailEndText = "";
|
2017-09-04 12:13:05 +00:00
|
|
|
|
2018-02-01 00:11:45 +00:00
|
|
|
scale = 1;
|
2017-09-04 12:44:10 +00:00
|
|
|
isWerewolf = false;
|
|
|
|
|
2017-10-25 04:21:00 +00:00
|
|
|
isReceivingQuickKeys = false;
|
2017-11-01 20:00:54 +00:00
|
|
|
isPlayingAnimation = false;
|
2018-02-25 19:33:04 +00:00
|
|
|
diedSinceArrestAttempt = false;
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
LocalPlayer::~LocalPlayer()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
Networking *LocalPlayer::getNetworking()
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
|
|
|
return mwmp::Main::get().getNetworking();
|
|
|
|
}
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
MWWorld::Ptr LocalPlayer::getPlayerPtr()
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
|
|
|
return MWBase::Environment::get().getWorld()->getPlayerPtr();
|
|
|
|
}
|
|
|
|
|
2016-11-15 19:54:06 +00:00
|
|
|
void LocalPlayer::update()
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2017-06-02 13:20:35 +00:00
|
|
|
static float updateTimer = 0;
|
|
|
|
const float timeoutSec = 0.015;
|
|
|
|
|
|
|
|
if ((updateTimer += MWBase::Environment::get().getFrameDuration()) >= timeoutSec)
|
|
|
|
{
|
|
|
|
updateTimer = 0;
|
|
|
|
updateCell();
|
|
|
|
updatePosition();
|
|
|
|
updateAnimFlags();
|
|
|
|
updateAttack();
|
|
|
|
updateDeadState();
|
|
|
|
updateEquipment();
|
|
|
|
updateStatsDynamic();
|
2017-09-04 17:21:06 +00:00
|
|
|
updateAttributes();
|
|
|
|
updateSkills();
|
2017-06-02 13:20:35 +00:00
|
|
|
updateLevel();
|
|
|
|
updateBounty();
|
2018-01-30 20:55:29 +00:00
|
|
|
updateReputation();
|
2017-06-02 13:20:35 +00:00
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2018-04-19 10:25:29 +00:00
|
|
|
bool LocalPlayer::processCharGen()
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
|
|
|
MWBase::WindowManager *windowManager = MWBase::Environment::get().getWindowManager();
|
2016-11-21 21:40:50 +00:00
|
|
|
|
2016-11-11 21:46:17 +00:00
|
|
|
// If we haven't finished CharGen and we're in a menu, it must be
|
|
|
|
// one of the CharGen menus, so go no further until it's closed
|
2018-04-19 10:25:29 +00:00
|
|
|
if (windowManager->isGuiMode() && !charGenState.isFinished)
|
|
|
|
{
|
2016-09-30 05:59:58 +00:00
|
|
|
return false;
|
2018-04-19 10:25:29 +00:00
|
|
|
}
|
2016-09-30 05:59:58 +00:00
|
|
|
|
2016-11-11 21:46:17 +00:00
|
|
|
// If the current stage of CharGen is not the last one,
|
|
|
|
// move to the next one
|
2018-04-19 10:25:29 +00:00
|
|
|
else if (charGenState.currentStage < charGenState.endStage)
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
2018-04-19 10:25:29 +00:00
|
|
|
switch (charGenState.currentStage)
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
2016-11-11 21:46:17 +00:00
|
|
|
case 0:
|
|
|
|
windowManager->pushGuiMode(MWGui::GM_Name);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
windowManager->pushGuiMode(MWGui::GM_Race);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
windowManager->pushGuiMode(MWGui::GM_Class);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
windowManager->pushGuiMode(MWGui::GM_Birth);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
windowManager->pushGuiMode(MWGui::GM_Review);
|
|
|
|
break;
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CHARGEN)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CHARGEN)->Send();
|
2018-04-19 10:25:29 +00:00
|
|
|
charGenState.currentStage++;
|
2016-09-30 05:59:58 +00:00
|
|
|
|
2016-11-11 21:46:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
2016-11-21 21:40:50 +00:00
|
|
|
|
2016-11-11 21:46:17 +00:00
|
|
|
// If we've reached the last stage of CharGen, send the
|
|
|
|
// corresponding packets and mark CharGen as finished
|
2018-04-19 10:25:29 +00:00
|
|
|
else if (!charGenState.isFinished)
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
2016-11-11 21:46:17 +00:00
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
|
|
|
npc = *ptrPlayer.get<ESM::NPC>()->mBase;
|
2017-01-25 15:06:15 +00:00
|
|
|
birthsign = world->getPlayer().getBirthSign();
|
2016-11-11 21:46:17 +00:00
|
|
|
|
2017-02-05 07:01:33 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_BASEINFO to server with my CharGen info");
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_BASEINFO)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_BASEINFO)->Send();
|
2016-11-11 21:46:17 +00:00
|
|
|
|
|
|
|
// Send stats packets if this is the 2nd round of CharGen that
|
|
|
|
// only happens for new characters
|
2018-04-19 10:25:29 +00:00
|
|
|
if (charGenState.endStage != 1)
|
2016-11-11 21:46:17 +00:00
|
|
|
{
|
2017-04-16 11:11:55 +00:00
|
|
|
updateStatsDynamic(true);
|
2016-11-11 21:46:17 +00:00
|
|
|
updateAttributes(true);
|
|
|
|
updateSkills(true);
|
|
|
|
updateLevel(true);
|
|
|
|
sendClass();
|
2017-01-20 07:07:07 +00:00
|
|
|
sendSpellbook();
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CHARGEN)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CHARGEN)->Send();
|
2016-11-11 21:46:17 +00:00
|
|
|
}
|
|
|
|
|
2018-04-19 10:25:29 +00:00
|
|
|
// Mark character generation as finished until overridden by a new ID_PLAYER_CHARGEN packet
|
|
|
|
charGenState.isFinished = true;
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
|
|
|
|
2016-11-11 21:46:17 +00:00
|
|
|
return true;
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2017-01-26 11:13:43 +00:00
|
|
|
bool LocalPlayer::hasFinishedCharGen()
|
|
|
|
{
|
2018-04-19 10:25:29 +00:00
|
|
|
return charGenState.isFinished;
|
2017-01-26 11:13:43 +00:00
|
|
|
}
|
|
|
|
|
2017-04-16 11:11:55 +00:00
|
|
|
void LocalPlayer::updateStatsDynamic(bool forceUpdate)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-27 00:45:17 +00:00
|
|
|
if (statsDynamicIndexChanges.size() > 0)
|
|
|
|
statsDynamicIndexChanges.clear();
|
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::CreatureStats *ptrCreatureStats = &ptrPlayer.getClass().getCreatureStats(ptrPlayer);
|
2016-09-29 07:19:39 +00:00
|
|
|
MWMechanics::DynamicStat<float> health(ptrCreatureStats->getHealth());
|
|
|
|
MWMechanics::DynamicStat<float> magicka(ptrCreatureStats->getMagicka());
|
|
|
|
MWMechanics::DynamicStat<float> fatigue(ptrCreatureStats->getFatigue());
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2016-09-29 07:19:39 +00:00
|
|
|
static MWMechanics::DynamicStat<float> oldHealth(ptrCreatureStats->getHealth());
|
|
|
|
static MWMechanics::DynamicStat<float> oldMagicka(ptrCreatureStats->getMagicka());
|
|
|
|
static MWMechanics::DynamicStat<float> oldFatigue(ptrCreatureStats->getFatigue());
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
// Update stats when they become 0 or they have changed enough
|
2017-06-27 13:01:04 +00:00
|
|
|
auto needUpdate = [](MWMechanics::DynamicStat<float> &oldVal, MWMechanics::DynamicStat<float> &newVal, int limit) {
|
|
|
|
return oldVal != newVal && (newVal.getCurrent() == 0 || oldVal.getCurrent() == 0
|
2017-07-26 17:15:35 +00:00
|
|
|
|| abs(oldVal.getCurrent() - newVal.getCurrent()) >= limit);
|
2017-06-27 13:01:04 +00:00
|
|
|
};
|
2017-05-05 21:59:54 +00:00
|
|
|
|
2018-04-20 21:43:49 +00:00
|
|
|
if (needUpdate(oldHealth, health, 2))
|
|
|
|
statsDynamicIndexChanges.push_back(0);
|
|
|
|
|
|
|
|
if (needUpdate(oldMagicka, magicka, 4))
|
|
|
|
statsDynamicIndexChanges.push_back(1);
|
|
|
|
|
|
|
|
if (needUpdate(oldFatigue, fatigue, 4))
|
|
|
|
statsDynamicIndexChanges.push_back(2);
|
|
|
|
|
|
|
|
if (statsDynamicIndexChanges.size() > 0 || forceUpdate)
|
2017-06-02 13:20:35 +00:00
|
|
|
{
|
|
|
|
oldHealth = health;
|
|
|
|
oldMagicka = magicka;
|
|
|
|
oldFatigue = fatigue;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
health.writeState(creatureStats.mDynamic[0]);
|
|
|
|
magicka.writeState(creatureStats.mDynamic[1]);
|
|
|
|
fatigue.writeState(creatureStats.mDynamic[2]);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2018-04-27 00:45:17 +00:00
|
|
|
exchangeFullInfo = false;
|
2017-06-02 13:20:35 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_STATS_DYNAMIC)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_STATS_DYNAMIC)->Send();
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-30 01:52:21 +00:00
|
|
|
void LocalPlayer::updateAttributes(bool forceUpdate)
|
2016-08-22 23:24:10 +00:00
|
|
|
{
|
2017-09-04 17:21:06 +00:00
|
|
|
// Only send attributes if we are not a werewolf, or they will be
|
|
|
|
// overwritten by the werewolf ones
|
|
|
|
if (isWerewolf) return;
|
|
|
|
|
2018-04-27 00:45:17 +00:00
|
|
|
if (attributeIndexChanges.size() > 0)
|
|
|
|
attributeIndexChanges.clear();
|
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
const MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2016-09-30 01:52:21 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < 8; ++i)
|
|
|
|
{
|
2017-11-30 21:03:34 +00:00
|
|
|
if (ptrNpcStats.getAttribute(i).getBase() != creatureStats.mAttributes[i].mBase ||
|
2018-04-19 14:26:20 +00:00
|
|
|
ptrNpcStats.getAttribute(i).getModifier() != creatureStats.mAttributes[i].mMod ||
|
|
|
|
ptrNpcStats.getSkillIncrease(i) != npcStats.mSkillIncrease[i] ||
|
|
|
|
forceUpdate)
|
2016-09-30 01:52:21 +00:00
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
attributeIndexChanges.push_back(i);
|
2017-01-25 15:06:15 +00:00
|
|
|
ptrNpcStats.getAttribute(i).writeState(creatureStats.mAttributes[i]);
|
2018-04-19 12:18:38 +00:00
|
|
|
npcStats.mSkillIncrease[i] = ptrNpcStats.getSkillIncrease(i);
|
|
|
|
}
|
2016-09-30 01:52:21 +00:00
|
|
|
}
|
2016-08-22 23:24:10 +00:00
|
|
|
|
2018-04-19 20:28:03 +00:00
|
|
|
if (attributeIndexChanges.size() > 0)
|
2016-09-30 01:52:21 +00:00
|
|
|
{
|
2018-04-27 00:45:17 +00:00
|
|
|
exchangeFullInfo = false;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_ATTRIBUTE)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_ATTRIBUTE)->Send();
|
2016-09-30 01:52:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::updateSkills(bool forceUpdate)
|
|
|
|
{
|
2017-09-04 17:21:06 +00:00
|
|
|
// Only send skills if we are not a werewolf, or they will be
|
|
|
|
// overwritten by the werewolf ones
|
|
|
|
if (isWerewolf) return;
|
|
|
|
|
2018-04-27 00:45:17 +00:00
|
|
|
if (skillIndexChanges.size() > 0)
|
|
|
|
skillIndexChanges.clear();
|
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
const MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2016-11-16 19:34:46 +00:00
|
|
|
|
2016-09-28 11:27:35 +00:00
|
|
|
for (int i = 0; i < 27; ++i)
|
|
|
|
{
|
2018-04-19 12:18:38 +00:00
|
|
|
// Update a skill if its base value has changed at all or its progress has changed enough
|
2017-12-04 13:06:27 +00:00
|
|
|
if (ptrNpcStats.getSkill(i).getBase() != npcStats.mSkills[i].mBase ||
|
2018-04-22 08:22:36 +00:00
|
|
|
ptrNpcStats.getSkill(i).getModifier() != npcStats.mSkills[i].mMod ||
|
2018-05-01 16:34:21 +00:00
|
|
|
abs(ptrNpcStats.getSkill(i).getProgress() - npcStats.mSkills[i].mProgress) > 0.75 ||
|
2018-04-19 12:18:38 +00:00
|
|
|
forceUpdate)
|
2016-09-28 11:27:35 +00:00
|
|
|
{
|
2018-04-19 20:28:03 +00:00
|
|
|
skillIndexChanges.push_back(i);
|
2018-04-20 19:46:16 +00:00
|
|
|
ptrNpcStats.getSkill(i).writeState(npcStats.mSkills[i]);
|
2016-08-22 23:24:10 +00:00
|
|
|
}
|
2016-09-30 04:15:59 +00:00
|
|
|
}
|
|
|
|
|
2018-04-19 20:28:03 +00:00
|
|
|
if (skillIndexChanges.size() > 0)
|
2016-09-30 04:15:59 +00:00
|
|
|
{
|
2018-04-27 00:45:17 +00:00
|
|
|
exchangeFullInfo = false;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SKILL)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SKILL)->Send();
|
2016-08-22 23:24:10 +00:00
|
|
|
}
|
2016-09-30 01:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::updateLevel(bool forceUpdate)
|
|
|
|
{
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2018-04-19 12:18:38 +00:00
|
|
|
const MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2016-08-22 23:24:10 +00:00
|
|
|
|
2018-04-19 12:18:38 +00:00
|
|
|
if (ptrNpcStats.getLevel() != creatureStats.mLevel ||
|
|
|
|
ptrNpcStats.getLevelProgress() != npcStats.mLevelProgress ||
|
|
|
|
forceUpdate)
|
2016-09-28 11:27:35 +00:00
|
|
|
{
|
2018-04-19 12:18:38 +00:00
|
|
|
creatureStats.mLevel = ptrNpcStats.getLevel();
|
|
|
|
npcStats.mLevelProgress = ptrNpcStats.getLevelProgress();
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_LEVEL)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_LEVEL)->Send();
|
2016-09-26 10:19:01 +00:00
|
|
|
}
|
2016-08-22 23:24:10 +00:00
|
|
|
}
|
|
|
|
|
2017-04-25 18:24:39 +00:00
|
|
|
void LocalPlayer::updateBounty(bool forceUpdate)
|
|
|
|
{
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
const MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2017-04-25 18:24:39 +00:00
|
|
|
|
|
|
|
if (ptrNpcStats.getBounty() != npcStats.mBounty || forceUpdate)
|
|
|
|
{
|
|
|
|
npcStats.mBounty = ptrNpcStats.getBounty();
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_BOUNTY)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_BOUNTY)->Send();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-30 20:55:29 +00:00
|
|
|
void LocalPlayer::updateReputation(bool forceUpdate)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
const MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
|
|
|
|
|
|
|
if (ptrNpcStats.getReputation() != npcStats.mReputation || forceUpdate)
|
|
|
|
{
|
|
|
|
npcStats.mReputation = ptrNpcStats.getReputation();
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_REPUTATION)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_REPUTATION)->Send();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
void LocalPlayer::updatePosition(bool forceUpdate)
|
|
|
|
{
|
2017-06-02 13:20:35 +00:00
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2016-08-23 19:27:12 +00:00
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
static bool posWasChanged = false;
|
|
|
|
static bool isJumping = false;
|
|
|
|
static bool sentJumpEnd = true;
|
|
|
|
static float oldRot[2] = {0};
|
2017-06-02 09:19:36 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
position = ptrPlayer.getRefData().getPosition();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
bool posIsChanging = (direction.pos[0] != 0 || direction.pos[1] != 0 ||
|
|
|
|
position.rot[0] != oldRot[0] || position.rot[2] != oldRot[1]);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-11-01 20:00:54 +00:00
|
|
|
// Animations can change a player's position without actually creating directional movement,
|
|
|
|
// so update positions accordingly
|
|
|
|
if (!posIsChanging && isPlayingAnimation)
|
|
|
|
{
|
|
|
|
if (MWBase::Environment::get().getMechanicsManager()->checkAnimationPlaying(ptrPlayer, animation.groupname))
|
|
|
|
posIsChanging = true;
|
|
|
|
else
|
|
|
|
isPlayingAnimation = false;
|
|
|
|
}
|
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
if (forceUpdate || posIsChanging || posWasChanged)
|
|
|
|
{
|
|
|
|
oldRot[0] = position.rot[0];
|
|
|
|
oldRot[1] = position.rot[2];
|
2017-06-02 09:19:36 +00:00
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
posWasChanged = posIsChanging;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
if (!isJumping && !world->isOnGround(ptrPlayer) && !world->isFlying(ptrPlayer))
|
2017-06-02 13:20:35 +00:00
|
|
|
isJumping = true;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-06-02 13:20:35 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_POSITION)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_POSITION)->Send();
|
|
|
|
}
|
2017-07-15 06:02:19 +00:00
|
|
|
else if (isJumping && world->isOnGround(ptrPlayer))
|
2017-06-02 13:20:35 +00:00
|
|
|
{
|
|
|
|
isJumping = false;
|
|
|
|
sentJumpEnd = false;
|
|
|
|
}
|
|
|
|
// Packet with jump end position has to be sent one tick after above check
|
|
|
|
else if (!sentJumpEnd)
|
|
|
|
{
|
|
|
|
sentJumpEnd = true;
|
2017-07-15 06:02:19 +00:00
|
|
|
position = ptrPlayer.getRefData().getPosition();
|
2017-06-02 13:20:35 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_POSITION)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_POSITION)->Send();
|
2016-08-23 19:27:12 +00:00
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::updateCell(bool forceUpdate)
|
2016-07-29 17:33:28 +00:00
|
|
|
{
|
2016-09-30 05:59:58 +00:00
|
|
|
const ESM::Cell *ptrCell = MWBase::Environment::get().getWorld()->getPlayerPtr().getCell()->getCell();
|
2016-07-29 17:33:28 +00:00
|
|
|
|
2017-04-23 09:41:13 +00:00
|
|
|
// If the LocalPlayer's Ptr cell is different from the LocalPlayer's packet cell, proceed
|
|
|
|
if (forceUpdate || !Main::get().getCellController()->isSameCell(*ptrCell, cell))
|
2016-08-03 21:55:28 +00:00
|
|
|
{
|
2017-04-23 09:41:13 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_CELL_CHANGE about LocalPlayer to server");
|
2016-09-29 10:10:32 +00:00
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Moved from %s to %s", cell.getDescription().c_str(),
|
|
|
|
ptrCell->getDescription().c_str());
|
2016-09-29 10:10:32 +00:00
|
|
|
|
2017-06-10 08:43:40 +00:00
|
|
|
if (!Misc::StringUtils::ciEqual(cell.mRegion, ptrCell->mRegion))
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Changed region from %s to %s",
|
|
|
|
cell.mRegion.empty() ? "none" : cell.mRegion.c_str(),
|
|
|
|
ptrCell->mRegion.empty() ? "none" : ptrCell->mRegion.c_str());
|
|
|
|
|
|
|
|
isChangingRegion = true;
|
|
|
|
}
|
|
|
|
|
2017-01-25 15:06:15 +00:00
|
|
|
cell = *ptrCell;
|
2017-08-04 18:45:52 +00:00
|
|
|
previousCellPosition = position;
|
2016-09-29 10:10:32 +00:00
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
// Make sure the position is updated before a cell packet is sent, or else
|
|
|
|
// cell change events in server scripts will have the wrong player position
|
|
|
|
updatePosition(true);
|
2016-09-29 10:10:32 +00:00
|
|
|
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CELL_CHANGE)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CELL_CHANGE)->Send();
|
2016-09-30 03:27:26 +00:00
|
|
|
|
2017-06-10 08:43:40 +00:00
|
|
|
isChangingRegion = false;
|
|
|
|
|
2016-11-17 13:32:04 +00:00
|
|
|
// Also check if the inventory needs to be updated
|
|
|
|
updateInventory();
|
2016-09-30 04:15:59 +00:00
|
|
|
}
|
2016-09-29 10:10:32 +00:00
|
|
|
}
|
|
|
|
|
2016-11-17 20:33:30 +00:00
|
|
|
void LocalPlayer::updateEquipment(bool forceUpdate)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-27 00:45:17 +00:00
|
|
|
if (equipmentIndexChanges.size() > 0)
|
|
|
|
equipmentIndexChanges.clear();
|
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::InventoryStore &invStore = ptrPlayer.getClass().getInventoryStore(ptrPlayer);
|
2016-09-30 09:36:20 +00:00
|
|
|
for (int slot = 0; slot < MWWorld::InventoryStore::Slots; slot++)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
auto &item = equipmentItems[slot];
|
2016-01-12 03:41:44 +00:00
|
|
|
MWWorld::ContainerStoreIterator it = invStore.getSlot(slot);
|
2018-01-25 22:45:39 +00:00
|
|
|
|
2017-06-29 06:52:23 +00:00
|
|
|
if (it != invStore.end())
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-29 02:40:42 +00:00
|
|
|
if (!::Misc::StringUtils::ciEqual(it->getCellRef().getRefId(), equipmentItems[slot].refId) ||
|
|
|
|
it->getCellRef().getCharge() != item.charge ||
|
|
|
|
it->getCellRef().getEnchantmentCharge() != item.enchantmentCharge ||
|
|
|
|
it->getRefData().getCount() != item.count ||
|
|
|
|
forceUpdate)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
equipmentIndexChanges.push_back(slot);
|
2017-06-29 06:52:23 +00:00
|
|
|
|
|
|
|
item.refId = it->getCellRef().getRefId();
|
|
|
|
item.charge = it->getCellRef().getCharge();
|
2017-12-23 11:16:38 +00:00
|
|
|
item.enchantmentCharge = it->getCellRef().getEnchantmentCharge();
|
[Client] Use correct count for items in equipment packets
Previously, throwing weapon sync was completely broken for players, as the count for their equipped throwing weapons was never set and – as a result – defaulted to a count of 1 on other clients. As a result, any time a player threw a dart, they would then appear as having switched to hand-to-hand for other players.
Moreover, the count of equipped items was mistakenly based on the total count of items with that refId in the inventory. As a result, if – for example – I equipped 1 Daedric Longsword and had 4 others in my inventory, my DedicatedPlayer on other clients would equip a Daedric Longsword with a count of 5. If I was overencumbered by having that many Daedric Longswords on me and then dropped 4 of them, allowing myself to move again, my DedicatedPlayer would still walk around with 5 Daedric Longswords and lack animations due to still being overencumbered on the other clients.
These problems were less prevalent for actors, but their equipment updating code has also been changed to match that of players.
2018-03-12 21:31:37 +00:00
|
|
|
item.count = it->getRefData().getCount();
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
2017-06-29 06:52:23 +00:00
|
|
|
else if (!item.refId.empty())
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
equipmentIndexChanges.push_back(slot);
|
2017-06-29 06:52:23 +00:00
|
|
|
item.refId = "";
|
|
|
|
item.count = 0;
|
2017-12-23 11:16:38 +00:00
|
|
|
item.charge = -1;
|
|
|
|
item.enchantmentCharge = -1;
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-20 19:46:16 +00:00
|
|
|
if (equipmentIndexChanges.size() > 0)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-04-27 00:45:17 +00:00
|
|
|
exchangeFullInfo = false;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_EQUIPMENT)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_EQUIPMENT)->Send();
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
void LocalPlayer::updateInventory(bool forceUpdate)
|
|
|
|
{
|
|
|
|
static bool invChanged = false;
|
2016-11-21 21:40:50 +00:00
|
|
|
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
if (forceUpdate)
|
|
|
|
invChanged = true;
|
|
|
|
|
2016-11-16 19:34:46 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
MWWorld::InventoryStore &ptrInventory = ptrPlayer.getClass().getInventoryStore(ptrPlayer);
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
mwmp::Item item;
|
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
auto setItem = [](Item &item, const MWWorld::Ptr &iter) {
|
|
|
|
item.refId = iter.getCellRef().getRefId();
|
|
|
|
if (item.refId.find("$dynamic") != string::npos) // skip generated items (self enchanted for e.g.)
|
|
|
|
return true;
|
|
|
|
item.count = iter.getRefData().getCount();
|
|
|
|
item.charge = iter.getCellRef().getCharge();
|
2017-12-23 11:16:38 +00:00
|
|
|
item.enchantmentCharge = iter.getCellRef().getEnchantmentCharge();
|
2017-06-27 12:43:39 +00:00
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2016-10-30 10:58:58 +00:00
|
|
|
if (!invChanged)
|
2016-11-16 19:34:46 +00:00
|
|
|
{
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &itemOld : inventoryChanges.items)
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
auto result = ptrInventory.begin();
|
2016-11-16 19:34:46 +00:00
|
|
|
for (; result != ptrInventory.end(); ++result)
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
if(setItem(item, *result))
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
continue;
|
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
if (item == itemOld)
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
break;
|
|
|
|
}
|
2016-11-16 19:34:46 +00:00
|
|
|
if (result == ptrInventory.end())
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
{
|
|
|
|
invChanged = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-11-16 19:34:46 +00:00
|
|
|
}
|
2016-11-21 21:40:50 +00:00
|
|
|
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
if (!invChanged)
|
2016-11-16 19:34:46 +00:00
|
|
|
{
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &iter : ptrInventory)
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
if(setItem(item, iter))
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
continue;
|
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
auto items = inventoryChanges.items;
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
if (find(items.begin(), items.end(), item) == items.end())
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
{
|
|
|
|
invChanged = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-11-16 19:34:46 +00:00
|
|
|
}
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
|
|
|
|
if (!invChanged)
|
|
|
|
return;
|
|
|
|
|
|
|
|
invChanged = false;
|
|
|
|
|
2016-11-16 19:34:46 +00:00
|
|
|
sendInventory();
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 19:06:04 +00:00
|
|
|
void LocalPlayer::updateAttack()
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2017-04-19 19:06:04 +00:00
|
|
|
if (attack.shouldSend)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2017-04-19 19:06:04 +00:00
|
|
|
if (attack.type == Attack::MAGIC)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2017-04-17 15:50:20 +00:00
|
|
|
attack.spellId = MWBase::Environment::get().getWindowManager()->getSelectedSpell();
|
2017-08-05 21:11:54 +00:00
|
|
|
|
|
|
|
if (attack.pressed)
|
|
|
|
attack.success = MechanicsHelper::getSpellSuccess(attack.spellId, getPlayerPtr());
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
2017-04-17 15:50:20 +00:00
|
|
|
|
2017-04-19 19:06:04 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_ATTACK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_ATTACK)->Send();
|
|
|
|
|
|
|
|
attack.shouldSend = false;
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::updateDeadState(bool forceUpdate)
|
|
|
|
{
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::NpcStats *ptrNpcStats = &ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2016-01-12 03:41:44 +00:00
|
|
|
static bool isDead = false;
|
|
|
|
|
2016-09-29 09:05:44 +00:00
|
|
|
if (ptrNpcStats->isDead() && !isDead)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2017-01-25 15:06:15 +00:00
|
|
|
creatureStats.mDead = true;
|
2017-03-03 21:36:29 +00:00
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_DEATH to server about myself");
|
2017-05-05 19:16:31 +00:00
|
|
|
LOG_APPEND(Log::LOG_INFO, "- deathReason was %s", deathReason.c_str());
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_DEATH)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_DEATH)->Send();
|
2016-01-12 03:41:44 +00:00
|
|
|
isDead = true;
|
|
|
|
}
|
2016-09-29 09:05:44 +00:00
|
|
|
else if (ptrNpcStats->getHealth().getCurrent() > 0 && isDead)
|
2017-05-05 19:16:31 +00:00
|
|
|
{
|
|
|
|
deathReason = "suicide";
|
2016-01-12 03:41:44 +00:00
|
|
|
isDead = false;
|
2017-05-05 19:16:31 +00:00
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2017-04-14 13:00:34 +00:00
|
|
|
void LocalPlayer::updateAnimFlags(bool forceUpdate)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::NpcStats ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2016-01-12 03:41:44 +00:00
|
|
|
using namespace MWMechanics;
|
|
|
|
|
2017-04-13 11:10:42 +00:00
|
|
|
static bool wasRunning = ptrNpcStats.getMovementFlag(CreatureStats::Flag_Run);
|
|
|
|
static bool wasSneaking = ptrNpcStats.getMovementFlag(CreatureStats::Flag_Sneak);
|
|
|
|
static bool wasForceJumping = ptrNpcStats.getMovementFlag(CreatureStats::Flag_ForceJump);
|
|
|
|
static bool wasForceMoveJumping = ptrNpcStats.getMovementFlag(CreatureStats::Flag_ForceMoveJump);
|
|
|
|
|
|
|
|
bool isRunning = ptrNpcStats.getMovementFlag(CreatureStats::Flag_Run);
|
|
|
|
bool isSneaking = ptrNpcStats.getMovementFlag(CreatureStats::Flag_Sneak);
|
|
|
|
bool isForceJumping = ptrNpcStats.getMovementFlag(CreatureStats::Flag_ForceJump);
|
|
|
|
bool isForceMoveJumping = ptrNpcStats.getMovementFlag(CreatureStats::Flag_ForceMoveJump);
|
2017-02-27 21:01:33 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
isFlying = world->isFlying(ptrPlayer);
|
|
|
|
bool isJumping = !world->isOnGround(ptrPlayer) && !isFlying;
|
2017-02-27 21:01:33 +00:00
|
|
|
|
|
|
|
// We need to send a new packet at the end of jumping and flying too,
|
|
|
|
// so keep track of what we were doing last frame
|
|
|
|
static bool wasJumping = false;
|
|
|
|
static bool wasFlying = false;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::DrawState_ currentDrawState = ptrPlayer.getClass().getNpcStats(ptrPlayer).getDrawState();
|
|
|
|
static MWMechanics::DrawState_ lastDrawState = ptrPlayer.getClass().getNpcStats(ptrPlayer).getDrawState();
|
2017-02-27 21:01:33 +00:00
|
|
|
|
2017-04-24 07:58:39 +00:00
|
|
|
if (wasRunning != isRunning ||
|
|
|
|
wasSneaking != isSneaking || wasForceJumping != isForceJumping ||
|
|
|
|
wasForceMoveJumping != isForceMoveJumping || lastDrawState != currentDrawState ||
|
2017-05-09 09:27:59 +00:00
|
|
|
wasJumping || isJumping || wasFlying != isFlying ||
|
2017-04-24 07:58:39 +00:00
|
|
|
forceUpdate)
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2017-04-13 11:10:42 +00:00
|
|
|
wasSneaking = isSneaking;
|
|
|
|
wasRunning = isRunning;
|
|
|
|
wasForceJumping = isForceJumping;
|
|
|
|
wasForceMoveJumping = isForceMoveJumping;
|
|
|
|
lastDrawState = currentDrawState;
|
2017-02-27 21:01:33 +00:00
|
|
|
|
|
|
|
wasFlying = isFlying;
|
|
|
|
wasJumping = isJumping;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
movementFlags = 0;
|
2017-04-13 11:10:42 +00:00
|
|
|
|
2016-01-12 03:41:44 +00:00
|
|
|
#define __SETFLAG(flag, value) (value) ? (movementFlags | flag) : (movementFlags & ~flag)
|
|
|
|
|
2017-04-13 11:10:42 +00:00
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_Sneak, isSneaking);
|
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_Run, isRunning);
|
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_ForceJump, isForceJumping);
|
2017-02-27 21:01:33 +00:00
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_ForceJump, isJumping);
|
2017-04-13 11:10:42 +00:00
|
|
|
movementFlags = __SETFLAG(CreatureStats::Flag_ForceMoveJump, isForceMoveJumping);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
|
|
|
#undef __SETFLAG
|
|
|
|
|
2017-04-13 11:10:42 +00:00
|
|
|
if (currentDrawState == MWMechanics::DrawState_Nothing)
|
2017-01-25 15:06:15 +00:00
|
|
|
drawState = 0;
|
2017-04-13 11:10:42 +00:00
|
|
|
else if (currentDrawState == MWMechanics::DrawState_Weapon)
|
2017-01-25 15:06:15 +00:00
|
|
|
drawState = 1;
|
2017-04-13 11:10:42 +00:00
|
|
|
else if (currentDrawState == MWMechanics::DrawState_Spell)
|
2017-01-25 15:06:15 +00:00
|
|
|
drawState = 2;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-02-27 21:01:33 +00:00
|
|
|
if (isJumping)
|
2017-04-13 11:10:42 +00:00
|
|
|
updatePosition(true); // fix position after jump;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-04-14 13:00:34 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_ANIM_FLAGS)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_ANIM_FLAGS)->Send();
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-19 13:18:37 +00:00
|
|
|
void LocalPlayer::addItems()
|
|
|
|
{
|
2017-01-20 07:07:07 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2017-01-19 13:18:37 +00:00
|
|
|
MWWorld::ContainerStore &ptrStore = ptrPlayer.getClass().getContainerStore(ptrPlayer);
|
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &item : inventoryChanges.items)
|
2017-01-19 13:18:37 +00:00
|
|
|
{
|
2017-08-31 05:35:46 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
MWWorld::Ptr itemPtr = *ptrStore.add(item.refId, item.count, ptrPlayer);
|
|
|
|
if (item.charge != -1)
|
|
|
|
itemPtr.getCellRef().setCharge(item.charge);
|
2017-12-23 11:16:38 +00:00
|
|
|
|
|
|
|
if (item.enchantmentCharge != -1)
|
|
|
|
itemPtr.getCellRef().setEnchantmentCharge(item.enchantmentCharge);
|
2017-08-31 05:35:46 +00:00
|
|
|
}
|
|
|
|
catch (std::exception&)
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Ignored addition of invalid inventory item %s", item.refId.c_str());
|
|
|
|
}
|
2017-01-19 13:18:37 +00:00
|
|
|
}
|
2018-03-26 07:01:26 +00:00
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->getInventoryWindow()->updateItemView();
|
2017-01-19 13:18:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::addSpells()
|
|
|
|
{
|
2017-01-20 07:07:07 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2017-01-19 13:18:37 +00:00
|
|
|
MWMechanics::Spells &ptrSpells = ptrPlayer.getClass().getCreatureStats(ptrPlayer).getSpells();
|
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &spell : spellbookChanges.spells)
|
2017-08-31 03:42:11 +00:00
|
|
|
// Only add spells that are ensured to exist
|
|
|
|
if (MWBase::Environment::get().getWorld()->getStore().get<ESM::Spell>().search(spell.mId))
|
|
|
|
ptrSpells.add(spell.mId);
|
2017-08-31 05:35:46 +00:00
|
|
|
else
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Ignored addition of invalid spell %s", spell.mId.c_str());
|
2017-01-19 13:18:37 +00:00
|
|
|
}
|
|
|
|
|
2017-01-24 17:32:25 +00:00
|
|
|
void LocalPlayer::addJournalItems()
|
|
|
|
{
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &journalItem : journalChanges.journalItems)
|
2017-01-24 17:32:25 +00:00
|
|
|
{
|
2017-08-31 06:08:29 +00:00
|
|
|
MWWorld::Ptr ptrFound;
|
|
|
|
|
2017-01-24 17:32:25 +00:00
|
|
|
if (journalItem.type == JournalItem::ENTRY)
|
|
|
|
{
|
2017-08-31 06:08:29 +00:00
|
|
|
ptrFound = MWBase::Environment::get().getWorld()->searchPtr(journalItem.actorRefId, false);
|
2017-01-24 17:32:25 +00:00
|
|
|
|
|
|
|
if (!ptrFound)
|
|
|
|
ptrFound = getPlayerPtr();
|
2017-08-31 06:08:29 +00:00
|
|
|
}
|
2017-01-24 17:32:25 +00:00
|
|
|
|
2017-08-31 06:08:29 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
if (journalItem.type == JournalItem::ENTRY)
|
|
|
|
MWBase::Environment::get().getJournal()->addEntry(journalItem.quest, journalItem.index, ptrFound);
|
|
|
|
else
|
|
|
|
MWBase::Environment::get().getJournal()->setJournalIndex(journalItem.quest, journalItem.index);
|
|
|
|
}
|
|
|
|
catch (std::exception&)
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Ignored addition of invalid journal quest %s", journalItem.quest.c_str());
|
2017-01-24 17:32:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-24 18:43:34 +00:00
|
|
|
void LocalPlayer::addTopics()
|
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
auto &env = MWBase::Environment::get();
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &topic : topicChanges.topics)
|
2017-05-24 18:43:34 +00:00
|
|
|
{
|
2017-05-31 11:11:16 +00:00
|
|
|
std::string topicId = topic.topicId;
|
2017-05-24 18:43:34 +00:00
|
|
|
|
2017-05-31 11:11:16 +00:00
|
|
|
// If we're using a translated version of Morrowind, translate this topic from English into our language
|
2017-06-27 12:43:39 +00:00
|
|
|
if (env.getWindowManager()->getTranslationDataStorage().hasTranslation())
|
|
|
|
topicId = env.getWindowManager()->getTranslationDataStorage().getLocalizedTopicId(topicId);
|
2017-05-31 11:11:16 +00:00
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
env.getDialogueManager()->addTopic(topicId);
|
2017-05-31 03:51:50 +00:00
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
if (env.getWindowManager()->containsMode(MWGui::GM_Dialogue))
|
2017-10-07 20:33:36 +00:00
|
|
|
env.getDialogueManager()->updateActorKnownTopics();
|
2017-05-24 18:43:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-19 13:18:37 +00:00
|
|
|
void LocalPlayer::removeItems()
|
|
|
|
{
|
2017-01-20 07:07:07 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2017-01-19 13:18:37 +00:00
|
|
|
MWWorld::ContainerStore &ptrStore = ptrPlayer.getClass().getContainerStore(ptrPlayer);
|
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &item : inventoryChanges.items)
|
2017-02-05 07:01:33 +00:00
|
|
|
ptrStore.remove(item.refId, item.count, ptrPlayer);
|
2017-01-19 13:18:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::removeSpells()
|
|
|
|
{
|
2017-01-20 07:07:07 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2017-01-19 13:18:37 +00:00
|
|
|
MWMechanics::Spells &ptrSpells = ptrPlayer.getClass().getCreatureStats(ptrPlayer).getSpells();
|
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
MWBase::WindowManager *wm = MWBase::Environment::get().getWindowManager();
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &spell : spellbookChanges.spells)
|
2017-01-19 13:18:37 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
ptrSpells.remove(spell.mId);
|
|
|
|
if (spell.mId == wm->getSelectedSpell())
|
2017-01-19 13:18:37 +00:00
|
|
|
wm->unsetSelectedSpell();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-27 17:09:55 +00:00
|
|
|
void LocalPlayer::resurrect()
|
|
|
|
{
|
|
|
|
creatureStats.mDead = false;
|
|
|
|
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
|
|
|
|
if (resurrectType == mwmp::RESURRECT_TYPE::IMPERIAL_SHRINE)
|
|
|
|
MWBase::Environment::get().getWorld()->teleportToClosestMarker(ptrPlayer, "divinemarker");
|
|
|
|
else if (resurrectType == mwmp::RESURRECT_TYPE::TRIBUNAL_TEMPLE)
|
|
|
|
MWBase::Environment::get().getWorld()->teleportToClosestMarker(ptrPlayer, "templemarker");
|
|
|
|
|
|
|
|
ptrPlayer.getClass().getCreatureStats(ptrPlayer).resurrect();
|
|
|
|
|
|
|
|
// The player could have died from a hand-to-hand attack, so reset their fatigue
|
|
|
|
// as well
|
|
|
|
if (creatureStats.mDynamic[2].mMod < 1)
|
|
|
|
creatureStats.mDynamic[2].mMod = 1;
|
|
|
|
|
|
|
|
creatureStats.mDynamic[2].mCurrent = creatureStats.mDynamic[2].mMod;
|
|
|
|
MWMechanics::DynamicStat<float> fatigue;
|
|
|
|
fatigue.readState(creatureStats.mDynamic[2]);
|
|
|
|
ptrPlayer.getClass().getCreatureStats(ptrPlayer).setFatigue(fatigue);
|
|
|
|
|
|
|
|
// If this player had a weapon or spell readied when dying, they will still have it
|
|
|
|
// readied but be unable to use it unless we clear it here
|
|
|
|
ptrPlayer.getClass().getNpcStats(ptrPlayer).setDrawState(MWMechanics::DrawState_Nothing);
|
|
|
|
|
|
|
|
// Record that the player has died since the last attempt was made to arrest them,
|
|
|
|
// used to make guards lenient enough to attempt an arrest again
|
|
|
|
diedSinceArrestAttempt = true;
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- diedSinceArrestAttempt is now true");
|
|
|
|
|
2018-02-13 19:54:46 +00:00
|
|
|
// Record that we are no longer a known werewolf, to avoid being attacked infinitely
|
|
|
|
MWBase::Environment::get().getWorld()->setGlobalInt("pcknownwerewolf", 0);
|
|
|
|
|
2018-01-27 18:19:39 +00:00
|
|
|
// Ensure we unequip any items with constant effects that can put us into an infinite
|
|
|
|
// death loop
|
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::DrainHealth);
|
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::FireDamage);
|
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::FrostDamage);
|
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::ShockDamage);
|
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::SunDamage);
|
|
|
|
|
2018-01-27 17:09:55 +00:00
|
|
|
Main::get().getNetworking()->getPlayerPacket(ID_PLAYER_RESURRECT)->setPlayer(this);
|
|
|
|
Main::get().getNetworking()->getPlayerPacket(ID_PLAYER_RESURRECT)->Send();
|
|
|
|
|
|
|
|
updateStatsDynamic(true);
|
|
|
|
}
|
|
|
|
|
2017-12-16 05:21:02 +00:00
|
|
|
void LocalPlayer::closeInventoryWindows()
|
|
|
|
{
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->containsMode(MWGui::GM_Container) ||
|
|
|
|
MWBase::Environment::get().getWindowManager()->containsMode(MWGui::GM_Inventory))
|
|
|
|
MWBase::Environment::get().getWindowManager()->popGuiMode();
|
|
|
|
|
2017-12-16 21:19:54 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->finishDragDrop();
|
2017-12-16 05:21:02 +00:00
|
|
|
}
|
|
|
|
|
2018-04-09 16:21:19 +00:00
|
|
|
void LocalPlayer::setCharacter()
|
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
|
|
|
|
// Ignore invalid races
|
|
|
|
if (world->getStore().get<ESM::Race>().search(npc.mRace) != 0)
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWorld()->getPlayer().setBirthSign(birthsign);
|
|
|
|
|
|
|
|
if (resetStats)
|
2018-04-10 15:22:27 +00:00
|
|
|
{
|
2018-04-09 16:21:19 +00:00
|
|
|
MWBase::Environment::get().getMechanicsManager()->setPlayerRace(npc.mRace, npc.isMale(), npc.mHead, npc.mHair);
|
2018-04-10 15:22:27 +00:00
|
|
|
setEquipment();
|
|
|
|
}
|
2018-04-09 16:21:19 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
ESM::NPC player = *world->getPlayerPtr().get<ESM::NPC>()->mBase;
|
|
|
|
|
|
|
|
player.mRace = npc.mRace;
|
|
|
|
player.mHead = npc.mHead;
|
|
|
|
player.mHair = npc.mHair;
|
|
|
|
player.setIsMale(npc.isMale());
|
|
|
|
world->createRecord(player);
|
|
|
|
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->playerLoaded();
|
|
|
|
}
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->getInventoryWindow()->rebuildAvatar();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Character update was ignored due to invalid race %s", npc.mRace.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setDynamicStats()
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2016-09-30 05:59:58 +00:00
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2016-09-30 05:59:58 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::CreatureStats *ptrCreatureStats = &ptrPlayer.getClass().getCreatureStats(ptrPlayer);
|
2016-09-30 05:59:58 +00:00
|
|
|
MWMechanics::DynamicStat<float> dynamicStat;
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i)
|
|
|
|
{
|
|
|
|
dynamicStat = ptrCreatureStats->getDynamic(i);
|
2017-01-25 15:06:15 +00:00
|
|
|
dynamicStat.setBase(creatureStats.mDynamic[i].mBase);
|
|
|
|
dynamicStat.setCurrent(creatureStats.mDynamic[i].mCurrent);
|
2016-09-30 05:59:58 +00:00
|
|
|
ptrCreatureStats->setDynamic(i, dynamicStat);
|
|
|
|
}
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setAttributes()
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2018-01-25 22:45:39 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::CreatureStats *ptrCreatureStats = &ptrPlayer.getClass().getCreatureStats(ptrPlayer);
|
2016-09-30 05:59:58 +00:00
|
|
|
MWMechanics::AttributeValue attributeValue;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2018-01-25 22:45:39 +00:00
|
|
|
for (int attributeIndex = 0; attributeIndex < 8; ++attributeIndex)
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
2017-12-02 07:22:36 +00:00
|
|
|
// If the server wants to clear our attribute's non-zero modifier, we need to remove
|
|
|
|
// the spell effect causing it, to avoid an infinite loop where the effect keeps resetting
|
|
|
|
// the modifier
|
2018-01-25 22:45:39 +00:00
|
|
|
if (creatureStats.mAttributes[attributeIndex].mMod == 0 && ptrCreatureStats->getAttribute(attributeIndex).getModifier() > 0)
|
|
|
|
{
|
|
|
|
ptrCreatureStats->getActiveSpells().purgeEffectByArg(ESM::MagicEffect::FortifyAttribute, attributeIndex);
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->updateMagicEffects(ptrPlayer);
|
|
|
|
|
|
|
|
// Is the modifier for this attribute still higher than 0? If so, unequip items that
|
|
|
|
// fortify the attribute
|
|
|
|
if (ptrCreatureStats->getAttribute(attributeIndex).getModifier() > 0)
|
|
|
|
{
|
2018-01-27 17:37:16 +00:00
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::FortifyAttribute, attributeIndex, -1);
|
2018-01-25 22:45:39 +00:00
|
|
|
mwmp::Main::get().getGUIController()->refreshGuiMode(MWGui::GM_Inventory);
|
|
|
|
}
|
|
|
|
}
|
2017-12-02 07:22:36 +00:00
|
|
|
|
2018-01-25 22:45:39 +00:00
|
|
|
attributeValue.readState(creatureStats.mAttributes[attributeIndex]);
|
|
|
|
ptrCreatureStats->setAttribute(attributeIndex, attributeValue);
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
|
|
|
}
|
2016-08-17 15:04:35 +00:00
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setSkills()
|
|
|
|
{
|
2018-01-25 22:45:39 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2016-08-17 15:04:35 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::NpcStats *ptrNpcStats = &ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2016-09-30 05:59:58 +00:00
|
|
|
MWMechanics::SkillValue skillValue;
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2018-01-25 22:45:39 +00:00
|
|
|
for (int skillIndex = 0; skillIndex < 27; ++skillIndex)
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
2017-12-04 13:06:27 +00:00
|
|
|
// If the server wants to clear our skill's non-zero modifier, we need to remove
|
|
|
|
// the spell effect causing it, to avoid an infinite loop where the effect keeps resetting
|
|
|
|
// the modifier
|
2018-01-25 22:45:39 +00:00
|
|
|
if (npcStats.mSkills[skillIndex].mMod == 0 && ptrNpcStats->getSkill(skillIndex).getModifier() > 0)
|
|
|
|
{
|
|
|
|
ptrNpcStats->getActiveSpells().purgeEffectByArg(ESM::MagicEffect::FortifySkill, skillIndex);
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->updateMagicEffects(ptrPlayer);
|
2017-12-04 13:06:27 +00:00
|
|
|
|
2018-01-25 22:45:39 +00:00
|
|
|
// Is the modifier for this skill still higher than 0? If so, unequip items that
|
|
|
|
// fortify the skill
|
|
|
|
if (ptrNpcStats->getSkill(skillIndex).getModifier() > 0)
|
|
|
|
{
|
2018-01-27 17:37:16 +00:00
|
|
|
MechanicsHelper::unequipItemsByEffect(ptrPlayer, ESM::Enchantment::ConstantEffect, ESM::MagicEffect::FortifySkill, -1, skillIndex);
|
2018-01-25 22:45:39 +00:00
|
|
|
mwmp::Main::get().getGUIController()->refreshGuiMode(MWGui::GM_Inventory);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
skillValue.readState(npcStats.mSkills[skillIndex]);
|
|
|
|
ptrNpcStats->setSkill(skillIndex, skillValue);
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2018-01-25 22:45:39 +00:00
|
|
|
for (int attributeIndex = 0; attributeIndex < 8; ++attributeIndex)
|
|
|
|
ptrNpcStats->setSkillIncrease(attributeIndex, npcStats.mSkillIncrease[attributeIndex]);
|
2016-01-12 03:41:44 +00:00
|
|
|
|
2017-01-25 15:06:15 +00:00
|
|
|
ptrNpcStats->setLevelProgress(npcStats.mLevelProgress);
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setLevel()
|
2016-01-12 03:41:44 +00:00
|
|
|
{
|
2016-09-30 05:59:58 +00:00
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2016-08-26 21:14:50 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::CreatureStats *ptrCreatureStats = &ptrPlayer.getClass().getCreatureStats(ptrPlayer);
|
2017-01-25 15:06:15 +00:00
|
|
|
ptrCreatureStats->setLevel(creatureStats.mLevel);
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
2016-09-02 03:57:13 +00:00
|
|
|
|
2017-04-25 18:24:39 +00:00
|
|
|
void LocalPlayer::setBounty()
|
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2017-04-25 18:24:39 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
MWMechanics::NpcStats *ptrNpcStats = &ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2017-04-25 18:24:39 +00:00
|
|
|
ptrNpcStats->setBounty(npcStats.mBounty);
|
|
|
|
}
|
|
|
|
|
2018-01-30 20:55:29 +00:00
|
|
|
void LocalPlayer::setReputation()
|
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
|
|
|
|
|
|
|
MWMechanics::NpcStats *ptrNpcStats = &ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
|
|
|
ptrNpcStats->setReputation(npcStats.mReputation);
|
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setPosition()
|
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2016-09-30 05:59:58 +00:00
|
|
|
|
2016-11-12 20:21:02 +00:00
|
|
|
// If we're ignoring this position packet because of an invalid cell change,
|
|
|
|
// don't make the next one get ignored as well
|
|
|
|
if (ignorePosPacket)
|
|
|
|
ignorePosPacket = false;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
world->getPlayer().setTeleported(true);
|
2017-08-04 18:43:01 +00:00
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
world->moveObject(ptrPlayer, position.pos[0], position.pos[1], position.pos[2]);
|
|
|
|
world->rotateObject(ptrPlayer, position.rot[0], position.rot[1], position.rot[2]);
|
2017-08-04 18:43:01 +00:00
|
|
|
world->setInertialForce(ptrPlayer, osg::Vec3f(0.f, 0.f, 0.f));
|
2016-11-12 20:21:02 +00:00
|
|
|
}
|
2016-09-30 05:59:58 +00:00
|
|
|
|
|
|
|
updatePosition(true);
|
2017-02-23 04:25:46 +00:00
|
|
|
|
|
|
|
// Make sure we update our draw state, or we'll end up with the wrong one
|
2017-04-14 13:00:34 +00:00
|
|
|
updateAnimFlags(true);
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
|
|
|
|
2018-04-29 20:47:17 +00:00
|
|
|
void LocalPlayer::setMomentum()
|
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
|
|
|
world->setInertialForce(ptrPlayer, momentum.asVec3());
|
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setCell()
|
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = world->getPlayerPtr();
|
2016-09-30 05:59:58 +00:00
|
|
|
ESM::Position pos;
|
|
|
|
|
2017-12-16 05:21:02 +00:00
|
|
|
// To avoid crashes, close container windows this player may be in
|
|
|
|
closeInventoryWindows();
|
2017-02-20 23:01:30 +00:00
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
world->getPlayer().setTeleported(true);
|
|
|
|
|
2017-01-25 15:06:15 +00:00
|
|
|
int x = cell.mData.mX;
|
|
|
|
int y = cell.mData.mY;
|
2016-09-30 05:59:58 +00:00
|
|
|
|
2017-01-25 15:06:15 +00:00
|
|
|
if (cell.isExterior())
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
|
|
|
world->indexToPosition(x, y, pos.pos[0], pos.pos[1], true);
|
|
|
|
pos.pos[2] = 0;
|
|
|
|
|
|
|
|
pos.rot[0] = pos.rot[1] = pos.rot[2] = 0;
|
|
|
|
|
|
|
|
world->changeToExteriorCell(pos, true);
|
2017-07-15 06:02:19 +00:00
|
|
|
world->fixPosition(ptrPlayer);
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
2017-01-25 15:06:15 +00:00
|
|
|
else if (world->findExteriorPosition(cell.mName, pos))
|
2016-09-30 05:59:58 +00:00
|
|
|
{
|
|
|
|
world->changeToExteriorCell(pos, true);
|
2017-07-15 06:02:19 +00:00
|
|
|
world->fixPosition(ptrPlayer);
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-11-12 20:21:02 +00:00
|
|
|
try
|
|
|
|
{
|
2017-01-25 15:06:15 +00:00
|
|
|
world->findInteriorPosition(cell.mName, pos);
|
|
|
|
world->changeToInteriorCell(cell.mName, pos, true);
|
2016-11-12 20:21:02 +00:00
|
|
|
}
|
|
|
|
// If we've been sent to an invalid interior, ignore the incoming
|
|
|
|
// packet about our position in that cell
|
|
|
|
catch (std::exception&)
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "%s", "- Cell doesn't exist on this client");
|
|
|
|
ignorePosPacket = true;
|
|
|
|
}
|
2016-09-30 05:59:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
updateCell(true);
|
2016-01-12 03:41:44 +00:00
|
|
|
}
|
2016-08-30 03:17:06 +00:00
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::setClass()
|
2016-08-30 03:17:06 +00:00
|
|
|
{
|
2018-01-24 00:47:56 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Received ID_PLAYER_CLASS from server");
|
|
|
|
|
2016-09-29 09:05:44 +00:00
|
|
|
if (charClass.mId.empty()) // custom class
|
2016-08-30 03:17:06 +00:00
|
|
|
{
|
2016-09-29 09:05:44 +00:00
|
|
|
charClass.mData.mIsPlayable = 0x1;
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->setPlayerClass(charClass);
|
|
|
|
MWBase::Environment::get().getWindowManager()->setPlayerClass(charClass);
|
2016-08-30 03:17:06 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-01-24 00:47:56 +00:00
|
|
|
const ESM::Class *existingCharClass = MWBase::Environment::get().getWorld()->getStore().get<ESM::Class>().search(charClass.mId);
|
2016-09-29 09:05:44 +00:00
|
|
|
|
|
|
|
if (existingCharClass)
|
2018-01-24 00:47:56 +00:00
|
|
|
{
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->setPlayerClass(charClass.mId);
|
2016-09-29 09:05:44 +00:00
|
|
|
MWBase::Environment::get().getWindowManager()->setPlayerClass(charClass);
|
2018-01-24 00:47:56 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Ignored invalid default class %s", charClass.mId.c_str());
|
2016-08-30 03:17:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-17 20:33:30 +00:00
|
|
|
void LocalPlayer::setEquipment()
|
2016-09-30 09:36:20 +00:00
|
|
|
{
|
2016-11-15 19:54:06 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2016-09-30 09:36:20 +00:00
|
|
|
|
|
|
|
MWWorld::InventoryStore &ptrInventory = ptrPlayer.getClass().getInventoryStore(ptrPlayer);
|
|
|
|
|
|
|
|
for (int slot = 0; slot < MWWorld::InventoryStore::Slots; slot++)
|
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
mwmp::Item ¤tItem = equipmentItems[slot];
|
2016-09-30 09:36:20 +00:00
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
if (!currentItem.refId.empty())
|
2016-09-30 09:36:20 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
auto it = find_if(ptrInventory.begin(), ptrInventory.end(), [¤tItem](const MWWorld::Ptr &a) {
|
|
|
|
return Misc::StringUtils::ciEqual(a.getCellRef().getRefId(), currentItem.refId);
|
|
|
|
});
|
|
|
|
|
2017-08-31 05:35:46 +00:00
|
|
|
if (it == ptrInventory.end()) // If the item is not in our inventory, add it
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
auto equipmentItem = equipmentItems[slot];
|
2017-08-31 05:35:46 +00:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
auto addIter = ptrInventory.ContainerStore::add(equipmentItem.refId.c_str(), equipmentItem.count, ptrPlayer);
|
2017-08-31 05:35:46 +00:00
|
|
|
ptrInventory.equip(slot, addIter, ptrPlayer);
|
|
|
|
}
|
|
|
|
catch (std::exception&)
|
|
|
|
{
|
2018-04-20 19:46:16 +00:00
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Ignored addition of invalid equipment item %s", equipmentItem.refId.c_str());
|
2017-08-31 05:35:46 +00:00
|
|
|
}
|
Implement inventory functions
AddItem, RemoveItem, GetItemName, GetItemCount, GetItemHealth, GetInventorySize SendInventory
Example:
```lua
tes3mp.AddItem(pid, "glass dagger", 1, 50)
tes3mp.AddItem(pid, "glass dagger", 1, -1)
tes3mp.SendInventory(pid)
tes3mp.RemoveItem(pid, "glass dagger", 1)
tes3mp.SendInventory(pid)
local invSize = tes3mp.GetInventorySize(pid) - 1
for i = 0, invSize do
print(("%s %d %d"):format(tes3mp.GetItemName(pid, i), tes3mp.GetItemCount(pid, i), tes3mp.GetItemHealth(pid, i)))
end
```
2016-10-22 18:57:37 +00:00
|
|
|
}
|
|
|
|
else
|
2018-04-10 15:22:27 +00:00
|
|
|
{
|
|
|
|
// Don't try to equip an item that is already equipped
|
2018-05-10 04:16:33 +00:00
|
|
|
if (ptrInventory.getSlot(slot) != it)
|
2018-04-10 15:22:27 +00:00
|
|
|
ptrInventory.equip(slot, it, ptrPlayer);
|
|
|
|
}
|
2016-09-30 09:36:20 +00:00
|
|
|
}
|
2016-11-17 23:50:55 +00:00
|
|
|
else
|
|
|
|
ptrInventory.unequipSlot(slot, ptrPlayer);
|
2016-09-30 09:36:20 +00:00
|
|
|
}
|
2018-03-26 07:01:26 +00:00
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->getInventoryWindow()->updatePlayer();
|
2016-09-30 09:36:20 +00:00
|
|
|
}
|
|
|
|
|
2017-01-19 13:18:37 +00:00
|
|
|
void LocalPlayer::setInventory()
|
|
|
|
{
|
2017-01-20 07:07:07 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2017-01-19 13:18:37 +00:00
|
|
|
MWWorld::ContainerStore &ptrStore = ptrPlayer.getClass().getContainerStore(ptrPlayer);
|
|
|
|
|
2017-12-25 03:41:13 +00:00
|
|
|
// Ensure no item is being drag and dropped
|
|
|
|
MWBase::Environment::get().getWindowManager()->finishDragDrop();
|
|
|
|
|
2017-01-19 13:18:37 +00:00
|
|
|
// Clear items in inventory
|
|
|
|
ptrStore.clear();
|
|
|
|
|
|
|
|
// Proceed by adding items
|
|
|
|
addItems();
|
|
|
|
|
|
|
|
// Don't automatically setEquipment() here, or the player could end
|
|
|
|
// up getting a new set of their starting clothes, or other items
|
|
|
|
// supposed to no longer exist
|
|
|
|
//
|
|
|
|
// Instead, expect server scripts to do that manually
|
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::setSpellbook()
|
|
|
|
{
|
2017-01-20 07:07:07 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2017-01-19 13:18:37 +00:00
|
|
|
MWMechanics::Spells &ptrSpells = ptrPlayer.getClass().getCreatureStats(ptrPlayer).getSpells();
|
|
|
|
|
2017-01-20 06:06:48 +00:00
|
|
|
// Clear spells in spellbook, while ignoring abilities, powers, etc.
|
2017-01-30 14:53:22 +00:00
|
|
|
while (true)
|
2017-01-20 06:06:48 +00:00
|
|
|
{
|
2017-01-30 14:37:09 +00:00
|
|
|
MWMechanics::Spells::TIterator iter = ptrSpells.begin();
|
|
|
|
for (; iter != ptrSpells.end(); iter++)
|
2017-01-20 06:06:48 +00:00
|
|
|
{
|
2017-01-30 14:37:09 +00:00
|
|
|
const ESM::Spell *spell = iter->first;
|
|
|
|
if (spell->mData.mType == ESM::Spell::ST_Spell)
|
|
|
|
{
|
|
|
|
ptrSpells.remove(spell->mId);
|
|
|
|
break;
|
|
|
|
}
|
2017-01-20 06:06:48 +00:00
|
|
|
}
|
2017-01-30 14:53:22 +00:00
|
|
|
if (iter == ptrSpells.end())
|
2017-01-30 14:37:09 +00:00
|
|
|
break;
|
2017-01-20 06:06:48 +00:00
|
|
|
}
|
2017-01-19 13:18:37 +00:00
|
|
|
|
|
|
|
// Proceed by adding spells
|
|
|
|
addSpells();
|
|
|
|
}
|
|
|
|
|
2017-10-25 04:21:00 +00:00
|
|
|
void LocalPlayer::setQuickKeys()
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Received ID_PLAYER_QUICKKEYS from server");
|
|
|
|
|
|
|
|
// Because we send QuickKeys packets from the same OpenMW methods that we use to set received ones with,
|
|
|
|
// we need a boolean to prevent their sending here
|
|
|
|
isReceivingQuickKeys = true;
|
|
|
|
|
|
|
|
for (const auto &quickKey : quickKeyChanges.quickKeys)
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- slot: %i, type: %i, itemId: %s", quickKey.slot, quickKey.type, quickKey.itemId.c_str());
|
|
|
|
|
|
|
|
if (quickKey.type == QuickKey::ITEM || quickKey.type == QuickKey::ITEM_MAGIC)
|
|
|
|
{
|
|
|
|
MWWorld::InventoryStore &ptrInventory = ptrPlayer.getClass().getInventoryStore(ptrPlayer);
|
|
|
|
|
|
|
|
auto it = find_if(ptrInventory.begin(), ptrInventory.end(), [&quickKey](const MWWorld::Ptr &inventoryItem) {
|
|
|
|
return Misc::StringUtils::ciEqual(inventoryItem.getCellRef().getRefId(), quickKey.itemId);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (it != ptrInventory.end())
|
|
|
|
MWBase::Environment::get().getWindowManager()->setQuickKey(quickKey.slot, quickKey.type, (*it));
|
|
|
|
}
|
|
|
|
else if (quickKey.type == QuickKey::MAGIC)
|
|
|
|
{
|
|
|
|
MWMechanics::Spells &ptrSpells = ptrPlayer.getClass().getCreatureStats(ptrPlayer).getSpells();
|
|
|
|
bool hasSpell = false;
|
|
|
|
|
|
|
|
MWMechanics::Spells::TIterator iter = ptrSpells.begin();
|
|
|
|
for (; iter != ptrSpells.end(); iter++)
|
|
|
|
{
|
|
|
|
const ESM::Spell *spell = iter->first;
|
|
|
|
if (Misc::StringUtils::ciEqual(spell->mId, quickKey.itemId))
|
|
|
|
{
|
|
|
|
hasSpell = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hasSpell)
|
|
|
|
MWBase::Environment::get().getWindowManager()->setQuickKey(quickKey.slot, quickKey.type, 0, quickKey.itemId);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
MWBase::Environment::get().getWindowManager()->setQuickKey(quickKey.slot, quickKey.type, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
isReceivingQuickKeys = false;
|
|
|
|
}
|
|
|
|
|
2017-05-18 16:27:20 +00:00
|
|
|
void LocalPlayer::setFactions()
|
|
|
|
{
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2017-05-18 16:27:20 +00:00
|
|
|
|
2018-01-24 00:25:44 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Received ID_PLAYER_FACTION from server\n- action: %i", factionChanges.action);
|
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &faction : factionChanges.factions)
|
2017-05-18 16:27:20 +00:00
|
|
|
{
|
2018-01-24 00:25:44 +00:00
|
|
|
const ESM::Faction *esmFaction = MWBase::Environment::get().getWorld()->getStore().get<ESM::Faction>().search(faction.factionId);
|
|
|
|
|
|
|
|
if (!esmFaction)
|
|
|
|
{
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- Ignored invalid faction %s", faction.factionId.c_str());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2017-05-18 16:27:20 +00:00
|
|
|
// If the player isn't in this faction, make them join it
|
|
|
|
if (!ptrNpcStats.isInFaction(faction.factionId))
|
|
|
|
ptrNpcStats.joinFaction(faction.factionId);
|
|
|
|
|
2017-07-08 09:27:48 +00:00
|
|
|
if (factionChanges.action == mwmp::FactionChanges::RANK)
|
2017-05-18 16:27:20 +00:00
|
|
|
{
|
2017-07-05 04:05:52 +00:00
|
|
|
// While the faction rank is different in the packet than in the NpcStats,
|
|
|
|
// adjust the NpcStats accordingly
|
|
|
|
while (faction.rank != ptrNpcStats.getFactionRanks().at(faction.factionId))
|
|
|
|
{
|
|
|
|
if (faction.rank > ptrNpcStats.getFactionRanks().at(faction.factionId))
|
|
|
|
ptrNpcStats.raiseRank(faction.factionId);
|
|
|
|
else
|
|
|
|
ptrNpcStats.lowerRank(faction.factionId);
|
|
|
|
}
|
2017-05-18 16:27:20 +00:00
|
|
|
}
|
2017-07-08 09:27:48 +00:00
|
|
|
else if (factionChanges.action == mwmp::FactionChanges::EXPULSION)
|
2017-05-18 16:27:20 +00:00
|
|
|
{
|
2017-07-05 04:05:52 +00:00
|
|
|
// If the expelled state is different in the packet than in the NpcStats,
|
|
|
|
// adjust the NpcStats accordingly
|
|
|
|
if (faction.isExpelled != ptrNpcStats.getExpelled(faction.factionId))
|
|
|
|
{
|
|
|
|
if (faction.isExpelled)
|
|
|
|
ptrNpcStats.expell(faction.factionId);
|
|
|
|
else
|
|
|
|
ptrNpcStats.clearExpelled(faction.factionId);
|
|
|
|
}
|
2017-05-18 16:27:20 +00:00
|
|
|
}
|
2017-07-13 10:36:00 +00:00
|
|
|
|
|
|
|
else if (factionChanges.action == mwmp::FactionChanges::REPUTATION)
|
|
|
|
ptrNpcStats.setFactionReputation(faction.factionId, faction.reputation);
|
2017-05-18 16:27:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-10 11:04:19 +00:00
|
|
|
void LocalPlayer::setKills()
|
|
|
|
{
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &kill : killChanges.kills)
|
2017-06-27 12:43:39 +00:00
|
|
|
MWBase::Environment::get().getMechanicsManager()->setDeaths(kill.refId, kill.number);
|
2017-06-10 11:04:19 +00:00
|
|
|
}
|
|
|
|
|
2017-06-27 05:27:14 +00:00
|
|
|
void LocalPlayer::setBooks()
|
|
|
|
{
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2017-06-27 05:27:14 +00:00
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &book : bookChanges.books)
|
2017-06-27 12:43:39 +00:00
|
|
|
ptrNpcStats.flagAsUsed(book.bookId);
|
2017-06-27 05:27:14 +00:00
|
|
|
}
|
|
|
|
|
2017-07-03 06:28:27 +00:00
|
|
|
void LocalPlayer::setMapExplored()
|
|
|
|
{
|
2017-07-15 06:02:19 +00:00
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
MWMechanics::NpcStats &ptrNpcStats = ptrPlayer.getClass().getNpcStats(ptrPlayer);
|
2017-07-03 06:28:27 +00:00
|
|
|
|
2018-05-08 02:57:04 +00:00
|
|
|
for (const auto &mapTile : mapChanges.mapTiles)
|
2017-07-03 06:28:27 +00:00
|
|
|
{
|
2018-05-08 02:57:04 +00:00
|
|
|
const MWWorld::CellStore *cellStore = MWBase::Environment::get().getWorld()->getExterior(mapTile.x, mapTile.y);
|
2017-07-03 06:28:27 +00:00
|
|
|
|
2018-05-08 02:57:04 +00:00
|
|
|
if (!cellStore->getCell()->mName.empty())
|
|
|
|
MWBase::Environment::get().getWindowManager()->addVisitedLocation(cellStore->getCell()->mName, mapTile.x, mapTile.y);
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->setGlobalMapImage(mapTile.x, mapTile.y, mapTile.imageData);
|
2017-07-03 06:28:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-15 06:02:19 +00:00
|
|
|
void LocalPlayer::setShapeshift()
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
2018-02-01 00:11:45 +00:00
|
|
|
|
|
|
|
MWBase::Environment::get().getWorld()->scaleObject(ptrPlayer, scale);
|
2017-07-15 06:02:19 +00:00
|
|
|
MWBase::Environment::get().getMechanicsManager()->setWerewolf(ptrPlayer, isWerewolf);
|
|
|
|
}
|
|
|
|
|
2018-01-31 02:50:29 +00:00
|
|
|
void LocalPlayer::setMarkLocation()
|
|
|
|
{
|
|
|
|
MWWorld::CellStore *ptrCellStore = Main::get().getCellController()->getCellStore(markCell);
|
|
|
|
|
|
|
|
if (ptrCellStore)
|
|
|
|
MWBase::Environment::get().getWorld()->getPlayer().markPosition(ptrCellStore, markPosition);
|
|
|
|
}
|
|
|
|
|
2018-02-06 04:36:46 +00:00
|
|
|
void LocalPlayer::setSelectedSpell()
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
|
|
|
|
MWMechanics::CreatureStats& stats = ptrPlayer.getClass().getCreatureStats(ptrPlayer);
|
|
|
|
MWMechanics::Spells& spells = stats.getSpells();
|
|
|
|
|
|
|
|
if (!spells.hasSpell(selectedSpellId))
|
|
|
|
return;
|
|
|
|
|
|
|
|
MWBase::Environment::get().getWindowManager()->setSelectedSpell(selectedSpellId,
|
|
|
|
int(MWMechanics::getSpellSuccessChance(selectedSpellId, ptrPlayer)));
|
|
|
|
}
|
|
|
|
|
2016-09-30 05:59:58 +00:00
|
|
|
void LocalPlayer::sendClass()
|
2016-08-30 03:17:06 +00:00
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
2018-01-24 00:47:56 +00:00
|
|
|
const ESM::NPC *npcBase = world->getPlayerPtr().get<ESM::NPC>()->mBase;
|
|
|
|
const ESM::Class *esmClass = world->getStore().get<ESM::Class>().find(npcBase->mClass);
|
2016-08-30 03:17:06 +00:00
|
|
|
|
2018-01-24 00:47:56 +00:00
|
|
|
if (npcBase->mClass.find("$dynamic") != string::npos) // custom class
|
2016-08-30 03:17:06 +00:00
|
|
|
{
|
2016-09-29 09:05:44 +00:00
|
|
|
charClass.mId = "";
|
2018-01-24 00:47:56 +00:00
|
|
|
charClass.mName = esmClass->mName;
|
|
|
|
charClass.mDescription = esmClass->mDescription;
|
|
|
|
charClass.mData = esmClass->mData;
|
2016-08-30 03:17:06 +00:00
|
|
|
}
|
|
|
|
else
|
2018-01-24 00:47:56 +00:00
|
|
|
charClass.mId = esmClass->mId;
|
2016-08-30 03:17:06 +00:00
|
|
|
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CHARCLASS)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CHARCLASS)->Send();
|
2016-08-30 03:17:06 +00:00
|
|
|
}
|
2016-09-30 05:59:58 +00:00
|
|
|
|
2016-11-16 19:34:46 +00:00
|
|
|
void LocalPlayer::sendInventory()
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
MWWorld::InventoryStore &ptrInventory = ptrPlayer.getClass().getInventoryStore(ptrPlayer);
|
|
|
|
mwmp::Item item;
|
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
inventoryChanges.items.clear();
|
2016-11-16 19:34:46 +00:00
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &iter : ptrInventory)
|
2016-11-16 19:34:46 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
item.refId = iter.getCellRef().getRefId();
|
2017-02-05 07:01:33 +00:00
|
|
|
if (item.refId.find("$dynamic") != string::npos) // skip generated items (self enchanted for e.g.)
|
2016-11-16 19:34:46 +00:00
|
|
|
continue;
|
|
|
|
|
2017-06-27 12:43:39 +00:00
|
|
|
item.count = iter.getRefData().getCount();
|
|
|
|
item.charge = iter.getCellRef().getCharge();
|
2017-12-23 11:16:38 +00:00
|
|
|
item.enchantmentCharge = iter.getCellRef().getEnchantmentCharge();
|
2016-11-16 19:34:46 +00:00
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
inventoryChanges.items.push_back(item);
|
2016-11-16 19:34:46 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
inventoryChanges.count = (unsigned int) inventoryChanges.items.size();
|
|
|
|
inventoryChanges.action = InventoryChanges::SET;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_INVENTORY)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_INVENTORY)->Send();
|
2016-11-16 19:34:46 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 07:07:07 +00:00
|
|
|
void LocalPlayer::sendSpellbook()
|
|
|
|
{
|
|
|
|
MWWorld::Ptr ptrPlayer = getPlayerPtr();
|
|
|
|
MWMechanics::Spells &ptrSpells = ptrPlayer.getClass().getCreatureStats(ptrPlayer).getSpells();
|
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.spells.clear();
|
2017-01-20 07:07:07 +00:00
|
|
|
|
|
|
|
// Send spells in spellbook, while ignoring abilities, powers, etc.
|
2017-06-27 14:27:02 +00:00
|
|
|
for (const auto &spell : ptrSpells)
|
2017-01-20 07:07:07 +00:00
|
|
|
{
|
2017-06-27 12:43:39 +00:00
|
|
|
if (spell.first->mData.mType == ESM::Spell::ST_Spell)
|
|
|
|
spellbookChanges.spells.push_back(*spell.first);
|
2017-01-20 07:07:07 +00:00
|
|
|
}
|
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.action = SpellbookChanges::SET;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->Send();
|
2017-01-20 07:07:07 +00:00
|
|
|
}
|
|
|
|
|
2017-02-03 18:27:40 +00:00
|
|
|
void LocalPlayer::sendCellStates()
|
|
|
|
{
|
2017-04-04 05:24:11 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_CELL_STATE to server");
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CELL_STATE)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_CELL_STATE)->Send();
|
2017-02-03 18:27:40 +00:00
|
|
|
}
|
|
|
|
|
2016-11-21 04:07:29 +00:00
|
|
|
void LocalPlayer::sendSpellAddition(std::string id)
|
|
|
|
{
|
2016-12-29 13:19:26 +00:00
|
|
|
if (id.find("$dynamic") != string::npos) // skip custom spells
|
|
|
|
return;
|
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.spells.clear();
|
2016-11-21 21:40:50 +00:00
|
|
|
|
2016-12-29 13:19:26 +00:00
|
|
|
ESM::Spell spell;
|
|
|
|
spell.mId = id;
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.spells.push_back(spell);
|
2016-11-21 04:07:29 +00:00
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.action = SpellbookChanges::ADD;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->Send();
|
2016-11-21 04:07:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::sendSpellRemoval(std::string id)
|
|
|
|
{
|
2016-12-29 13:19:26 +00:00
|
|
|
if (id.find("$dynamic") != string::npos) // skip custom spells
|
|
|
|
return;
|
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.spells.clear();
|
2016-11-21 04:07:29 +00:00
|
|
|
|
2016-12-29 13:19:26 +00:00
|
|
|
ESM::Spell spell;
|
|
|
|
spell.mId = id;
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.spells.push_back(spell);
|
2016-11-21 04:07:29 +00:00
|
|
|
|
2017-01-20 10:43:05 +00:00
|
|
|
spellbookChanges.action = SpellbookChanges::REMOVE;
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->Send();
|
2016-11-21 04:07:29 +00:00
|
|
|
}
|
|
|
|
|
2016-12-29 13:19:26 +00:00
|
|
|
void LocalPlayer::sendSpellAddition(const ESM::Spell &spell)
|
|
|
|
{
|
2017-08-31 14:07:45 +00:00
|
|
|
/*
|
2017-07-30 17:45:47 +00:00
|
|
|
spellbookChanges.spells.clear();
|
|
|
|
|
|
|
|
spellbookChanges.spells.push_back(spell);
|
|
|
|
|
|
|
|
spellbookChanges.action = SpellbookChanges::ADD;
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->Send();
|
2017-08-31 14:07:45 +00:00
|
|
|
*/
|
2016-12-29 13:19:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::sendSpellRemoval(const ESM::Spell &spell)
|
|
|
|
{
|
2017-08-31 14:07:45 +00:00
|
|
|
/*
|
2017-07-30 17:45:47 +00:00
|
|
|
spellbookChanges.spells.clear();
|
|
|
|
|
|
|
|
spellbookChanges.spells.push_back(spell);
|
|
|
|
|
|
|
|
spellbookChanges.action = SpellbookChanges::REMOVE;
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SPELLBOOK)->Send();
|
2017-08-31 14:07:45 +00:00
|
|
|
*/
|
2016-12-29 13:19:26 +00:00
|
|
|
}
|
|
|
|
|
2017-10-25 04:21:00 +00:00
|
|
|
void LocalPlayer::sendQuickKey(unsigned short slot, int type, const std::string& itemId)
|
|
|
|
{
|
|
|
|
quickKeyChanges.quickKeys.clear();
|
|
|
|
|
|
|
|
mwmp::QuickKey quickKey;
|
|
|
|
quickKey.slot = slot;
|
|
|
|
quickKey.type = type;
|
|
|
|
quickKey.itemId = itemId;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_QUICKKEYS", itemId.c_str());
|
|
|
|
LOG_APPEND(Log::LOG_INFO, "- slot: %i, type: %i, itemId: %s", quickKey.slot, quickKey.type, quickKey.itemId.c_str());
|
|
|
|
|
|
|
|
quickKeyChanges.quickKeys.push_back(quickKey);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_QUICKKEYS)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_QUICKKEYS)->Send();
|
|
|
|
}
|
|
|
|
|
2017-01-24 17:32:25 +00:00
|
|
|
void LocalPlayer::sendJournalEntry(const std::string& quest, int index, const MWWorld::Ptr& actor)
|
2017-01-20 10:05:45 +00:00
|
|
|
{
|
2017-01-24 17:32:25 +00:00
|
|
|
journalChanges.journalItems.clear();
|
|
|
|
|
|
|
|
mwmp::JournalItem journalItem;
|
|
|
|
journalItem.type = JournalItem::ENTRY;
|
|
|
|
journalItem.quest = quest;
|
|
|
|
journalItem.index = index;
|
2017-05-18 07:32:23 +00:00
|
|
|
journalItem.actorRefId = actor.getCellRef().getRefId();
|
2017-01-24 17:32:25 +00:00
|
|
|
|
|
|
|
journalChanges.journalItems.push_back(journalItem);
|
|
|
|
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_JOURNAL)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_JOURNAL)->Send();
|
2017-01-20 10:05:45 +00:00
|
|
|
}
|
|
|
|
|
2017-01-24 17:32:25 +00:00
|
|
|
void LocalPlayer::sendJournalIndex(const std::string& quest, int index)
|
2017-01-20 10:05:45 +00:00
|
|
|
{
|
2017-01-24 17:32:25 +00:00
|
|
|
journalChanges.journalItems.clear();
|
|
|
|
|
|
|
|
mwmp::JournalItem journalItem;
|
|
|
|
journalItem.type = JournalItem::INDEX;
|
|
|
|
journalItem.quest = quest;
|
|
|
|
journalItem.index = index;
|
|
|
|
|
|
|
|
journalChanges.journalItems.push_back(journalItem);
|
|
|
|
|
2017-03-06 09:44:08 +00:00
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_JOURNAL)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_JOURNAL)->Send();
|
2017-01-20 10:05:45 +00:00
|
|
|
}
|
|
|
|
|
2017-07-05 04:05:52 +00:00
|
|
|
void LocalPlayer::sendFactionRank(const std::string& factionId, int rank)
|
2017-05-18 16:27:20 +00:00
|
|
|
{
|
|
|
|
factionChanges.factions.clear();
|
2017-07-05 04:05:52 +00:00
|
|
|
factionChanges.action = FactionChanges::RANK;
|
2017-05-18 16:27:20 +00:00
|
|
|
|
|
|
|
mwmp::Faction faction;
|
|
|
|
faction.factionId = factionId;
|
|
|
|
faction.rank = rank;
|
2017-07-05 04:05:52 +00:00
|
|
|
|
|
|
|
factionChanges.factions.push_back(faction);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_FACTION)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_FACTION)->Send();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::sendFactionExpulsionState(const std::string& factionId, bool isExpelled)
|
|
|
|
{
|
|
|
|
factionChanges.factions.clear();
|
|
|
|
factionChanges.action = FactionChanges::EXPULSION;
|
|
|
|
|
|
|
|
mwmp::Faction faction;
|
|
|
|
faction.factionId = factionId;
|
2017-05-18 16:27:20 +00:00
|
|
|
faction.isExpelled = isExpelled;
|
|
|
|
|
|
|
|
factionChanges.factions.push_back(faction);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_FACTION)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_FACTION)->Send();
|
|
|
|
}
|
|
|
|
|
2017-07-13 10:36:00 +00:00
|
|
|
void LocalPlayer::sendFactionReputation(const std::string& factionId, int reputation)
|
|
|
|
{
|
|
|
|
factionChanges.factions.clear();
|
|
|
|
factionChanges.action = FactionChanges::REPUTATION;
|
|
|
|
|
|
|
|
mwmp::Faction faction;
|
|
|
|
faction.factionId = factionId;
|
|
|
|
faction.reputation = reputation;
|
|
|
|
|
|
|
|
factionChanges.factions.push_back(faction);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_FACTION)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_FACTION)->Send();
|
|
|
|
}
|
|
|
|
|
2017-05-24 18:43:34 +00:00
|
|
|
void LocalPlayer::sendTopic(const std::string& topicId)
|
|
|
|
{
|
|
|
|
topicChanges.topics.clear();
|
|
|
|
|
|
|
|
mwmp::Topic topic;
|
2017-05-31 11:11:16 +00:00
|
|
|
|
|
|
|
// For translated versions of the game, make sure we translate the topic back into English first
|
|
|
|
if (MWBase::Environment::get().getWindowManager()->getTranslationDataStorage().hasTranslation())
|
|
|
|
topic.topicId = MWBase::Environment::get().getWindowManager()->getTranslationDataStorage().topicID(topicId);
|
|
|
|
else
|
|
|
|
topic.topicId = topicId;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_TOPIC with topic %s", topic.topicId.c_str());
|
2017-05-24 18:43:34 +00:00
|
|
|
|
|
|
|
topicChanges.topics.push_back(topic);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_TOPIC)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_TOPIC)->Send();
|
|
|
|
}
|
|
|
|
|
2017-06-10 11:04:19 +00:00
|
|
|
void LocalPlayer::sendKill(const std::string& refId, int number)
|
|
|
|
{
|
|
|
|
killChanges.kills.clear();
|
|
|
|
|
|
|
|
mwmp::Kill kill;
|
|
|
|
kill.refId = refId;
|
|
|
|
kill.number = number;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_KILL_COUNT with refId %s, number %i", refId.c_str(), number);
|
|
|
|
|
|
|
|
killChanges.kills.push_back(kill);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_KILL_COUNT)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_KILL_COUNT)->Send();
|
|
|
|
}
|
|
|
|
|
2017-06-27 05:27:14 +00:00
|
|
|
void LocalPlayer::sendBook(const std::string& bookId)
|
|
|
|
{
|
|
|
|
bookChanges.books.clear();
|
|
|
|
|
|
|
|
mwmp::Book book;
|
|
|
|
book.bookId = bookId;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_BOOK with book %s", book.bookId.c_str());
|
|
|
|
|
|
|
|
bookChanges.books.push_back(book);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_BOOK)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_BOOK)->Send();
|
|
|
|
}
|
|
|
|
|
2018-02-01 00:11:45 +00:00
|
|
|
void LocalPlayer::sendScale(float newScale)
|
|
|
|
{
|
|
|
|
scale = newScale;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_SHAPESHIFT with scale of %f", scale);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SHAPESHIFT)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SHAPESHIFT)->Send();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LocalPlayer::sendWerewolfState(bool werewolfState)
|
2017-07-15 06:02:19 +00:00
|
|
|
{
|
|
|
|
isWerewolf = werewolfState;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_SHAPESHIFT with isWerewolf of %s", isWerewolf ? "true" : "false");
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SHAPESHIFT)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_SHAPESHIFT)->Send();
|
|
|
|
}
|
|
|
|
|
2018-01-31 02:50:29 +00:00
|
|
|
void LocalPlayer::sendMarkLocation(const ESM::Cell& newMarkCell, const ESM::Position& newMarkPosition)
|
|
|
|
{
|
|
|
|
miscellaneousChangeType = mwmp::MISCELLANEOUS_CHANGE_TYPE::MARK_LOCATION;
|
|
|
|
markCell = newMarkCell;
|
|
|
|
markPosition = newMarkPosition;
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_MISCELLANEOUS)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_MISCELLANEOUS)->Send();
|
|
|
|
}
|
|
|
|
|
2018-02-06 04:36:46 +00:00
|
|
|
void LocalPlayer::sendSelectedSpell(const std::string& newSelectedSpellId)
|
|
|
|
{
|
|
|
|
miscellaneousChangeType = mwmp::MISCELLANEOUS_CHANGE_TYPE::SELECTED_SPELL;
|
|
|
|
selectedSpellId = newSelectedSpellId;
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_MISCELLANEOUS)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_MISCELLANEOUS)->Send();
|
|
|
|
}
|
|
|
|
|
2018-05-08 02:57:04 +00:00
|
|
|
void LocalPlayer::sendMapExplored(int x, int y, const std::vector<char>& imageData)
|
|
|
|
{
|
|
|
|
mapChanges.mapTiles.clear();
|
|
|
|
|
|
|
|
mwmp::MapTile mapTile;
|
|
|
|
mapTile.x = x;
|
|
|
|
mapTile.y = y;
|
|
|
|
mapTile.imageData = imageData;
|
|
|
|
|
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_INFO, "Sending ID_PLAYER_MAP with x: %i, y: %i", x, y);
|
|
|
|
|
|
|
|
mapChanges.mapTiles.push_back(mapTile);
|
|
|
|
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_MAP)->setPlayer(this);
|
|
|
|
getNetworking()->getPlayerPacket(ID_PLAYER_MAP)->Send();
|
|
|
|
}
|
|
|
|
|
2017-02-03 18:27:40 +00:00
|
|
|
void LocalPlayer::clearCellStates()
|
|
|
|
{
|
|
|
|
cellStateChanges.cellStates.clear();
|
|
|
|
}
|
|
|
|
|
2017-02-05 16:45:23 +00:00
|
|
|
void LocalPlayer::clearCurrentContainer()
|
|
|
|
{
|
|
|
|
currentContainer.refId = "";
|
|
|
|
currentContainer.refNumIndex = 0;
|
2017-04-04 08:07:16 +00:00
|
|
|
currentContainer.mpNum = 0;
|
2017-02-05 16:45:23 +00:00
|
|
|
}
|
|
|
|
|
2018-01-28 19:19:03 +00:00
|
|
|
void LocalPlayer::storeCellState(const ESM::Cell& cell, int stateType)
|
2017-02-03 18:27:40 +00:00
|
|
|
{
|
2017-02-04 07:46:27 +00:00
|
|
|
std::vector<CellState>::iterator iter;
|
|
|
|
|
|
|
|
for (iter = cellStateChanges.cellStates.begin(); iter != cellStateChanges.cellStates.end(); )
|
|
|
|
{
|
|
|
|
// If there's already a cell state recorded for this particular cell,
|
|
|
|
// remove it
|
|
|
|
if (cell.getDescription() == (*iter).cell.getDescription())
|
|
|
|
iter = cellStateChanges.cellStates.erase(iter);
|
|
|
|
else
|
|
|
|
++iter;
|
|
|
|
}
|
|
|
|
|
2017-02-03 18:27:40 +00:00
|
|
|
CellState cellState;
|
|
|
|
cellState.cell = cell;
|
|
|
|
cellState.type = stateType;
|
|
|
|
|
|
|
|
cellStateChanges.cellStates.push_back(cellState);
|
|
|
|
}
|
|
|
|
|
2017-10-07 20:33:36 +00:00
|
|
|
void LocalPlayer::storeCurrentContainer(const MWWorld::Ptr &container)
|
2017-02-05 16:45:23 +00:00
|
|
|
{
|
|
|
|
currentContainer.refId = container.getCellRef().getRefId();
|
|
|
|
currentContainer.refNumIndex = container.getCellRef().getRefNum().mIndex;
|
2017-04-04 08:07:16 +00:00
|
|
|
currentContainer.mpNum = container.getCellRef().getMpNum();
|
2017-02-05 16:45:23 +00:00
|
|
|
}
|
2017-10-27 06:10:29 +00:00
|
|
|
|
|
|
|
void LocalPlayer::playAnimation()
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getMechanicsManager()->playAnimationGroup(getPlayerPtr(),
|
|
|
|
animation.groupname, animation.mode, animation.count, animation.persist);
|
2017-11-01 20:00:54 +00:00
|
|
|
|
|
|
|
isPlayingAnimation = true;
|
2017-10-27 06:10:29 +00:00
|
|
|
}
|
2017-10-31 13:19:14 +00:00
|
|
|
|
|
|
|
void LocalPlayer::playSpeech()
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getSoundManager()->say(getPlayerPtr(), sound);
|
|
|
|
|
|
|
|
MWBase::WindowManager *winMgr = MWBase::Environment::get().getWindowManager();
|
|
|
|
if (winMgr->getSubtitlesEnabled())
|
|
|
|
winMgr->messageBox(MWBase::Environment::get().getDialogueManager()->getVoiceCaption(sound), MWGui::ShowInDialogueMode_Never);
|
|
|
|
}
|