2017-04-17 11:37:19 +00:00
|
|
|
#include <components/openmw-mp/Log.hpp>
|
|
|
|
|
2017-04-19 14:09:35 +00:00
|
|
|
#include <components/misc/rng.hpp>
|
|
|
|
|
2017-04-17 13:09:07 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
|
|
|
|
2017-04-24 18:06:23 +00:00
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
2017-04-17 13:09:07 +00:00
|
|
|
#include "../mwmechanics/combat.hpp"
|
2017-05-06 05:40:36 +00:00
|
|
|
#include "../mwmechanics/levelledlist.hpp"
|
2017-04-19 14:09:35 +00:00
|
|
|
#include "../mwmechanics/spellcasting.hpp"
|
2017-04-17 13:09:07 +00:00
|
|
|
|
|
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwworld/inventorystore.hpp"
|
|
|
|
|
2017-04-17 11:37:19 +00:00
|
|
|
#include "MechanicsHelper.hpp"
|
|
|
|
#include "Main.hpp"
|
2017-05-06 05:40:36 +00:00
|
|
|
#include "Networking.hpp"
|
2017-04-17 13:09:07 +00:00
|
|
|
#include "LocalPlayer.hpp"
|
|
|
|
#include "DedicatedPlayer.hpp"
|
2017-04-30 11:57:43 +00:00
|
|
|
#include "PlayerList.hpp"
|
2018-05-12 21:42:24 +00:00
|
|
|
#include "ObjectList.hpp"
|
2017-04-19 07:36:23 +00:00
|
|
|
#include "CellController.hpp"
|
2017-04-17 13:09:07 +00:00
|
|
|
|
2017-04-17 11:37:19 +00:00
|
|
|
using namespace mwmp;
|
|
|
|
|
|
|
|
osg::Vec3f MechanicsHelper::getLinearInterpolation(osg::Vec3f start, osg::Vec3f end, float percent)
|
|
|
|
{
|
|
|
|
osg::Vec3f position(percent, percent, percent);
|
|
|
|
|
|
|
|
return (start + osg::componentMultiply(position, (end - start)));
|
|
|
|
}
|
2017-04-17 13:09:07 +00:00
|
|
|
|
2017-05-06 05:40:36 +00:00
|
|
|
// Inspired by similar code in mwclass\creaturelevlist.cpp
|
|
|
|
void MechanicsHelper::spawnLeveledCreatures(MWWorld::CellStore* cellStore)
|
|
|
|
{
|
|
|
|
MWWorld::CellRefList<ESM::CreatureLevList> *creatureLevList = cellStore->getCreatureLists();
|
2018-05-12 21:42:24 +00:00
|
|
|
mwmp::ObjectList *objectList = mwmp::Main::get().getNetworking()->getObjectList();
|
|
|
|
objectList->reset();
|
2017-05-06 05:40:36 +00:00
|
|
|
|
2017-06-10 08:04:17 +00:00
|
|
|
int spawnCount = 0;
|
|
|
|
|
2017-06-27 14:27:02 +00:00
|
|
|
for (auto &lref : creatureLevList->mList)
|
2017-05-06 05:40:36 +00:00
|
|
|
{
|
2017-06-27 08:34:32 +00:00
|
|
|
MWWorld::Ptr ptr(&lref, cellStore);
|
2017-05-06 05:40:36 +00:00
|
|
|
|
2017-06-27 08:34:32 +00:00
|
|
|
std::string id = MWMechanics::getLevelledItem(ptr.get<ESM::CreatureLevList>()->mBase, true);
|
2017-05-06 05:40:36 +00:00
|
|
|
|
|
|
|
if (!id.empty())
|
|
|
|
{
|
|
|
|
const MWWorld::ESMStore& store = MWBase::Environment::get().getWorld()->getStore();
|
|
|
|
MWWorld::ManualRef manualRef(store, id);
|
|
|
|
manualRef.getPtr().getCellRef().setPosition(ptr.getCellRef().getPosition());
|
2017-06-27 08:34:32 +00:00
|
|
|
MWWorld::Ptr placed = MWBase::Environment::get().getWorld()->placeObject(manualRef.getPtr(), ptr.getCell(),
|
|
|
|
ptr.getCellRef().getPosition());
|
2018-05-12 21:42:24 +00:00
|
|
|
objectList->addObjectSpawn(placed);
|
2017-05-06 05:40:36 +00:00
|
|
|
MWBase::Environment::get().getWorld()->deleteObject(placed);
|
2017-06-10 08:04:17 +00:00
|
|
|
|
|
|
|
spawnCount++;
|
2017-05-06 05:40:36 +00:00
|
|
|
}
|
|
|
|
}
|
2017-05-06 17:44:14 +00:00
|
|
|
|
2017-06-10 08:04:17 +00:00
|
|
|
if (spawnCount > 0)
|
2018-05-12 21:42:24 +00:00
|
|
|
objectList->sendObjectSpawn();
|
2017-05-06 05:40:36 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 07:36:23 +00:00
|
|
|
Attack *MechanicsHelper::getLocalAttack(const MWWorld::Ptr& ptr)
|
|
|
|
{
|
|
|
|
if (ptr == MWBase::Environment::get().getWorld()->getPlayerPtr())
|
|
|
|
return &mwmp::Main::get().getLocalPlayer()->attack;
|
|
|
|
else if (mwmp::Main::get().getCellController()->isLocalActor(ptr))
|
|
|
|
return &mwmp::Main::get().getCellController()->getLocalActor(ptr)->attack;
|
|
|
|
|
2017-06-27 14:49:28 +00:00
|
|
|
return nullptr;
|
2017-04-19 07:36:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Attack *MechanicsHelper::getDedicatedAttack(const MWWorld::Ptr& ptr)
|
|
|
|
{
|
|
|
|
if (mwmp::PlayerList::isDedicatedPlayer(ptr))
|
|
|
|
return &mwmp::PlayerList::getPlayer(ptr)->attack;
|
|
|
|
else if (mwmp::Main::get().getCellController()->isDedicatedActor(ptr))
|
|
|
|
return &mwmp::Main::get().getCellController()->getDedicatedActor(ptr)->attack;
|
|
|
|
|
2017-06-27 14:49:28 +00:00
|
|
|
return nullptr;
|
2017-04-19 07:36:23 +00:00
|
|
|
}
|
|
|
|
|
2017-05-30 07:11:01 +00:00
|
|
|
MWWorld::Ptr MechanicsHelper::getPlayerPtr(const Target& target)
|
|
|
|
{
|
2018-04-05 09:48:53 +00:00
|
|
|
if (target.guid == mwmp::Main::get().getLocalPlayer()->guid)
|
|
|
|
return MWBase::Environment::get().getWorld()->getPlayerPtr();
|
|
|
|
else if (PlayerList::getPlayer(target.guid) != nullptr)
|
|
|
|
return PlayerList::getPlayer(target.guid)->getPtr();
|
2017-05-30 07:11:01 +00:00
|
|
|
|
2017-06-27 14:49:28 +00:00
|
|
|
return nullptr;
|
2017-05-30 07:11:01 +00:00
|
|
|
}
|
|
|
|
|
2017-04-19 08:05:13 +00:00
|
|
|
void MechanicsHelper::assignAttackTarget(Attack* attack, const MWWorld::Ptr& target)
|
|
|
|
{
|
2017-04-19 17:51:31 +00:00
|
|
|
if (target == MWBase::Environment::get().getWorld()->getPlayerPtr())
|
|
|
|
{
|
2018-04-05 09:48:53 +00:00
|
|
|
attack->target.isPlayer = true;
|
2017-04-19 17:51:31 +00:00
|
|
|
attack->target.guid = mwmp::Main::get().getLocalPlayer()->guid;
|
|
|
|
}
|
|
|
|
else if (mwmp::PlayerList::isDedicatedPlayer(target))
|
2017-04-19 08:05:13 +00:00
|
|
|
{
|
2018-04-05 09:48:53 +00:00
|
|
|
attack->target.isPlayer = true;
|
2017-04-19 08:05:13 +00:00
|
|
|
attack->target.guid = mwmp::PlayerList::getPlayer(target)->guid;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MWWorld::CellRef *targetRef = &target.getCellRef();
|
|
|
|
|
2018-04-05 09:48:53 +00:00
|
|
|
attack->target.isPlayer = false;
|
2017-04-19 08:05:13 +00:00
|
|
|
attack->target.refId = targetRef->getRefId();
|
|
|
|
attack->target.refNumIndex = targetRef->getRefNum().mIndex;
|
|
|
|
attack->target.mpNum = targetRef->getMpNum();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-19 14:09:35 +00:00
|
|
|
void MechanicsHelper::resetAttack(Attack* attack)
|
|
|
|
{
|
|
|
|
attack->success = false;
|
|
|
|
attack->knockdown = false;
|
|
|
|
attack->block = false;
|
2018-04-03 11:12:27 +00:00
|
|
|
attack->applyWeaponEnchantment = false;
|
|
|
|
attack->applyProjectileEnchantment = false;
|
2017-04-19 14:09:35 +00:00
|
|
|
attack->target.guid = RakNet::RakNetGUID();
|
|
|
|
attack->target.refId.clear();
|
2018-04-05 09:48:53 +00:00
|
|
|
attack->target.refNumIndex = 0;
|
|
|
|
attack->target.mpNum = 0;
|
2017-04-19 14:09:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool MechanicsHelper::getSpellSuccess(std::string spellId, const MWWorld::Ptr& caster)
|
|
|
|
{
|
|
|
|
return Misc::Rng::roll0to99() < MWMechanics::getSpellSuccessChance(spellId, caster);
|
|
|
|
}
|
|
|
|
|
2017-04-19 08:05:13 +00:00
|
|
|
void MechanicsHelper::processAttack(Attack attack, const MWWorld::Ptr& attacker)
|
2017-04-17 13:09:07 +00:00
|
|
|
{
|
2017-06-27 08:34:32 +00:00
|
|
|
if (!attack.pressed)
|
2017-04-17 13:09:07 +00:00
|
|
|
{
|
2017-04-19 19:06:04 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_VERBOSE, "Processing attack from %s",
|
|
|
|
attacker.getCellRef().getRefId().c_str());
|
|
|
|
LOG_APPEND(Log::LOG_VERBOSE, "- success: %s", attack.success ? "true" : "false");
|
2017-04-17 13:09:07 +00:00
|
|
|
|
2017-06-27 08:34:32 +00:00
|
|
|
if (attack.success)
|
2017-04-19 19:06:04 +00:00
|
|
|
LOG_APPEND(Log::LOG_VERBOSE, "- damage: %f", attack.damage);
|
2017-04-17 13:09:07 +00:00
|
|
|
}
|
|
|
|
|
2017-04-24 18:06:23 +00:00
|
|
|
MWMechanics::CreatureStats &attackerStats = attacker.getClass().getCreatureStats(attacker);
|
2017-04-17 14:24:11 +00:00
|
|
|
attackerStats.getSpells().setSelectedSpell(attack.spellId);
|
2017-04-17 13:09:07 +00:00
|
|
|
|
|
|
|
MWWorld::Ptr victim;
|
2017-04-19 19:06:04 +00:00
|
|
|
|
2018-04-05 09:48:53 +00:00
|
|
|
if (attack.target.isPlayer)
|
2017-04-19 19:06:04 +00:00
|
|
|
{
|
|
|
|
if (attack.target.guid == mwmp::Main::get().getLocalPlayer()->guid)
|
|
|
|
victim = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
2017-06-27 14:49:28 +00:00
|
|
|
else if (PlayerList::getPlayer(attack.target.guid) != nullptr)
|
2017-04-19 19:06:04 +00:00
|
|
|
victim = PlayerList::getPlayer(attack.target.guid)->getPtr();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-06-27 08:34:32 +00:00
|
|
|
auto controller = mwmp::Main::get().getCellController();
|
|
|
|
if (controller->isLocalActor(attack.target.refNumIndex, attack.target.mpNum))
|
|
|
|
victim = controller->getLocalActor(attack.target.refNumIndex, attack.target.mpNum)->getPtr();
|
|
|
|
else if (controller->isDedicatedActor(attack.target.refNumIndex, attack.target.mpNum))
|
|
|
|
victim = controller->getDedicatedActor(attack.target.refNumIndex, attack.target.mpNum)->getPtr();
|
2017-04-19 19:06:04 +00:00
|
|
|
}
|
|
|
|
|
2017-04-17 13:09:07 +00:00
|
|
|
// Get the weapon used (if hand-to-hand, weapon = inv.end())
|
2017-05-27 01:19:28 +00:00
|
|
|
if (attack.type == attack.MELEE)
|
2017-04-17 13:09:07 +00:00
|
|
|
{
|
2017-04-24 18:06:23 +00:00
|
|
|
MWWorld::Ptr weapon;
|
2018-04-03 11:12:27 +00:00
|
|
|
MWWorld::Ptr projectile;
|
2017-04-17 13:09:07 +00:00
|
|
|
|
2017-04-24 18:06:23 +00:00
|
|
|
if (attacker.getClass().hasInventoryStore(attacker))
|
|
|
|
{
|
2018-04-03 11:12:27 +00:00
|
|
|
MWWorld::InventoryStore &inventoryStore = attacker.getClass().getInventoryStore(attacker);
|
|
|
|
MWWorld::ContainerStoreIterator weaponSlot = inventoryStore.getSlot(
|
2017-04-24 18:06:23 +00:00
|
|
|
MWWorld::InventoryStore::Slot_CarriedRight);
|
2018-04-03 11:12:27 +00:00
|
|
|
MWWorld::ContainerStoreIterator projectileSlot = inventoryStore.getSlot(
|
|
|
|
MWWorld::InventoryStore::Slot_Ammunition);
|
|
|
|
|
|
|
|
// TODO: Fix for when arrows, bolts and throwing weapons have just run out
|
|
|
|
weapon = weaponSlot != inventoryStore.end() ? *weaponSlot : MWWorld::Ptr();
|
|
|
|
projectile = projectileSlot != inventoryStore.end() ? *projectileSlot : MWWorld::Ptr();
|
2017-04-24 18:06:23 +00:00
|
|
|
|
|
|
|
if (!weapon.isEmpty() && weapon.getTypeName() != typeid(ESM::Weapon).name())
|
|
|
|
weapon = MWWorld::Ptr();
|
|
|
|
}
|
2017-04-17 13:09:07 +00:00
|
|
|
|
2017-06-27 14:49:28 +00:00
|
|
|
if (victim.mRef != nullptr)
|
2017-04-17 13:09:07 +00:00
|
|
|
{
|
2017-04-24 18:06:23 +00:00
|
|
|
bool healthdmg = true;
|
|
|
|
|
|
|
|
if (weapon.isEmpty())
|
2017-04-17 13:09:07 +00:00
|
|
|
{
|
2017-04-24 18:06:23 +00:00
|
|
|
if (attacker.getClass().isBipedal(attacker))
|
|
|
|
{
|
|
|
|
MWMechanics::CreatureStats &otherstats = victim.getClass().getCreatureStats(victim);
|
|
|
|
healthdmg = otherstats.isParalyzed() || otherstats.getKnockedDown();
|
|
|
|
}
|
2017-04-17 13:09:07 +00:00
|
|
|
}
|
2017-04-24 18:06:23 +00:00
|
|
|
else
|
2018-04-03 07:27:02 +00:00
|
|
|
{
|
2018-04-03 11:12:27 +00:00
|
|
|
LOG_APPEND(Log::LOG_VERBOSE, "- weapon: %s", weapon.getCellRef().getRefId().c_str());
|
|
|
|
|
2017-04-17 13:09:07 +00:00
|
|
|
MWMechanics::blockMeleeAttack(attacker, victim, weapon, attack.damage, 1);
|
|
|
|
|
2018-04-03 11:12:27 +00:00
|
|
|
if (attack.applyWeaponEnchantment)
|
2018-04-03 07:27:02 +00:00
|
|
|
{
|
|
|
|
MWMechanics::CastSpell cast(attacker, victim, false);
|
|
|
|
cast.mHitPosition = osg::Vec3f();
|
|
|
|
cast.cast(weapon, false);
|
|
|
|
}
|
2018-04-03 11:12:27 +00:00
|
|
|
|
|
|
|
if (attack.applyProjectileEnchantment)
|
|
|
|
{
|
|
|
|
MWMechanics::CastSpell cast(attacker, victim, false);
|
|
|
|
cast.mHitPosition = osg::Vec3f();
|
|
|
|
cast.cast(projectile, false);
|
|
|
|
}
|
2018-04-03 07:27:02 +00:00
|
|
|
}
|
|
|
|
|
2017-04-24 18:06:23 +00:00
|
|
|
victim.getClass().onHit(victim, attack.damage, healthdmg, weapon, attacker, osg::Vec3f(),
|
2017-04-17 13:09:07 +00:00
|
|
|
attack.success);
|
|
|
|
}
|
|
|
|
}
|
2017-07-28 17:49:26 +00:00
|
|
|
else if (attack.type == attack.MAGIC)
|
2017-04-17 13:09:07 +00:00
|
|
|
{
|
2017-07-28 17:49:26 +00:00
|
|
|
if (attack.instant)
|
|
|
|
{
|
|
|
|
MWBase::Environment::get().getWorld()->castSpell(attacker);
|
|
|
|
attack.instant = false;
|
|
|
|
}
|
|
|
|
|
2017-04-17 14:24:11 +00:00
|
|
|
LOG_MESSAGE_SIMPLE(Log::LOG_VERBOSE, "SpellId: %s", attack.spellId.c_str());
|
2017-04-17 13:09:07 +00:00
|
|
|
LOG_APPEND(Log::LOG_VERBOSE, " - success: %d", attack.success);
|
|
|
|
}
|
|
|
|
}
|
2018-01-25 22:45:39 +00:00
|
|
|
|
2018-01-26 01:18:01 +00:00
|
|
|
bool MechanicsHelper::doesEffectListContainEffect(const ESM::EffectList& effectList, short effectId, short attributeId, short skillId)
|
|
|
|
{
|
|
|
|
for (const auto &effect : effectList.mList)
|
|
|
|
{
|
|
|
|
if (effect.mEffectID == effectId)
|
|
|
|
{
|
|
|
|
if (attributeId == -1 || effect.mAttribute == attributeId)
|
|
|
|
{
|
|
|
|
if (skillId == -1 || effect.mSkill == skillId)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-01-27 17:37:16 +00:00
|
|
|
void MechanicsHelper::unequipItemsByEffect(const MWWorld::Ptr& ptr, short enchantmentType, short effectId, short attributeId, short skillId)
|
2018-01-25 22:45:39 +00:00
|
|
|
{
|
|
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
MWWorld::InventoryStore &ptrInventory = ptr.getClass().getInventoryStore(ptr);
|
|
|
|
|
|
|
|
for (int slot = 0; slot < MWWorld::InventoryStore::Slots; slot++)
|
|
|
|
{
|
|
|
|
if (ptrInventory.getSlot(slot) != ptrInventory.end())
|
|
|
|
{
|
|
|
|
MWWorld::ConstContainerStoreIterator itemIterator = ptrInventory.getSlot(slot);
|
|
|
|
std::string enchantmentName = itemIterator->getClass().getEnchantment(*itemIterator);
|
|
|
|
|
|
|
|
if (!enchantmentName.empty())
|
|
|
|
{
|
|
|
|
const ESM::Enchantment* enchantment = world->getStore().get<ESM::Enchantment>().find(enchantmentName);
|
|
|
|
|
2018-01-27 17:37:16 +00:00
|
|
|
if (enchantment->mData.mType == enchantmentType && doesEffectListContainEffect(enchantment->mEffects, effectId, attributeId, skillId))
|
2018-01-26 01:18:01 +00:00
|
|
|
ptrInventory.unequipSlot(slot, ptr);
|
2018-01-25 22:45:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|