2011-04-08 13:58:21 +00:00
|
|
|
#include "loadnpc.hpp"
|
|
|
|
|
2012-09-23 18:41:41 +00:00
|
|
|
#include "esmreader.hpp"
|
|
|
|
#include "esmwriter.hpp"
|
2013-09-24 11:17:28 +00:00
|
|
|
#include "defs.hpp"
|
2012-09-17 07:37:50 +00:00
|
|
|
|
2011-04-08 13:58:21 +00:00
|
|
|
namespace ESM
|
|
|
|
{
|
2013-09-24 11:17:28 +00:00
|
|
|
unsigned int NPC::sRecordId = REC_NPC_;
|
2011-04-08 13:58:21 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
void NPC::load(ESMReader &esm, bool &isDeleted)
|
2012-04-06 19:04:30 +00:00
|
|
|
{
|
2015-07-20 14:23:14 +00:00
|
|
|
isDeleted = false;
|
2021-06-29 13:25:26 +00:00
|
|
|
mRecordFlags = esm.getRecordFlags();
|
2015-07-20 14:23:14 +00:00
|
|
|
|
2015-02-10 22:16:25 +00:00
|
|
|
mSpells.mList.clear();
|
|
|
|
mInventory.mList.clear();
|
2015-03-08 19:44:41 +00:00
|
|
|
mTransport.mList.clear();
|
2015-02-10 22:16:25 +00:00
|
|
|
mAiPackage.mList.clear();
|
2019-01-06 18:03:19 +00:00
|
|
|
mAiData.blank();
|
2019-02-08 13:32:34 +00:00
|
|
|
mAiData.mHello = mAiData.mFight = mAiData.mFlee = 30;
|
2011-04-08 13:58:21 +00:00
|
|
|
|
2015-07-16 16:52:31 +00:00
|
|
|
bool hasName = false;
|
2015-02-10 22:16:25 +00:00
|
|
|
bool hasNpdt = false;
|
|
|
|
bool hasFlags = false;
|
|
|
|
while (esm.hasMoreSubs())
|
|
|
|
{
|
|
|
|
esm.getSubName();
|
2016-05-07 17:32:51 +00:00
|
|
|
switch (esm.retSubName().intval)
|
2015-02-10 22:16:25 +00:00
|
|
|
{
|
2015-11-13 17:07:18 +00:00
|
|
|
case ESM::SREC_NAME:
|
2015-07-16 16:52:31 +00:00
|
|
|
mId = esm.getHString();
|
|
|
|
hasName = true;
|
|
|
|
break;
|
2015-02-10 22:16:25 +00:00
|
|
|
case ESM::FourCC<'M','O','D','L'>::value:
|
|
|
|
mModel = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'F','N','A','M'>::value:
|
|
|
|
mName = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'R','N','A','M'>::value:
|
|
|
|
mRace = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'C','N','A','M'>::value:
|
|
|
|
mClass = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'A','N','A','M'>::value:
|
|
|
|
mFaction = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'B','N','A','M'>::value:
|
|
|
|
mHead = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'K','N','A','M'>::value:
|
|
|
|
mHair = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'S','C','R','I'>::value:
|
|
|
|
mScript = esm.getHString();
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'N','P','D','T'>::value:
|
|
|
|
hasNpdt = true;
|
|
|
|
esm.getSubHeader();
|
|
|
|
if (esm.getSubSize() == 52)
|
|
|
|
{
|
|
|
|
mNpdtType = NPC_DEFAULT;
|
2018-05-08 22:25:07 +00:00
|
|
|
esm.getExact(&mNpdt, 52);
|
2015-02-10 22:16:25 +00:00
|
|
|
}
|
|
|
|
else if (esm.getSubSize() == 12)
|
|
|
|
{
|
2018-05-09 16:02:02 +00:00
|
|
|
//Reading into temporary NPDTstruct12 object
|
|
|
|
NPDTstruct12 npdt12;
|
2015-02-10 22:16:25 +00:00
|
|
|
mNpdtType = NPC_WITH_AUTOCALCULATED_STATS;
|
2018-05-08 22:25:07 +00:00
|
|
|
esm.getExact(&npdt12, 12);
|
|
|
|
|
|
|
|
//Clearing the mNdpt struct to initialize all values
|
|
|
|
blankNpdt();
|
|
|
|
//Swiching to an internal representation
|
|
|
|
mNpdt.mLevel = npdt12.mLevel;
|
|
|
|
mNpdt.mDisposition = npdt12.mDisposition;
|
|
|
|
mNpdt.mReputation = npdt12.mReputation;
|
|
|
|
mNpdt.mRank = npdt12.mRank;
|
|
|
|
mNpdt.mGold = npdt12.mGold;
|
2015-02-10 22:16:25 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
esm.fail("NPC_NPDT must be 12 or 52 bytes long");
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'F','L','A','G'>::value:
|
|
|
|
hasFlags = true;
|
2018-12-31 14:55:46 +00:00
|
|
|
int flags;
|
|
|
|
esm.getHT(flags);
|
|
|
|
mFlags = flags & 0xFF;
|
|
|
|
mBloodType = ((flags >> 8) & 0xFF) >> 2;
|
2015-02-10 22:16:25 +00:00
|
|
|
break;
|
|
|
|
case ESM::FourCC<'N','P','C','S'>::value:
|
|
|
|
mSpells.add(esm);
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'N','P','C','O'>::value:
|
|
|
|
mInventory.add(esm);
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'A','I','D','T'>::value:
|
|
|
|
esm.getHExact(&mAiData, sizeof(mAiData));
|
|
|
|
break;
|
|
|
|
case ESM::FourCC<'D','O','D','T'>::value:
|
|
|
|
case ESM::FourCC<'D','N','A','M'>::value:
|
2015-03-08 19:44:41 +00:00
|
|
|
mTransport.add(esm);
|
2015-02-10 22:16:25 +00:00
|
|
|
break;
|
|
|
|
case AI_Wander:
|
|
|
|
case AI_Activate:
|
|
|
|
case AI_Escort:
|
|
|
|
case AI_Follow:
|
|
|
|
case AI_Travel:
|
|
|
|
case AI_CNDT:
|
|
|
|
mAiPackage.add(esm);
|
|
|
|
break;
|
2015-11-13 17:07:18 +00:00
|
|
|
case ESM::SREC_DELE:
|
2015-07-20 14:23:14 +00:00
|
|
|
esm.skipHSub();
|
|
|
|
isDeleted = true;
|
|
|
|
break;
|
2015-02-10 22:16:25 +00:00
|
|
|
default:
|
2015-02-12 03:56:05 +00:00
|
|
|
esm.fail("Unknown subrecord");
|
2015-07-16 16:52:31 +00:00
|
|
|
break;
|
2015-02-10 22:16:25 +00:00
|
|
|
}
|
2013-05-06 12:10:43 +00:00
|
|
|
}
|
2015-07-16 16:52:31 +00:00
|
|
|
|
|
|
|
if (!hasName)
|
|
|
|
esm.fail("Missing NAME subrecord");
|
2015-07-20 14:23:14 +00:00
|
|
|
if (!hasNpdt && !isDeleted)
|
2015-02-10 22:16:25 +00:00
|
|
|
esm.fail("Missing NPDT subrecord");
|
2015-07-20 14:23:14 +00:00
|
|
|
if (!hasFlags && !isDeleted)
|
2015-02-10 22:16:25 +00:00
|
|
|
esm.fail("Missing FLAG subrecord");
|
2012-08-29 10:01:10 +00:00
|
|
|
}
|
2015-07-20 14:23:14 +00:00
|
|
|
void NPC::save(ESMWriter &esm, bool isDeleted) const
|
2015-02-10 22:16:25 +00:00
|
|
|
{
|
2015-07-07 12:37:42 +00:00
|
|
|
esm.writeHNCString("NAME", mId);
|
2015-07-16 16:52:31 +00:00
|
|
|
|
2015-07-20 14:23:14 +00:00
|
|
|
if (isDeleted)
|
2015-07-07 12:37:42 +00:00
|
|
|
{
|
2021-07-06 04:57:58 +00:00
|
|
|
esm.writeHNString("DELE", "", 3);
|
2015-07-07 12:37:42 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-02-10 22:16:25 +00:00
|
|
|
esm.writeHNOCString("MODL", mModel);
|
|
|
|
esm.writeHNOCString("FNAM", mName);
|
|
|
|
esm.writeHNCString("RNAM", mRace);
|
|
|
|
esm.writeHNCString("CNAM", mClass);
|
|
|
|
esm.writeHNCString("ANAM", mFaction);
|
|
|
|
esm.writeHNCString("BNAM", mHead);
|
|
|
|
esm.writeHNCString("KNAM", mHair);
|
|
|
|
esm.writeHNOCString("SCRI", mScript);
|
|
|
|
|
|
|
|
if (mNpdtType == NPC_DEFAULT)
|
2018-05-08 22:25:07 +00:00
|
|
|
{
|
2018-05-09 16:02:02 +00:00
|
|
|
esm.writeHNT("NPDT", mNpdt, 52);
|
2018-05-08 22:25:07 +00:00
|
|
|
}
|
2015-02-10 22:16:25 +00:00
|
|
|
else if (mNpdtType == NPC_WITH_AUTOCALCULATED_STATS)
|
2018-05-08 22:25:07 +00:00
|
|
|
{
|
2018-05-09 16:02:02 +00:00
|
|
|
NPDTstruct12 npdt12;
|
|
|
|
npdt12.mLevel = mNpdt.mLevel;
|
|
|
|
npdt12.mDisposition = mNpdt.mDisposition;
|
|
|
|
npdt12.mReputation = mNpdt.mReputation;
|
|
|
|
npdt12.mRank = mNpdt.mRank;
|
2018-05-08 22:25:07 +00:00
|
|
|
npdt12.mGold = mNpdt.mGold;
|
2018-05-09 16:02:02 +00:00
|
|
|
esm.writeHNT("NPDT", npdt12, 12);
|
2018-05-08 22:25:07 +00:00
|
|
|
}
|
2015-02-10 22:16:25 +00:00
|
|
|
|
2018-12-31 14:55:46 +00:00
|
|
|
esm.writeHNT("FLAG", ((mBloodType << 10) + mFlags));
|
2015-02-10 22:16:25 +00:00
|
|
|
|
|
|
|
mInventory.save(esm);
|
|
|
|
mSpells.save(esm);
|
2019-02-08 13:32:34 +00:00
|
|
|
esm.writeHNT("AIDT", mAiData, sizeof(mAiData));
|
2012-09-18 15:30:19 +00:00
|
|
|
|
2015-03-08 19:44:41 +00:00
|
|
|
mTransport.save(esm);
|
|
|
|
|
2015-02-10 22:16:25 +00:00
|
|
|
mAiPackage.save(esm);
|
2012-09-18 15:30:19 +00:00
|
|
|
}
|
2011-04-08 13:58:21 +00:00
|
|
|
|
2013-05-06 12:10:43 +00:00
|
|
|
bool NPC::isMale() const {
|
|
|
|
return (mFlags & Female) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void NPC::setIsMale(bool value) {
|
|
|
|
mFlags |= Female;
|
|
|
|
if (value) {
|
|
|
|
mFlags ^= Female;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void NPC::blank()
|
|
|
|
{
|
2015-05-13 03:39:04 +00:00
|
|
|
mNpdtType = NPC_DEFAULT;
|
2018-05-08 22:25:07 +00:00
|
|
|
blankNpdt();
|
2018-12-31 14:55:46 +00:00
|
|
|
mBloodType = 0;
|
2013-05-06 12:10:43 +00:00
|
|
|
mFlags = 0;
|
|
|
|
mInventory.mList.clear();
|
|
|
|
mSpells.mList.clear();
|
|
|
|
mAiData.blank();
|
2019-02-08 13:32:34 +00:00
|
|
|
mAiData.mHello = mAiData.mFight = mAiData.mFlee = 30;
|
2015-03-08 19:44:41 +00:00
|
|
|
mTransport.mList.clear();
|
2013-05-06 12:10:43 +00:00
|
|
|
mAiPackage.mList.clear();
|
|
|
|
mName.clear();
|
|
|
|
mModel.clear();
|
|
|
|
mRace.clear();
|
|
|
|
mClass.clear();
|
|
|
|
mFaction.clear();
|
|
|
|
mScript.clear();
|
|
|
|
mHair.clear();
|
|
|
|
mHead.clear();
|
|
|
|
}
|
2015-01-27 16:32:21 +00:00
|
|
|
|
2018-05-08 22:25:07 +00:00
|
|
|
void NPC::blankNpdt()
|
|
|
|
{
|
|
|
|
mNpdt.mLevel = 0;
|
|
|
|
mNpdt.mStrength = mNpdt.mIntelligence = mNpdt.mWillpower = mNpdt.mAgility =
|
|
|
|
mNpdt.mSpeed = mNpdt.mEndurance = mNpdt.mPersonality = mNpdt.mLuck = 0;
|
|
|
|
for (int i=0; i< Skill::Length; ++i) mNpdt.mSkills[i] = 0;
|
|
|
|
mNpdt.mReputation = 0;
|
|
|
|
mNpdt.mHealth = mNpdt.mMana = mNpdt.mFatigue = 0;
|
|
|
|
mNpdt.mDisposition = 0;
|
2019-12-18 17:42:54 +00:00
|
|
|
mNpdt.mUnknown1 = 0;
|
2018-05-08 22:25:07 +00:00
|
|
|
mNpdt.mRank = 0;
|
2019-12-18 17:42:54 +00:00
|
|
|
mNpdt.mUnknown2 = 0;
|
2018-05-08 22:25:07 +00:00
|
|
|
mNpdt.mGold = 0;
|
|
|
|
}
|
|
|
|
|
2015-01-27 16:32:21 +00:00
|
|
|
int NPC::getFactionRank() const
|
|
|
|
{
|
|
|
|
if (mFaction.empty())
|
|
|
|
return -1;
|
2018-05-08 22:25:07 +00:00
|
|
|
else
|
|
|
|
return mNpdt.mRank;
|
2015-01-27 16:32:21 +00:00
|
|
|
}
|
2015-03-08 19:44:41 +00:00
|
|
|
|
|
|
|
const std::vector<Transport::Dest>& NPC::getTransport() const
|
|
|
|
{
|
|
|
|
return mTransport.mList;
|
|
|
|
}
|
2011-04-08 13:58:21 +00:00
|
|
|
}
|