1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-10 15:39:02 +00:00
OpenMW/apps/openmw/mwworld/manualref.cpp
elsid 069d4255b9
Make ESM::RefId to be fixed size cheap to copy
Use std::variant. Store refId strings in unordered_set and use pointer to an
item there. Inserts to unordered_set do not invalidate pointers to values so the
pointer is always valid. Elements are not removed. Assume there is finite number
of string refIds.
2023-03-19 17:20:43 +01:00

105 lines
3.5 KiB
C++

#include "manualref.hpp"
#include <components/esm/records.hpp>
#include <components/esm4/loadstat.hpp>
#include "esmstore.hpp"
namespace
{
template <typename T>
void create(const MWWorld::Store<T>& list, const ESM::RefId& name, std::any& refValue, MWWorld::Ptr& ptrValue)
{
const T* base = list.find(name);
ESM::CellRef cellRef;
cellRef.blank();
cellRef.mRefID = name;
MWWorld::LiveCellRef<T> ref(cellRef, base);
refValue = ref;
ptrValue = MWWorld::Ptr(&std::any_cast<MWWorld::LiveCellRef<T>&>(refValue), nullptr);
}
}
MWWorld::ManualRef::ManualRef(const MWWorld::ESMStore& store, const ESM::RefId& name, const int count)
{
switch (store.find(name))
{
case ESM::REC_ACTI:
create(store.get<ESM::Activator>(), name, mRef, mPtr);
break;
case ESM::REC_ALCH:
create(store.get<ESM::Potion>(), name, mRef, mPtr);
break;
case ESM::REC_APPA:
create(store.get<ESM::Apparatus>(), name, mRef, mPtr);
break;
case ESM::REC_ARMO:
create(store.get<ESM::Armor>(), name, mRef, mPtr);
break;
case ESM::REC_BOOK:
create(store.get<ESM::Book>(), name, mRef, mPtr);
break;
case ESM::REC_CLOT:
create(store.get<ESM::Clothing>(), name, mRef, mPtr);
break;
case ESM::REC_CONT:
create(store.get<ESM::Container>(), name, mRef, mPtr);
break;
case ESM::REC_CREA:
create(store.get<ESM::Creature>(), name, mRef, mPtr);
break;
case ESM::REC_DOOR:
create(store.get<ESM::Door>(), name, mRef, mPtr);
break;
case ESM::REC_INGR:
create(store.get<ESM::Ingredient>(), name, mRef, mPtr);
break;
case ESM::REC_LEVC:
create(store.get<ESM::CreatureLevList>(), name, mRef, mPtr);
break;
case ESM::REC_LEVI:
create(store.get<ESM::ItemLevList>(), name, mRef, mPtr);
break;
case ESM::REC_LIGH:
create(store.get<ESM::Light>(), name, mRef, mPtr);
break;
case ESM::REC_LOCK:
create(store.get<ESM::Lockpick>(), name, mRef, mPtr);
break;
case ESM::REC_MISC:
create(store.get<ESM::Miscellaneous>(), name, mRef, mPtr);
break;
case ESM::REC_NPC_:
create(store.get<ESM::NPC>(), name, mRef, mPtr);
break;
case ESM::REC_PROB:
create(store.get<ESM::Probe>(), name, mRef, mPtr);
break;
case ESM::REC_REPA:
create(store.get<ESM::Repair>(), name, mRef, mPtr);
break;
case ESM::REC_STAT:
create(store.get<ESM::Static>(), name, mRef, mPtr);
break;
case ESM::REC_WEAP:
create(store.get<ESM::Weapon>(), name, mRef, mPtr);
break;
case ESM::REC_BODY:
create(store.get<ESM::BodyPart>(), name, mRef, mPtr);
break;
case ESM::REC_STAT4:
create(store.get<ESM4::Static>(), name, mRef, mPtr);
break;
case 0:
throw std::logic_error("failed to create manual cell ref for " + name.toDebugString() + " (unknown ID)");
default:
throw std::logic_error("failed to create manual cell ref for " + name.toDebugString() + " (unknown type)");
}
mPtr.getRefData().setCount(count);
}