2015-01-17 00:11:36 +01:00
|
|
|
#include "importcellref.hpp"
|
|
|
|
|
2022-01-22 15:58:41 +01:00
|
|
|
#include <components/esm3/esmreader.hpp>
|
2024-03-23 12:15:09 +01:00
|
|
|
#include <components/misc/concepts.hpp>
|
|
|
|
|
2023-12-17 15:21:12 +01:00
|
|
|
#include <cstdint>
|
2015-01-17 00:11:36 +01:00
|
|
|
|
|
|
|
namespace ESSImport
|
|
|
|
{
|
2024-03-23 12:15:09 +01:00
|
|
|
template <Misc::SameAsWithoutCvref<ACDT> T>
|
|
|
|
void decompose(T&& v, const auto& f)
|
|
|
|
{
|
|
|
|
f(v.mUnknown, v.mFlags, v.mBreathMeter, v.mUnknown2, v.mDynamic, v.mUnknown3, v.mAttributes, v.mMagicEffects,
|
|
|
|
v.mUnknown4, v.mGoldPool, v.mCountDown, v.mUnknown5);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <Misc::SameAsWithoutCvref<ACSC> T>
|
|
|
|
void decompose(T&& v, const auto& f)
|
|
|
|
{
|
|
|
|
f(v.mUnknown1, v.mFlags, v.mUnknown2, v.mCorpseClearCountdown, v.mUnknown3);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <Misc::SameAsWithoutCvref<ANIS> T>
|
|
|
|
void decompose(T&& v, const auto& f)
|
|
|
|
{
|
|
|
|
f(v.mGroupIndex, v.mUnknown, v.mTime);
|
|
|
|
}
|
2015-01-17 00:11:36 +01:00
|
|
|
|
|
|
|
void CellRef::load(ESM::ESMReader& esm)
|
|
|
|
{
|
2015-01-21 22:25:37 +01:00
|
|
|
blank();
|
|
|
|
|
2022-07-30 19:13:41 +03:00
|
|
|
esm.getHNT(mRefNum.mIndex, "FRMR");
|
2015-01-17 00:11:36 +01:00
|
|
|
|
2015-01-18 16:13:52 +01:00
|
|
|
// this is required since openmw supports more than 255 content files
|
|
|
|
int pluginIndex = (mRefNum.mIndex & 0xff000000) >> 24;
|
|
|
|
mRefNum.mContentFile = pluginIndex - 1;
|
|
|
|
mRefNum.mIndex &= 0x00ffffff;
|
|
|
|
|
2015-01-17 00:11:36 +01:00
|
|
|
mIndexedRefId = esm.getHNString("NAME");
|
|
|
|
|
2022-07-30 19:13:41 +03:00
|
|
|
if (esm.isNextSub("ACTN"))
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Activation flags:
|
|
|
|
ActivationFlag_UseEnabled = 1
|
|
|
|
ActivationFlag_OnActivate = 2
|
|
|
|
ActivationFlag_OnDeath = 10h
|
|
|
|
ActivationFlag_OnKnockout = 20h
|
|
|
|
ActivationFlag_OnMurder = 40h
|
|
|
|
ActivationFlag_DoorOpening = 100h
|
|
|
|
ActivationFlag_DoorClosing = 200h
|
|
|
|
ActivationFlag_DoorJammedOpening = 400h
|
|
|
|
ActivationFlag_DoorJammedClosing = 800h
|
|
|
|
*/
|
|
|
|
esm.skipHSub();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (esm.isNextSub("STPR"))
|
|
|
|
esm.skipHSub();
|
|
|
|
|
|
|
|
if (esm.isNextSub("MNAM"))
|
|
|
|
esm.skipHSub();
|
|
|
|
|
|
|
|
bool isDeleted = false;
|
|
|
|
ESM::CellRef::loadData(esm, isDeleted);
|
|
|
|
|
2024-03-23 12:15:09 +01:00
|
|
|
mActorData.mHasACDT = esm.getOptionalComposite("ACDT", mActorData.mACDT);
|
2022-07-30 19:13:41 +03:00
|
|
|
|
2024-03-23 12:15:09 +01:00
|
|
|
mActorData.mHasACSC = esm.getOptionalComposite("ACSC", mActorData.mACSC);
|
2022-07-30 19:13:41 +03:00
|
|
|
|
|
|
|
if (esm.isNextSub("ACSL"))
|
|
|
|
esm.skipHSubSize(112);
|
|
|
|
|
|
|
|
if (esm.isNextSub("CSTN"))
|
|
|
|
esm.skipHSub(); // "PlayerSaveGame", link to some object?
|
|
|
|
|
|
|
|
if (esm.isNextSub("LSTN"))
|
|
|
|
esm.skipHSub(); // "PlayerSaveGame", link to some object?
|
|
|
|
|
|
|
|
// unsure at which point between LSTN and TGTN
|
|
|
|
if (esm.isNextSub("CSHN"))
|
|
|
|
esm.skipHSub(); // "PlayerSaveGame", link to some object?
|
|
|
|
|
|
|
|
// unsure if before or after CSTN/LSTN
|
|
|
|
if (esm.isNextSub("LSHN"))
|
|
|
|
esm.skipHSub(); // "PlayerSaveGame", link to some object?
|
|
|
|
|
|
|
|
while (esm.isNextSub("TGTN"))
|
|
|
|
esm.skipHSub(); // "PlayerSaveGame", link to some object?
|
|
|
|
|
|
|
|
while (esm.isNextSub("FGTN"))
|
|
|
|
esm.getHString(); // fight target?
|
|
|
|
|
|
|
|
// unsure at which point between TGTN and CRED
|
|
|
|
if (esm.isNextSub("AADT"))
|
|
|
|
{
|
|
|
|
// occurred when a creature was in the middle of its attack, 44 bytes
|
|
|
|
esm.skipHSub();
|
|
|
|
}
|
|
|
|
|
|
|
|
// unsure at which point between FGTN and CHRD
|
|
|
|
if (esm.isNextSub("PWPC"))
|
|
|
|
esm.skipHSub();
|
|
|
|
if (esm.isNextSub("PWPS"))
|
|
|
|
esm.skipHSub();
|
|
|
|
|
|
|
|
if (esm.isNextSub("WNAM"))
|
|
|
|
{
|
|
|
|
std::string id = esm.getHString();
|
|
|
|
|
|
|
|
if (esm.isNextSub("XNAM"))
|
|
|
|
mActorData.mSelectedEnchantItem = esm.getHString();
|
|
|
|
else
|
2023-07-29 11:44:39 +04:00
|
|
|
mActorData.mSelectedSpell = std::move(id);
|
2022-07-30 19:13:41 +03:00
|
|
|
|
|
|
|
if (esm.isNextSub("YNAM"))
|
|
|
|
esm.skipHSub(); // 4 byte, 0
|
|
|
|
}
|
|
|
|
|
|
|
|
while (esm.isNextSub("APUD"))
|
|
|
|
{
|
|
|
|
// used power
|
|
|
|
esm.getSubHeader();
|
2023-02-12 17:03:01 +01:00
|
|
|
std::string id = esm.getMaybeFixedStringSize(32);
|
2022-07-30 19:13:41 +03:00
|
|
|
(void)id;
|
|
|
|
// timestamp can't be used: this is the total hours passed, calculated by
|
|
|
|
// timestamp = 24 * (365 * year + cumulativeDays[month] + day)
|
|
|
|
// unfortunately cumulativeDays[month] is not clearly defined,
|
|
|
|
// in the (non-MCP) vanilla version the first month was missing, but MCP added it.
|
|
|
|
double timestamp;
|
|
|
|
esm.getT(timestamp);
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: not all actors have this, add flag
|
2023-12-20 12:28:34 +01:00
|
|
|
esm.getHNOT("CHRD", mActorData.mSkills); // npc only
|
2022-07-30 19:13:41 +03:00
|
|
|
|
2023-12-20 12:28:34 +01:00
|
|
|
esm.getHNOT("CRED", mActorData.mCombatStats); // creature only
|
2022-07-30 19:13:41 +03:00
|
|
|
|
|
|
|
mActorData.mSCRI.load(esm);
|
|
|
|
|
|
|
|
if (esm.isNextSub("ND3D"))
|
|
|
|
esm.skipHSub();
|
|
|
|
|
2024-03-23 12:15:09 +01:00
|
|
|
mActorData.mHasANIS = esm.getOptionalComposite("ANIS", mActorData.mANIS);
|
2022-07-30 19:13:41 +03:00
|
|
|
|
2015-01-18 22:52:11 +01:00
|
|
|
if (esm.isNextSub("LVCR"))
|
2015-01-19 23:51:43 +01:00
|
|
|
{
|
2015-02-01 00:26:06 +01:00
|
|
|
// occurs on levelled creature spawner references
|
2015-01-22 04:12:08 +01:00
|
|
|
// probably some identifier for the creature that has been spawned?
|
2015-01-19 23:51:43 +01:00
|
|
|
unsigned char lvcr;
|
|
|
|
esm.getHT(lvcr);
|
|
|
|
// std::cout << "LVCR: " << (int)lvcr << std::endl;
|
|
|
|
}
|
2015-01-18 16:13:52 +01:00
|
|
|
|
|
|
|
mEnabled = true;
|
|
|
|
esm.getHNOT(mEnabled, "ZNAM");
|
|
|
|
|
2015-02-01 00:26:06 +01:00
|
|
|
// DATA should occur for all references, except levelled creature spawners
|
2015-01-21 22:25:37 +01:00
|
|
|
// I've seen DATA *twice* on a creature record, and with the exact same content too! weird
|
2015-01-18 22:52:11 +01:00
|
|
|
// alarmvoi0000.ess
|
2023-12-17 14:03:45 +01:00
|
|
|
for (int i = 0; i < 2; ++i)
|
2024-03-23 12:15:09 +01:00
|
|
|
esm.getOptionalComposite("DATA", mPos);
|
2015-01-21 22:25:37 +01:00
|
|
|
|
|
|
|
mDeleted = 0;
|
|
|
|
if (esm.isNextSub("DELE"))
|
|
|
|
{
|
2023-12-17 14:03:45 +01:00
|
|
|
uint32_t deleted;
|
2015-01-21 22:25:37 +01:00
|
|
|
esm.getHT(deleted);
|
2015-03-06 23:19:57 +13:00
|
|
|
mDeleted = ((deleted >> 24) & 0x2) != 0; // the other 3 bytes seem to be uninitialized garbage
|
2015-01-21 22:25:37 +01:00
|
|
|
}
|
2015-01-18 22:52:11 +01:00
|
|
|
|
|
|
|
if (esm.isNextSub("MVRF"))
|
|
|
|
{
|
|
|
|
esm.skipHSub();
|
|
|
|
esm.getSubName();
|
|
|
|
esm.skipHSub();
|
|
|
|
}
|
2015-01-17 00:11:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|