2010-08-03 12:14:04 +00:00
|
|
|
#include "armor.hpp"
|
|
|
|
|
2018-05-27 13:05:40 +00:00
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Include additional headers for multiplayer purposes
|
|
|
|
*/
|
2018-06-28 01:53:00 +00:00
|
|
|
#include <components/openmw-mp/Utils.hpp>
|
2018-05-27 13:05:40 +00:00
|
|
|
#include "../mwmp/Main.hpp"
|
|
|
|
#include "../mwmp/Networking.hpp"
|
[General] Implement RecordDynamic packet, part 1
Spell, potion, enchantment, creature, NPC, armor, book, clothing, miscellaneous and weapon record data can now be sent in a RecordDynamic packet. Additionally, the packets include data related to associated magical effects (for spells, potions and enchantments), data related to default inventory contents (for creatures and NPCs) and data related to body parts affected (for armor and clothing).
The server now has associated script functions for setting most of the details of the above, with the main exception being individual creature and NPC stats.
Records can either be created entirely from scratch or can use an existing record (set via the baseId variable) as a starting point for their values. In the latter case, only the values that are specifically set override the starting values. Creature and NPC records also have an inventoryBaseId that can be used on top of the baseId to base their inventories on another existing record.
The client's RecordHelper class has been heavily expanded to allow for the above mentioned functionality.
When players create spells, potions and enchantments as part of regular gameplay, they send RecordDynamic packets that provide the server with the complete details of the records that should be created. When they create enchantments, they also provide the server with armor, book, clothing and weapon records corresponding to the items they've enchanted.
This functionality added by this packet was originally supposed to be exclusive to the rewrite, but I've gone ahead and tried to provide it for the pre-rewrite in a way that can mostly be reused for the rewrite.
2018-07-30 07:56:26 +00:00
|
|
|
#include "../mwmp/Worldstate.hpp"
|
2018-05-27 13:05:40 +00:00
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
|
|
|
|
2010-08-03 12:14:04 +00:00
|
|
|
#include <components/esm/loadarmo.hpp>
|
2012-03-13 14:35:06 +00:00
|
|
|
#include <components/esm/loadskil.hpp>
|
|
|
|
#include <components/esm/loadgmst.hpp>
|
2010-08-03 12:14:04 +00:00
|
|
|
|
2012-07-03 10:30:50 +00:00
|
|
|
#include "../mwbase/environment.hpp"
|
|
|
|
#include "../mwbase/world.hpp"
|
2012-08-12 16:11:09 +00:00
|
|
|
#include "../mwbase/windowmanager.hpp"
|
2012-07-03 10:30:50 +00:00
|
|
|
|
2010-08-03 12:14:04 +00:00
|
|
|
#include "../mwworld/ptr.hpp"
|
2012-05-16 14:08:55 +00:00
|
|
|
#include "../mwworld/actionequip.hpp"
|
2012-03-13 14:35:06 +00:00
|
|
|
#include "../mwworld/inventorystore.hpp"
|
2012-06-29 14:48:50 +00:00
|
|
|
#include "../mwworld/cellstore.hpp"
|
2014-12-19 10:26:54 +00:00
|
|
|
#include "../mwworld/esmstore.hpp"
|
2015-05-09 23:09:00 +00:00
|
|
|
#include "../mwphysics/physicssystem.hpp"
|
2013-02-17 14:56:22 +00:00
|
|
|
#include "../mwworld/nullaction.hpp"
|
2013-04-05 13:42:05 +00:00
|
|
|
#include "../mwworld/containerstore.hpp"
|
2010-08-03 12:14:04 +00:00
|
|
|
|
2012-01-27 14:11:02 +00:00
|
|
|
#include "../mwrender/objects.hpp"
|
2012-07-03 11:15:20 +00:00
|
|
|
#include "../mwrender/renderinginterface.hpp"
|
2015-08-21 09:12:39 +00:00
|
|
|
#include "../mwmechanics/actorutil.hpp"
|
2018-12-26 09:45:28 +00:00
|
|
|
#include "../mwmechanics/weapontype.hpp"
|
2010-08-14 08:02:54 +00:00
|
|
|
|
2012-07-03 11:15:20 +00:00
|
|
|
#include "../mwgui/tooltips.hpp"
|
2012-04-16 20:58:16 +00:00
|
|
|
|
2010-08-03 12:14:04 +00:00
|
|
|
namespace MWClass
|
|
|
|
{
|
2014-07-18 07:56:58 +00:00
|
|
|
|
2015-01-12 10:29:56 +00:00
|
|
|
void Armor::insertObjectRendering (const MWWorld::Ptr& ptr, const std::string& model, MWRender::RenderingInterface& renderingInterface) const
|
2010-08-14 08:02:54 +00:00
|
|
|
{
|
2012-07-24 16:22:11 +00:00
|
|
|
if (!model.empty()) {
|
2013-08-07 10:51:57 +00:00
|
|
|
renderingInterface.getObjects().insertModel(ptr, model);
|
2010-08-14 08:02:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-09 23:09:00 +00:00
|
|
|
void Armor::insertObject(const MWWorld::Ptr& ptr, const std::string& model, MWPhysics::PhysicsSystem& physics) const
|
2012-07-24 16:22:11 +00:00
|
|
|
{
|
2015-05-12 01:02:15 +00:00
|
|
|
// TODO: add option somewhere to enable collision for placeable objects
|
2018-05-27 13:05:40 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Make it possible to enable collision for this object class from a packet
|
|
|
|
*/
|
|
|
|
if (!model.empty())
|
|
|
|
{
|
|
|
|
mwmp::BaseWorldstate *worldstate = mwmp::Main::get().getNetworking()->getWorldstate();
|
|
|
|
|
2018-07-03 10:34:41 +00:00
|
|
|
if (worldstate->hasPlacedObjectCollision || Utils::vectorContains(worldstate->enforcedCollisionRefIds, ptr.getCellRef().getRefId()))
|
2018-05-27 13:05:40 +00:00
|
|
|
{
|
|
|
|
if (worldstate->useActorCollisionForPlacedObjects)
|
|
|
|
physics.addObject(ptr, model, MWPhysics::CollisionType_Actor);
|
|
|
|
else
|
|
|
|
physics.addObject(ptr, model, MWPhysics::CollisionType_World);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
2012-07-24 16:22:11 +00:00
|
|
|
}
|
2012-08-09 12:33:21 +00:00
|
|
|
|
2015-12-18 14:51:05 +00:00
|
|
|
std::string Armor::getModel(const MWWorld::ConstPtr &ptr) const
|
2011-11-12 04:01:12 +00:00
|
|
|
{
|
2015-12-18 14:51:05 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2011-11-12 04:01:12 +00:00
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
const std::string &model = ref->mBase->mModel;
|
2012-07-24 16:22:11 +00:00
|
|
|
if (!model.empty()) {
|
|
|
|
return "meshes\\" + model;
|
2011-11-12 04:01:12 +00:00
|
|
|
}
|
2012-07-24 16:22:11 +00:00
|
|
|
return "";
|
2011-11-12 04:01:12 +00:00
|
|
|
}
|
|
|
|
|
2015-12-18 14:27:06 +00:00
|
|
|
std::string Armor::getName (const MWWorld::ConstPtr& ptr) const
|
2010-08-03 15:11:41 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2019-09-10 21:06:50 +00:00
|
|
|
const std::string& name = ref->mBase->mName;
|
2010-08-03 15:11:41 +00:00
|
|
|
|
2019-09-10 21:06:50 +00:00
|
|
|
return !name.empty() ? name : ref->mBase->mId;
|
2010-08-03 15:11:41 +00:00
|
|
|
}
|
|
|
|
|
2017-05-05 19:21:11 +00:00
|
|
|
std::shared_ptr<MWWorld::Action> Armor::activate (const MWWorld::Ptr& ptr,
|
2012-04-23 13:27:03 +00:00
|
|
|
const MWWorld::Ptr& actor) const
|
2010-08-07 18:25:17 +00:00
|
|
|
{
|
2013-08-09 05:34:53 +00:00
|
|
|
return defaultItemActivate(ptr, actor);
|
2010-08-07 18:25:17 +00:00
|
|
|
}
|
|
|
|
|
2015-12-18 15:06:31 +00:00
|
|
|
bool Armor::hasItemHealth (const MWWorld::ConstPtr& ptr) const
|
2010-08-03 12:14:04 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-12-18 14:27:06 +00:00
|
|
|
int Armor::getItemMaxHealth (const MWWorld::ConstPtr& ptr) const
|
2010-08-03 12:14:04 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2010-08-03 12:14:04 +00:00
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
return ref->mBase->mData.mHealth;
|
2010-08-03 12:14:04 +00:00
|
|
|
}
|
|
|
|
|
2015-12-17 23:12:03 +00:00
|
|
|
std::string Armor::getScript (const MWWorld::ConstPtr& ptr) const
|
2010-08-05 13:40:03 +00:00
|
|
|
{
|
2015-12-17 23:12:03 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2010-08-05 13:40:03 +00:00
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
return ref->mBase->mScript;
|
2010-08-05 13:40:03 +00:00
|
|
|
}
|
|
|
|
|
2015-12-18 15:06:31 +00:00
|
|
|
std::pair<std::vector<int>, bool> Armor::getEquipmentSlots (const MWWorld::ConstPtr& ptr) const
|
2012-03-13 14:35:06 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-03-13 14:35:06 +00:00
|
|
|
|
2013-11-12 22:23:19 +00:00
|
|
|
std::vector<int> slots_;
|
2012-03-13 14:35:06 +00:00
|
|
|
|
|
|
|
const int size = 11;
|
|
|
|
|
|
|
|
static const int sMapping[size][2] =
|
|
|
|
{
|
|
|
|
{ ESM::Armor::Helmet, MWWorld::InventoryStore::Slot_Helmet },
|
|
|
|
{ ESM::Armor::Cuirass, MWWorld::InventoryStore::Slot_Cuirass },
|
|
|
|
{ ESM::Armor::LPauldron, MWWorld::InventoryStore::Slot_LeftPauldron },
|
|
|
|
{ ESM::Armor::RPauldron, MWWorld::InventoryStore::Slot_RightPauldron },
|
|
|
|
{ ESM::Armor::Greaves, MWWorld::InventoryStore::Slot_Greaves },
|
|
|
|
{ ESM::Armor::Boots, MWWorld::InventoryStore::Slot_Boots },
|
|
|
|
{ ESM::Armor::LGauntlet, MWWorld::InventoryStore::Slot_LeftGauntlet },
|
|
|
|
{ ESM::Armor::RGauntlet, MWWorld::InventoryStore::Slot_RightGauntlet },
|
|
|
|
{ ESM::Armor::Shield, MWWorld::InventoryStore::Slot_CarriedLeft },
|
|
|
|
{ ESM::Armor::LBracer, MWWorld::InventoryStore::Slot_LeftGauntlet },
|
|
|
|
{ ESM::Armor::RBracer, MWWorld::InventoryStore::Slot_RightGauntlet }
|
|
|
|
};
|
|
|
|
|
|
|
|
for (int i=0; i<size; ++i)
|
2012-11-05 12:07:59 +00:00
|
|
|
if (sMapping[i][0]==ref->mBase->mData.mType)
|
2012-03-13 14:35:06 +00:00
|
|
|
{
|
2013-11-12 22:23:19 +00:00
|
|
|
slots_.push_back (int (sMapping[i][1]));
|
2012-03-13 14:35:06 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-12 22:23:19 +00:00
|
|
|
return std::make_pair (slots_, false);
|
2012-03-13 14:35:06 +00:00
|
|
|
}
|
|
|
|
|
2015-12-18 14:27:06 +00:00
|
|
|
int Armor::getEquipmentSkill (const MWWorld::ConstPtr& ptr) const
|
2012-03-13 14:35:06 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-03-13 14:35:06 +00:00
|
|
|
|
|
|
|
std::string typeGmst;
|
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
switch (ref->mBase->mData.mType)
|
2012-03-13 14:35:06 +00:00
|
|
|
{
|
|
|
|
case ESM::Armor::Helmet: typeGmst = "iHelmWeight"; break;
|
|
|
|
case ESM::Armor::Cuirass: typeGmst = "iCuirassWeight"; break;
|
|
|
|
case ESM::Armor::LPauldron:
|
|
|
|
case ESM::Armor::RPauldron: typeGmst = "iPauldronWeight"; break;
|
|
|
|
case ESM::Armor::Greaves: typeGmst = "iGreavesWeight"; break;
|
|
|
|
case ESM::Armor::Boots: typeGmst = "iBootsWeight"; break;
|
|
|
|
case ESM::Armor::LGauntlet:
|
|
|
|
case ESM::Armor::RGauntlet: typeGmst = "iGauntletWeight"; break;
|
2012-04-28 19:23:24 +00:00
|
|
|
case ESM::Armor::Shield: typeGmst = "iShieldWeight"; break;
|
2012-03-13 14:35:06 +00:00
|
|
|
case ESM::Armor::LBracer:
|
|
|
|
case ESM::Armor::RBracer: typeGmst = "iGauntletWeight"; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeGmst.empty())
|
|
|
|
return -1;
|
|
|
|
|
2012-11-05 14:39:42 +00:00
|
|
|
const MWWorld::Store<ESM::GameSetting> &gmst =
|
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::GameSetting>();
|
2012-03-13 14:35:06 +00:00
|
|
|
|
2018-08-29 15:38:12 +00:00
|
|
|
float iWeight = floor(gmst.find(typeGmst)->mValue.getFloat());
|
2012-11-05 14:39:42 +00:00
|
|
|
|
2015-03-08 00:07:29 +00:00
|
|
|
float epsilon = 0.0005f;
|
2014-07-30 18:26:47 +00:00
|
|
|
|
2018-08-29 15:38:12 +00:00
|
|
|
if (ref->mBase->mData.mWeight <= iWeight * gmst.find ("fLightMaxMod")->mValue.getFloat() + epsilon)
|
2012-03-13 14:35:06 +00:00
|
|
|
return ESM::Skill::LightArmor;
|
|
|
|
|
2018-08-29 15:38:12 +00:00
|
|
|
if (ref->mBase->mData.mWeight <= iWeight * gmst.find ("fMedMaxMod")->mValue.getFloat() + epsilon)
|
2012-03-13 14:35:06 +00:00
|
|
|
return ESM::Skill::MediumArmor;
|
|
|
|
|
2014-07-30 18:26:47 +00:00
|
|
|
else
|
|
|
|
return ESM::Skill::HeavyArmor;
|
2012-03-13 14:35:06 +00:00
|
|
|
}
|
|
|
|
|
2015-12-18 14:27:06 +00:00
|
|
|
int Armor::getValue (const MWWorld::ConstPtr& ptr) const
|
2012-04-07 17:53:49 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-04-07 17:53:49 +00:00
|
|
|
|
2014-06-26 12:59:33 +00:00
|
|
|
return ref->mBase->mData.mValue;
|
2012-04-07 17:53:49 +00:00
|
|
|
}
|
|
|
|
|
2010-08-03 12:14:04 +00:00
|
|
|
void Armor::registerSelf()
|
|
|
|
{
|
2017-05-05 19:21:11 +00:00
|
|
|
std::shared_ptr<Class> instance (new Armor);
|
2010-08-03 12:14:04 +00:00
|
|
|
|
|
|
|
registerClass (typeid (ESM::Armor).name(), instance);
|
|
|
|
}
|
2012-03-13 16:05:38 +00:00
|
|
|
|
2015-12-18 15:09:08 +00:00
|
|
|
std::string Armor::getUpSoundId (const MWWorld::ConstPtr& ptr) const
|
2012-03-13 16:05:38 +00:00
|
|
|
{
|
2012-04-23 13:27:03 +00:00
|
|
|
int es = getEquipmentSkill(ptr);
|
2012-03-13 22:38:44 +00:00
|
|
|
if (es == ESM::Skill::LightArmor)
|
2012-03-13 16:05:38 +00:00
|
|
|
return std::string("Item Armor Light Up");
|
2012-03-13 22:38:44 +00:00
|
|
|
else if (es == ESM::Skill::MediumArmor)
|
2012-03-13 16:05:38 +00:00
|
|
|
return std::string("Item Armor Medium Up");
|
|
|
|
else
|
|
|
|
return std::string("Item Armor Heavy Up");
|
|
|
|
}
|
|
|
|
|
2015-12-18 15:09:08 +00:00
|
|
|
std::string Armor::getDownSoundId (const MWWorld::ConstPtr& ptr) const
|
2012-03-13 16:05:38 +00:00
|
|
|
{
|
2012-04-23 13:27:03 +00:00
|
|
|
int es = getEquipmentSkill(ptr);
|
2012-03-13 22:38:44 +00:00
|
|
|
if (es == ESM::Skill::LightArmor)
|
2012-03-13 16:05:38 +00:00
|
|
|
return std::string("Item Armor Light Down");
|
2012-03-13 22:38:44 +00:00
|
|
|
else if (es == ESM::Skill::MediumArmor)
|
2012-03-13 16:05:38 +00:00
|
|
|
return std::string("Item Armor Medium Down");
|
|
|
|
else
|
|
|
|
return std::string("Item Armor Heavy Down");
|
|
|
|
}
|
2012-04-15 15:52:39 +00:00
|
|
|
|
2015-12-18 14:53:47 +00:00
|
|
|
std::string Armor::getInventoryIcon (const MWWorld::ConstPtr& ptr) const
|
2012-04-15 15:52:39 +00:00
|
|
|
{
|
2015-12-18 14:53:47 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-04-15 15:52:39 +00:00
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
return ref->mBase->mIcon;
|
2012-04-15 15:52:39 +00:00
|
|
|
}
|
2012-05-11 08:40:40 +00:00
|
|
|
|
2015-12-19 15:29:07 +00:00
|
|
|
MWGui::ToolTipInfo Armor::getToolTipInfo (const MWWorld::ConstPtr& ptr, int count) const
|
2012-04-16 20:58:16 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-04-16 20:58:16 +00:00
|
|
|
|
|
|
|
MWGui::ToolTipInfo info;
|
2019-09-10 18:56:10 +00:00
|
|
|
info.caption = MyGUI::TextIterator::toTagsString(getName(ptr)) + MWGui::ToolTips::getCountString(count);
|
2012-11-05 12:07:59 +00:00
|
|
|
info.icon = ref->mBase->mIcon;
|
2012-04-16 20:58:16 +00:00
|
|
|
|
|
|
|
std::string text;
|
|
|
|
|
|
|
|
// get armor type string (light/medium/heavy)
|
|
|
|
std::string typeText;
|
2016-10-04 13:56:58 +00:00
|
|
|
if (ref->mBase->mData.mWeight == 0)
|
2016-08-14 09:36:46 +00:00
|
|
|
typeText = "";
|
2016-10-04 13:56:58 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
int armorType = getEquipmentSkill(ptr);
|
|
|
|
if (armorType == ESM::Skill::LightArmor)
|
|
|
|
typeText = "#{sLight}";
|
|
|
|
else if (armorType == ESM::Skill::MediumArmor)
|
|
|
|
typeText = "#{sMedium}";
|
|
|
|
else
|
|
|
|
typeText = "#{sHeavy}";
|
|
|
|
}
|
2012-04-16 20:58:16 +00:00
|
|
|
|
2017-02-17 02:11:37 +00:00
|
|
|
text += "\n#{sArmorRating}: " + MWGui::ToolTips::toString(static_cast<int>(getEffectiveArmorRating(ptr,
|
|
|
|
MWMechanics::getPlayer())));
|
2012-04-16 20:58:16 +00:00
|
|
|
|
2014-05-25 12:13:07 +00:00
|
|
|
int remainingHealth = getItemHealth(ptr);
|
2013-03-22 11:24:09 +00:00
|
|
|
text += "\n#{sCondition}: " + MWGui::ToolTips::toString(remainingHealth) + "/"
|
|
|
|
+ MWGui::ToolTips::toString(ref->mBase->mData.mHealth);
|
2012-04-16 20:58:16 +00:00
|
|
|
|
2016-08-14 09:36:46 +00:00
|
|
|
if (typeText != "")
|
|
|
|
text += "\n#{sWeight}: " + MWGui::ToolTips::toString(ref->mBase->mData.mWeight) + " (" + typeText + ")";
|
|
|
|
|
2014-06-12 19:49:18 +00:00
|
|
|
text += MWGui::ToolTips::getValueString(ref->mBase->mData.mValue, "#{sValue}");
|
2012-04-16 20:58:16 +00:00
|
|
|
|
2012-04-24 00:02:03 +00:00
|
|
|
if (MWBase::Environment::get().getWindowManager()->getFullHelp()) {
|
2014-07-22 18:03:35 +00:00
|
|
|
text += MWGui::ToolTips::getCellRefString(ptr.getCellRef());
|
2012-11-05 12:07:59 +00:00
|
|
|
text += MWGui::ToolTips::getMiscString(ref->mBase->mScript, "Script");
|
2012-04-16 20:58:16 +00:00
|
|
|
}
|
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
info.enchant = ref->mBase->mEnchant;
|
2013-04-03 19:14:49 +00:00
|
|
|
if (!info.enchant.empty())
|
2015-03-08 00:07:29 +00:00
|
|
|
info.remainingEnchantCharge = static_cast<int>(ptr.getCellRef().getEnchantmentCharge());
|
2012-04-29 22:57:41 +00:00
|
|
|
|
2012-04-16 20:58:16 +00:00
|
|
|
info.text = text;
|
|
|
|
|
|
|
|
return info;
|
|
|
|
}
|
2012-05-12 14:17:03 +00:00
|
|
|
|
2015-12-18 14:56:45 +00:00
|
|
|
std::string Armor::getEnchantment (const MWWorld::ConstPtr& ptr) const
|
2012-05-12 14:17:03 +00:00
|
|
|
{
|
2015-12-18 14:56:45 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-05-12 14:17:03 +00:00
|
|
|
|
2012-11-05 12:07:59 +00:00
|
|
|
return ref->mBase->mEnchant;
|
2012-05-12 14:17:03 +00:00
|
|
|
}
|
2012-05-16 14:08:55 +00:00
|
|
|
|
2015-12-18 15:43:11 +00:00
|
|
|
std::string Armor::applyEnchantment(const MWWorld::ConstPtr &ptr, const std::string& enchId, int enchCharge, const std::string& newName) const
|
2013-03-28 16:41:00 +00:00
|
|
|
{
|
2015-12-18 15:43:11 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2013-03-28 22:39:20 +00:00
|
|
|
|
|
|
|
ESM::Armor newItem = *ref->mBase;
|
2013-03-28 16:41:00 +00:00
|
|
|
newItem.mId="";
|
|
|
|
newItem.mName=newName;
|
|
|
|
newItem.mData.mEnchant=enchCharge;
|
|
|
|
newItem.mEnchant=enchId;
|
[General] Implement RecordDynamic packet, part 1
Spell, potion, enchantment, creature, NPC, armor, book, clothing, miscellaneous and weapon record data can now be sent in a RecordDynamic packet. Additionally, the packets include data related to associated magical effects (for spells, potions and enchantments), data related to default inventory contents (for creatures and NPCs) and data related to body parts affected (for armor and clothing).
The server now has associated script functions for setting most of the details of the above, with the main exception being individual creature and NPC stats.
Records can either be created entirely from scratch or can use an existing record (set via the baseId variable) as a starting point for their values. In the latter case, only the values that are specifically set override the starting values. Creature and NPC records also have an inventoryBaseId that can be used on top of the baseId to base their inventories on another existing record.
The client's RecordHelper class has been heavily expanded to allow for the above mentioned functionality.
When players create spells, potions and enchantments as part of regular gameplay, they send RecordDynamic packets that provide the server with the complete details of the records that should be created. When they create enchantments, they also provide the server with armor, book, clothing and weapon records corresponding to the items they've enchanted.
This functionality added by this packet was originally supposed to be exclusive to the rewrite, but I've gone ahead and tried to provide it for the pre-rewrite in a way that can mostly be reused for the rewrite.
2018-07-30 07:56:26 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
Start of tes3mp addition
|
|
|
|
|
|
|
|
Send the newly created record to the server and expect it to be
|
|
|
|
returned with a server-set id
|
|
|
|
*/
|
|
|
|
mwmp::Main::get().getNetworking()->getWorldstate()->sendArmorRecord(&newItem, ref->mBase->mId);
|
|
|
|
/*
|
|
|
|
End of tes3mp addition
|
|
|
|
*/
|
|
|
|
|
2013-03-28 16:41:00 +00:00
|
|
|
const ESM::Armor *record = MWBase::Environment::get().getWorld()->createRecord (newItem);
|
2014-05-24 12:48:37 +00:00
|
|
|
return record->mId;
|
2013-03-28 16:41:00 +00:00
|
|
|
}
|
|
|
|
|
2017-02-17 02:11:37 +00:00
|
|
|
float Armor::getEffectiveArmorRating(const MWWorld::ConstPtr &ptr, const MWWorld::Ptr &actor) const
|
2015-03-01 18:28:20 +00:00
|
|
|
{
|
2015-12-18 14:27:06 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2015-03-01 18:28:20 +00:00
|
|
|
|
|
|
|
int armorSkillType = getEquipmentSkill(ptr);
|
2018-12-23 11:18:33 +00:00
|
|
|
float armorSkill = actor.getClass().getSkill(actor, armorSkillType);
|
2015-03-01 18:28:20 +00:00
|
|
|
|
|
|
|
const MWBase::World *world = MWBase::Environment::get().getWorld();
|
2018-08-29 15:38:12 +00:00
|
|
|
int iBaseArmorSkill = world->getStore().get<ESM::GameSetting>().find("iBaseArmorSkill")->mValue.getInteger();
|
2015-03-01 18:28:20 +00:00
|
|
|
|
|
|
|
if(ref->mBase->mData.mWeight == 0)
|
|
|
|
return ref->mBase->mData.mArmor;
|
|
|
|
else
|
2017-02-17 02:11:37 +00:00
|
|
|
return ref->mBase->mData.mArmor * armorSkill / static_cast<float>(iBaseArmorSkill);
|
2015-03-01 18:28:20 +00:00
|
|
|
}
|
|
|
|
|
2015-12-18 15:06:31 +00:00
|
|
|
std::pair<int, std::string> Armor::canBeEquipped(const MWWorld::ConstPtr &ptr, const MWWorld::Ptr &npc) const
|
2013-04-05 13:42:05 +00:00
|
|
|
{
|
2017-02-26 21:24:51 +00:00
|
|
|
const MWWorld::InventoryStore& invStore = npc.getClass().getInventoryStore(npc);
|
2013-04-05 13:42:05 +00:00
|
|
|
|
2018-10-24 15:51:34 +00:00
|
|
|
if (getItemHealth(ptr) == 0)
|
2014-01-02 00:03:44 +00:00
|
|
|
return std::make_pair(0, "#{sInventoryMessage1}");
|
|
|
|
|
2013-04-05 13:42:05 +00:00
|
|
|
// slots that this item can be equipped in
|
2018-10-28 14:03:38 +00:00
|
|
|
std::pair<std::vector<int>, bool> slots_ = getEquipmentSlots(ptr);
|
2013-04-05 13:42:05 +00:00
|
|
|
|
2013-12-30 18:04:35 +00:00
|
|
|
if (slots_.first.empty())
|
|
|
|
return std::make_pair(0, "");
|
|
|
|
|
2014-01-19 10:42:58 +00:00
|
|
|
if (npc.getClass().isNpc())
|
2013-04-05 13:42:05 +00:00
|
|
|
{
|
2014-01-19 10:42:58 +00:00
|
|
|
std::string npcRace = npc.get<ESM::NPC>()->mBase->mRace;
|
2013-04-05 13:42:05 +00:00
|
|
|
|
2014-01-19 10:42:58 +00:00
|
|
|
// Beast races cannot equip shoes / boots, or full helms (head part vs hair part)
|
|
|
|
const ESM::Race* race = MWBase::Environment::get().getWorld()->getStore().get<ESM::Race>().find(npcRace);
|
|
|
|
if(race->mData.mFlags & ESM::Race::Beast)
|
2013-04-05 13:42:05 +00:00
|
|
|
{
|
2014-01-19 10:42:58 +00:00
|
|
|
std::vector<ESM::PartReference> parts = ptr.get<ESM::Armor>()->mBase->mParts.mParts;
|
|
|
|
|
|
|
|
for(std::vector<ESM::PartReference>::iterator itr = parts.begin(); itr != parts.end(); ++itr)
|
|
|
|
{
|
|
|
|
if((*itr).mPart == ESM::PRT_Head)
|
|
|
|
return std::make_pair(0, "#{sNotifyMessage13}");
|
|
|
|
if((*itr).mPart == ESM::PRT_LFoot || (*itr).mPart == ESM::PRT_RFoot)
|
|
|
|
return std::make_pair(0, "#{sNotifyMessage14}");
|
|
|
|
}
|
2013-04-08 15:50:03 +00:00
|
|
|
}
|
2013-12-30 18:04:35 +00:00
|
|
|
}
|
2013-04-05 13:42:05 +00:00
|
|
|
|
2013-12-30 18:04:35 +00:00
|
|
|
for (std::vector<int>::const_iterator slot=slots_.first.begin();
|
|
|
|
slot!=slots_.first.end(); ++slot)
|
|
|
|
{
|
|
|
|
// If equipping a shield, check if there's a twohanded weapon conflicting with it
|
2013-04-08 15:50:03 +00:00
|
|
|
if(*slot == MWWorld::InventoryStore::Slot_CarriedLeft)
|
|
|
|
{
|
2017-02-26 21:24:51 +00:00
|
|
|
MWWorld::ConstContainerStoreIterator weapon = invStore.getSlot(MWWorld::InventoryStore::Slot_CarriedRight);
|
2018-12-26 09:45:28 +00:00
|
|
|
if(weapon != invStore.end() && weapon->getTypeName() == typeid(ESM::Weapon).name())
|
2013-04-08 15:50:03 +00:00
|
|
|
{
|
2018-12-26 09:45:28 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Weapon> *ref = weapon->get<ESM::Weapon>();
|
|
|
|
if (MWMechanics::getWeaponType(ref->mBase->mData.mType)->mFlags & ESM::WeaponType::TwoHanded)
|
|
|
|
return std::make_pair(3,"");
|
2013-04-08 15:50:03 +00:00
|
|
|
}
|
2018-12-26 09:45:28 +00:00
|
|
|
|
2013-04-15 00:56:23 +00:00
|
|
|
return std::make_pair(1,"");
|
2013-04-05 13:42:05 +00:00
|
|
|
}
|
|
|
|
}
|
2013-04-15 00:56:23 +00:00
|
|
|
return std::make_pair(1,"");
|
2013-04-05 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2018-07-09 15:31:40 +00:00
|
|
|
std::shared_ptr<MWWorld::Action> Armor::use (const MWWorld::Ptr& ptr, bool force) const
|
2012-05-16 14:08:55 +00:00
|
|
|
{
|
2018-07-09 15:31:40 +00:00
|
|
|
std::shared_ptr<MWWorld::Action> action(new MWWorld::ActionEquip(ptr, force));
|
2012-08-19 23:11:50 +00:00
|
|
|
|
|
|
|
action->setSound(getUpSoundId(ptr));
|
2012-05-16 14:08:55 +00:00
|
|
|
|
2012-08-19 23:11:50 +00:00
|
|
|
return action;
|
2012-05-16 14:08:55 +00:00
|
|
|
}
|
2012-07-25 13:18:17 +00:00
|
|
|
|
2015-12-18 15:24:24 +00:00
|
|
|
MWWorld::Ptr Armor::copyToCellImpl(const MWWorld::ConstPtr &ptr, MWWorld::CellStore &cell) const
|
2012-07-25 13:18:17 +00:00
|
|
|
{
|
2015-12-18 15:24:24 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2012-07-25 13:18:17 +00:00
|
|
|
|
2015-11-14 16:12:05 +00:00
|
|
|
return MWWorld::Ptr(cell.insert(ref), &cell);
|
2012-07-25 13:18:17 +00:00
|
|
|
}
|
2013-03-16 18:00:14 +00:00
|
|
|
|
2015-12-18 15:48:19 +00:00
|
|
|
int Armor::getEnchantmentPoints (const MWWorld::ConstPtr& ptr) const
|
2013-03-16 18:00:14 +00:00
|
|
|
{
|
2015-12-18 15:48:19 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2013-03-16 18:00:14 +00:00
|
|
|
|
2014-01-27 00:58:04 +00:00
|
|
|
return ref->mBase->mData.mEnchant;
|
2013-03-16 18:00:14 +00:00
|
|
|
}
|
2013-04-07 19:38:53 +00:00
|
|
|
|
2015-12-18 14:58:23 +00:00
|
|
|
bool Armor::canSell (const MWWorld::ConstPtr& item, int npcServices) const
|
2013-04-07 19:38:53 +00:00
|
|
|
{
|
2014-09-01 00:04:44 +00:00
|
|
|
return (npcServices & ESM::NPC::Armor)
|
|
|
|
|| ((npcServices & ESM::NPC::MagicItems) && !getEnchantment(item).empty());
|
2013-04-07 19:38:53 +00:00
|
|
|
}
|
2013-05-11 16:38:27 +00:00
|
|
|
|
2015-12-18 15:00:50 +00:00
|
|
|
float Armor::getWeight(const MWWorld::ConstPtr &ptr) const
|
2013-05-11 16:38:27 +00:00
|
|
|
{
|
2015-12-18 15:00:50 +00:00
|
|
|
const MWWorld::LiveCellRef<ESM::Armor> *ref = ptr.get<ESM::Armor>();
|
2013-05-11 16:38:27 +00:00
|
|
|
return ref->mBase->mData.mWeight;
|
|
|
|
}
|
2010-08-03 12:14:04 +00:00
|
|
|
}
|