forked from teamnwah/openmw-tes3coop
[Client] Fix problems with NPC equipment sync
This commit is contained in:
parent
34378fbb4f
commit
6ac7ee8fcd
3 changed files with 68 additions and 15 deletions
|
@ -163,39 +163,39 @@ void DedicatedActor::setEquipment()
|
||||||
return;
|
return;
|
||||||
|
|
||||||
MWWorld::InventoryStore& invStore = ptr.getClass().getInventoryStore(ptr);
|
MWWorld::InventoryStore& invStore = ptr.getClass().getInventoryStore(ptr);
|
||||||
|
|
||||||
for (int slot = 0; slot < MWWorld::InventoryStore::Slots; ++slot)
|
for (int slot = 0; slot < MWWorld::InventoryStore::Slots; ++slot)
|
||||||
{
|
{
|
||||||
MWWorld::ContainerStoreIterator it = invStore.getSlot(slot);
|
MWWorld::ContainerStoreIterator it = invStore.getSlot(slot);
|
||||||
|
|
||||||
const string &dedicItem = equipedItems[slot].refId;
|
const string &packetRefId = equipedItems[slot].refId;
|
||||||
std::string item = "";
|
int packetCharge = equipedItems[slot].charge;
|
||||||
|
std::string storeRefId = "";
|
||||||
bool equal = false;
|
bool equal = false;
|
||||||
|
|
||||||
if (it != invStore.end())
|
if (it != invStore.end())
|
||||||
{
|
{
|
||||||
item = it->getCellRef().getRefId();
|
storeRefId = it->getCellRef().getRefId();
|
||||||
if (!Misc::StringUtils::ciEqual(item, dedicItem)) // if other item equiped
|
|
||||||
|
if (!Misc::StringUtils::ciEqual(storeRefId, packetRefId)) // if other item equiped
|
||||||
{
|
{
|
||||||
MWWorld::ContainerStore &store = ptr.getClass().getContainerStore(ptr);
|
invStore.unequipSlot(slot, ptr);
|
||||||
store.remove(item, store.count(item), ptr);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
equal = true;
|
equal = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dedicItem.empty() || equal)
|
if (packetRefId.empty() || equal)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
int count = equipedItems[slot].count;
|
int count = equipedItems[slot].count;
|
||||||
ptr.getClass().getContainerStore(ptr).add(dedicItem, count, ptr);
|
|
||||||
|
|
||||||
for (MWWorld::ContainerStoreIterator it2 = invStore.begin(); it2 != invStore.end(); ++it2)
|
if (hasItem(packetRefId, packetCharge))
|
||||||
|
equipItem(packetRefId, packetCharge);
|
||||||
|
else
|
||||||
{
|
{
|
||||||
if (::Misc::StringUtils::ciEqual(it2->getCellRef().getRefId(), dedicItem)) // equip item
|
ptr.getClass().getContainerStore(ptr).add(packetRefId, count, ptr);
|
||||||
{
|
equipItem(packetRefId, packetCharge);
|
||||||
boost::shared_ptr<MWWorld::Action> action = it2->getClass().use(*it2);
|
|
||||||
action->execute(ptr);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -225,6 +225,36 @@ void DedicatedActor::playSound()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool DedicatedActor::hasItem(std::string refId, int charge)
|
||||||
|
{
|
||||||
|
MWWorld::InventoryStore& invStore = ptr.getClass().getInventoryStore(ptr);
|
||||||
|
|
||||||
|
for (MWWorld::ContainerStoreIterator it = invStore.begin(); it != invStore.end(); ++it)
|
||||||
|
{
|
||||||
|
if (::Misc::StringUtils::ciEqual(it->getCellRef().getRefId(), refId) && it->getCellRef().getCharge() == charge)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DedicatedActor::equipItem(std::string refId, int charge)
|
||||||
|
{
|
||||||
|
MWWorld::InventoryStore& invStore = ptr.getClass().getInventoryStore(ptr);
|
||||||
|
|
||||||
|
for (MWWorld::ContainerStoreIterator it = invStore.begin(); it != invStore.end(); ++it)
|
||||||
|
{
|
||||||
|
if (::Misc::StringUtils::ciEqual(it->getCellRef().getRefId(), refId) && it->getCellRef().getCharge() == charge)
|
||||||
|
{
|
||||||
|
boost::shared_ptr<MWWorld::Action> action = it->getClass().use(*it);
|
||||||
|
action->execute(ptr);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
MWWorld::Ptr DedicatedActor::getPtr()
|
MWWorld::Ptr DedicatedActor::getPtr()
|
||||||
{
|
{
|
||||||
return ptr;
|
return ptr;
|
||||||
|
|
|
@ -25,6 +25,9 @@ namespace mwmp
|
||||||
void playAnimation();
|
void playAnimation();
|
||||||
void playSound();
|
void playSound();
|
||||||
|
|
||||||
|
bool hasItem(std::string refId, int charge);
|
||||||
|
void equipItem(std::string refId, int charge);
|
||||||
|
|
||||||
MWWorld::Ptr getPtr();
|
MWWorld::Ptr getPtr();
|
||||||
void setPtr(const MWWorld::Ptr& newPtr);
|
void setPtr(const MWWorld::Ptr& newPtr);
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
#include "LocalPlayer.hpp"
|
#include "LocalPlayer.hpp"
|
||||||
#include "DedicatedPlayer.hpp"
|
#include "DedicatedPlayer.hpp"
|
||||||
#include "PlayerList.hpp"
|
#include "PlayerList.hpp"
|
||||||
|
#include "CellController.hpp"
|
||||||
|
|
||||||
#include <components/openmw-mp/Log.hpp>
|
#include <components/openmw-mp/Log.hpp>
|
||||||
|
|
||||||
|
@ -23,6 +24,7 @@
|
||||||
#include "../mwworld/class.hpp"
|
#include "../mwworld/class.hpp"
|
||||||
#include "../mwworld/containerstore.hpp"
|
#include "../mwworld/containerstore.hpp"
|
||||||
#include "../mwworld/esmstore.hpp"
|
#include "../mwworld/esmstore.hpp"
|
||||||
|
#include "../mwworld/inventorystore.hpp"
|
||||||
#include "../mwworld/manualref.hpp"
|
#include "../mwworld/manualref.hpp"
|
||||||
|
|
||||||
using namespace mwmp;
|
using namespace mwmp;
|
||||||
|
@ -108,6 +110,15 @@ void WorldEvent::editContainers(MWWorld::CellStore* cellStore)
|
||||||
if (iter->getCellRef().getCharge() == containerItem.charge &&
|
if (iter->getCellRef().getCharge() == containerItem.charge &&
|
||||||
iter->getRefData().getCount() == containerItem.count)
|
iter->getRefData().getCount() == containerItem.count)
|
||||||
{
|
{
|
||||||
|
// Is this an actor's container? If so, unequip this item if it was equipped
|
||||||
|
if (ptrFound.getClass().isActor())
|
||||||
|
{
|
||||||
|
MWWorld::InventoryStore& invStore = ptrFound.getClass().getInventoryStore(ptrFound);
|
||||||
|
|
||||||
|
if (invStore.isEquipped(*iter))
|
||||||
|
invStore.unequipItemQuantity(*iter, ptrFound, containerItem.count);
|
||||||
|
}
|
||||||
|
|
||||||
containerStore.remove(*iter, containerItem.actionCount, ownerPtr);
|
containerStore.remove(*iter, containerItem.actionCount, ownerPtr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -115,6 +126,15 @@ void WorldEvent::editContainers(MWWorld::CellStore* cellStore)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Was this a SET or ADD action on an actor's container, and are we the authority
|
||||||
|
// over the actor? If so, autoequip the actor
|
||||||
|
if ((action == BaseEvent::ADD || action == BaseEvent::SET) && ptrFound.getClass().isActor() &&
|
||||||
|
mwmp::Main::get().getCellController()->isLocalActor(ptrFound))
|
||||||
|
{
|
||||||
|
MWWorld::InventoryStore& invStore = ptrFound.getClass().getInventoryStore(ptrFound);
|
||||||
|
invStore.autoEquip(ptrFound);
|
||||||
|
}
|
||||||
|
|
||||||
// If we are in a container, and it happens to be this container, update its view
|
// If we are in a container, and it happens to be this container, update its view
|
||||||
if (MWBase::Environment::get().getWindowManager()->containsMode(MWGui::GM_Container))
|
if (MWBase::Environment::get().getWindowManager()->containsMode(MWGui::GM_Container))
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in a new issue