#include "refidcollection.hpp" #include #include "refidadapter.hpp" #include "refidadapterimp.hpp" CSMWorld::RefIdColumn::RefIdColumn (const std::string& title, Display displayType, int flag, bool editable, bool userEditable) : ColumnBase (title, displayType, flag), mEditable (editable), mUserEditable (userEditable) {} bool CSMWorld::RefIdColumn::isEditable() const { return mEditable; } bool CSMWorld::RefIdColumn::isUserEditable() const { return mUserEditable; } const CSMWorld::RefIdAdapter& CSMWorld::RefIdCollection::findAdaptor (UniversalId::Type type) const { std::map::const_iterator iter = mAdapters.find (type); if (iter==mAdapters.end()) throw std::logic_error ("unsupported type in RefIdCollection"); return *iter->second; } CSMWorld::RefIdCollection::RefIdCollection() { BaseColumns baseColumns; mColumns.push_back (RefIdColumn ("ID", ColumnBase::Display_String, ColumnBase::Flag_Table | ColumnBase::Flag_Dialogue, false, false)); baseColumns.mId = &mColumns.back(); mColumns.push_back (RefIdColumn ("*", ColumnBase::Display_Integer, ColumnBase::Flag_Table | ColumnBase::Flag_Dialogue, false, false)); baseColumns.mModified = &mColumns.back(); mColumns.push_back (RefIdColumn ("Type", ColumnBase::Display_Integer, ColumnBase::Flag_Table | ColumnBase::Flag_Dialogue, false, false)); baseColumns.mType = &mColumns.back(); ModelColumns modelColumns (baseColumns); mColumns.push_back (RefIdColumn ("Model", ColumnBase::Display_String)); modelColumns.mModel = &mColumns.back(); NameColumns nameColumns (modelColumns); mColumns.push_back (RefIdColumn ("Name", ColumnBase::Display_String)); nameColumns.mName = &mColumns.back(); mColumns.push_back (RefIdColumn ("Script", ColumnBase::Display_String)); nameColumns.mScript = &mColumns.back(); mAdapters.insert (std::make_pair (UniversalId::Type_Activator, new NameRefIdAdapter (UniversalId::Type_Activator, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Potion, new NameRefIdAdapter (UniversalId::Type_Potion, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Apparatus, new NameRefIdAdapter (UniversalId::Type_Apparatus, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Armor, new NameRefIdAdapter (UniversalId::Type_Armor, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Book, new NameRefIdAdapter (UniversalId::Type_Book, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Clothing, new NameRefIdAdapter (UniversalId::Type_Clothing, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Container, new NameRefIdAdapter (UniversalId::Type_Container, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Creature, new NameRefIdAdapter (UniversalId::Type_Creature, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Door, new NameRefIdAdapter (UniversalId::Type_Door, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Ingredient, new NameRefIdAdapter (UniversalId::Type_Ingredient, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_CreatureLevelledList, new BaseRefIdAdapter ( UniversalId::Type_CreatureLevelledList, baseColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_ItemLevelledList, new BaseRefIdAdapter (UniversalId::Type_ItemLevelledList, baseColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Light, new NameRefIdAdapter (UniversalId::Type_Light, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Lockpick, new NameRefIdAdapter (UniversalId::Type_Lockpick, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Miscellaneous, new NameRefIdAdapter (UniversalId::Type_Miscellaneous, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Npc, new NameRefIdAdapter (UniversalId::Type_Npc, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Probe, new NameRefIdAdapter (UniversalId::Type_Probe, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Repair, new NameRefIdAdapter (UniversalId::Type_Repair, nameColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Static, new ModelRefIdAdapter (UniversalId::Type_Static, modelColumns))); mAdapters.insert (std::make_pair (UniversalId::Type_Weapon, new NameRefIdAdapter (UniversalId::Type_Weapon, nameColumns))); } CSMWorld::RefIdCollection::~RefIdCollection() { for (std::map::iterator iter (mAdapters.begin()); iter!=mAdapters.end(); ++iter) delete iter->second; } int CSMWorld::RefIdCollection::getSize() const { return mData.getSize(); } std::string CSMWorld::RefIdCollection::getId (int index) const { return getData (index, 0).toString().toUtf8().constData(); } int CSMWorld::RefIdCollection::getIndex (const std::string& id) const { int index = searchId (id); if (index==-1) throw std::runtime_error ("invalid ID: " + id); return index; } int CSMWorld::RefIdCollection::getColumns() const { return mColumns.size(); } const CSMWorld::ColumnBase& CSMWorld::RefIdCollection::getColumn (int column) const { return mColumns.at (column); } QVariant CSMWorld::RefIdCollection::getData (int index, int column) const { RefIdData::LocalIndex localIndex = mData.globalToLocalIndex (index); const RefIdAdapter& adaptor = findAdaptor (localIndex.second); return adaptor.getData (&mColumns.at (column), mData, localIndex.first); } void CSMWorld::RefIdCollection::setData (int index, int column, const QVariant& data) { RefIdData::LocalIndex localIndex = mData.globalToLocalIndex (index); const RefIdAdapter& adaptor = findAdaptor (localIndex.second); adaptor.setData (&mColumns.at (column), mData, localIndex.first, data); } void CSMWorld::RefIdCollection::removeRows (int index, int count) { mData.erase (index, count); } void CSMWorld::RefIdCollection::appendBlankRecord (const std::string& id, UniversalId::Type type) { mData.appendRecord (type, id); } int CSMWorld::RefIdCollection::searchId (const std::string& id) const { RefIdData::LocalIndex localIndex = mData.searchId (id); if (localIndex.first==-1) return -1; return mData.localToGlobalIndex (localIndex); } void CSMWorld::RefIdCollection::replace (int index, const RecordBase& record) { mData.getRecord (mData.globalToLocalIndex (index)).assign (record); } void CSMWorld::RefIdCollection::appendRecord (const RecordBase& record, UniversalId::Type type) { std::string id = findAdaptor (type).getId (record); int index = mData.getAppendIndex (type); mData.appendRecord (type, id); mData.getRecord (mData.globalToLocalIndex (index)).assign (record); } const CSMWorld::RecordBase& CSMWorld::RefIdCollection::getRecord (const std::string& id) const { return mData.getRecord (mData.searchId (id)); } const CSMWorld::RecordBase& CSMWorld::RefIdCollection::getRecord (int index) const { return mData.getRecord (mData.globalToLocalIndex (index)); } void CSMWorld::RefIdCollection::load (ESM::ESMReader& reader, bool base, UniversalId::Type type) { std::string id = reader.getHNOString ("NAME"); int index = searchId (id); if (reader.isNextSub ("DELE")) { reader.skipRecord(); if (index==-1) { // deleting a record that does not exist // ignore it for now /// \todo report the problem to the user } else if (base) { mData.erase (index, 1); } else { mData.getRecord (mData.globalToLocalIndex (index)).mState = RecordBase::State_Deleted; } } else { if (index==-1) { // new record int index = mData.getAppendIndex (type); mData.appendRecord (type, id); RefIdData::LocalIndex localIndex = mData.globalToLocalIndex (index); mData.load (localIndex, reader, base); mData.getRecord (localIndex).mState = base ? RecordBase::State_BaseOnly : RecordBase::State_ModifiedOnly; } else { // old record RefIdData::LocalIndex localIndex = mData.globalToLocalIndex (index); if (!base) if (mData.getRecord (localIndex).mState==RecordBase::State_Erased) throw std::logic_error ("attempt to access a deleted record"); mData.load (localIndex, reader, base); if (!base) mData.getRecord (localIndex).mState = RecordBase::State_Modified; } } } int CSMWorld::RefIdCollection::getAppendIndex (UniversalId::Type type) const { return mData.getAppendIndex (type); }