mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-30 23:56:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			441 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			441 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "topicinfocheck.hpp"
 | |
| 
 | |
| #include <sstream>
 | |
| 
 | |
| #include "../world/infoselectwrapper.hpp"
 | |
| 
 | |
| CSMTools::TopicInfoCheckStage::TopicInfoCheckStage(
 | |
|     const CSMWorld::InfoCollection& topicInfos,
 | |
|     const CSMWorld::IdCollection<CSMWorld::Cell>& cells,
 | |
|     const CSMWorld::IdCollection<ESM::Class>& classes,
 | |
|     const CSMWorld::IdCollection<ESM::Faction>& factions,
 | |
|     const CSMWorld::IdCollection<ESM::GameSetting>& gmsts,
 | |
|     const CSMWorld::IdCollection<ESM::Global>& globals,
 | |
|     const CSMWorld::IdCollection<ESM::Dialogue>& journals,
 | |
|     const CSMWorld::IdCollection<ESM::Race>& races,
 | |
|     const CSMWorld::IdCollection<ESM::Region>& regions,
 | |
|     const CSMWorld::IdCollection<ESM::Dialogue> &topics,
 | |
|     const CSMWorld::RefIdData& referencables,
 | |
|     const CSMWorld::Resources& soundFiles)
 | |
|     : mTopicInfos(topicInfos),
 | |
|       mCells(cells),
 | |
|       mClasses(classes),
 | |
|       mFactions(factions),
 | |
|       mGameSettings(gmsts),
 | |
|       mGlobals(globals),
 | |
|       mJournals(journals),
 | |
|       mRaces(races),
 | |
|       mRegions(regions),
 | |
|       mTopics(topics),
 | |
|       mReferencables(referencables),
 | |
|       mSoundFiles(soundFiles)
 | |
| {}
 | |
| 
 | |
| int CSMTools::TopicInfoCheckStage::setup()
 | |
| {
 | |
|     // Generate list of cell names for reference checking
 | |
| 
 | |
|     mCellNames.clear();
 | |
|     for (int i = 0; i < mCells.getSize(); ++i)
 | |
|     {
 | |
|         const CSMWorld::Record<CSMWorld::Cell>& cellRecord = mCells.getRecord(i);
 | |
| 
 | |
|         if (cellRecord.isDeleted())
 | |
|             continue;
 | |
| 
 | |
|         mCellNames.insert(cellRecord.get().mName);
 | |
|     }
 | |
|     // Cell names can also include region names
 | |
|     for (int i = 0; i < mRegions.getSize(); ++i)
 | |
|     {
 | |
|         const CSMWorld::Record<ESM::Region>& regionRecord = mRegions.getRecord(i);
 | |
| 
 | |
|         if (regionRecord.isDeleted())
 | |
|             continue;
 | |
| 
 | |
|         mCellNames.insert(regionRecord.get().mName);
 | |
|     }
 | |
|     // Default cell name
 | |
|     int index = mGameSettings.searchId("sDefaultCellname");
 | |
|     if (index != -1)
 | |
|     {
 | |
|         const CSMWorld::Record<ESM::GameSetting>& gmstRecord = mGameSettings.getRecord(index);
 | |
| 
 | |
|         if (!gmstRecord.isDeleted() && gmstRecord.get().mValue.getType() == ESM::VT_String)
 | |
|         {
 | |
|             mCellNames.insert(gmstRecord.get().mValue.getString());
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return mTopicInfos.getSize();
 | |
| }
 | |
| 
 | |
| void CSMTools::TopicInfoCheckStage::perform(int stage, CSMDoc::Messages& messages)
 | |
| {
 | |
|     const CSMWorld::Record<CSMWorld::Info>& infoRecord = mTopicInfos.getRecord(stage);
 | |
| 
 | |
|     if (infoRecord.isDeleted())
 | |
|         return;
 | |
| 
 | |
|     const CSMWorld::Info& topicInfo = infoRecord.get();
 | |
| 
 | |
|     // There should always be a topic that matches
 | |
|     int topicIndex = mTopics.searchId(topicInfo.mTopicId);
 | |
| 
 | |
|     const CSMWorld::Record<ESM::Dialogue>& topicRecord = mTopics.getRecord(topicIndex);
 | |
| 
 | |
|     if (topicRecord.isDeleted())
 | |
|         return;
 | |
| 
 | |
|     const ESM::Dialogue& topic = topicRecord.get();
 | |
| 
 | |
|     CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_TopicInfo, topicInfo.mId);
 | |
| 
 | |
|     // Check fields
 | |
| 
 | |
|     if (!topicInfo.mActor.empty())
 | |
|     {
 | |
|         verifyActor(topicInfo.mActor, id, messages);
 | |
|     }
 | |
| 
 | |
|     if (!topicInfo.mClass.empty())
 | |
|     {
 | |
|         verifyId(topicInfo.mClass, mClasses, id, messages);
 | |
|     }
 | |
| 
 | |
|     if (!topicInfo.mCell.empty())
 | |
|     {
 | |
|         verifyCell(topicInfo.mCell, id, messages);
 | |
|     }
 | |
| 
 | |
|     if (!topicInfo.mFaction.empty() && !topicInfo.mFactionLess)
 | |
|     {
 | |
|         if (verifyId(topicInfo.mFaction, mFactions, id, messages))
 | |
|         {
 | |
|             verifyFactionRank(topicInfo.mFaction, topicInfo.mData.mRank, id, messages);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (!topicInfo.mPcFaction.empty())
 | |
|     {
 | |
|         if (verifyId(topicInfo.mPcFaction, mFactions, id, messages))
 | |
|         {
 | |
|             verifyFactionRank(topicInfo.mPcFaction, topicInfo.mData.mPCrank, id, messages);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (topicInfo.mData.mGender < -1 || topicInfo.mData.mGender > 1)
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         messages.add(id, "Gender: Value is invalid", "", CSMDoc::Message::Severity_Error);
 | |
|     }
 | |
| 
 | |
|     if (!topicInfo.mRace.empty())
 | |
|     {
 | |
|         verifyId(topicInfo.mRace, mRaces, id, messages);
 | |
|     }
 | |
| 
 | |
|     if (!topicInfo.mSound.empty())
 | |
|     {
 | |
|         verifySound(topicInfo.mSound, id, messages);
 | |
|     }
 | |
| 
 | |
|     if (topicInfo.mResponse.empty() && topic.mType != ESM::Dialogue::Voice)
 | |
|     {
 | |
|         messages.add(id, "Response is empty", "", CSMDoc::Message::Severity_Warning);
 | |
|     }
 | |
| 
 | |
|     // Check info conditions
 | |
| 
 | |
|     for (std::vector<ESM::DialInfo::SelectStruct>::const_iterator it = topicInfo.mSelects.begin();
 | |
|          it != topicInfo.mSelects.end(); ++it)
 | |
|     {
 | |
|         verifySelectStruct((*it), id, messages);
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Verification functions
 | |
| 
 | |
| bool CSMTools::TopicInfoCheckStage::verifyActor(const std::string& actor, const CSMWorld::UniversalId& id,
 | |
|     CSMDoc::Messages& messages)
 | |
| {
 | |
|     const std::string specifier = "Actor";
 | |
| 
 | |
|     CSMWorld::RefIdData::LocalIndex index = mReferencables.searchId(actor);
 | |
| 
 | |
|     if (index.first == -1)
 | |
|     {
 | |
|         writeMissingIdError(specifier, actor, id, messages);
 | |
|         return false;
 | |
|     }
 | |
|     else if (mReferencables.getRecord(index).isDeleted())
 | |
|     {
 | |
|         writeDeletedRecordError(specifier, actor, id, messages);
 | |
|         return false;
 | |
|     }
 | |
|     else if (index.second != CSMWorld::UniversalId::Type_Npc && index.second != CSMWorld::UniversalId::Type_Creature)
 | |
|     {
 | |
|         writeInvalidTypeError(specifier, actor, index.second, "NPC or Creature", id, messages);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool CSMTools::TopicInfoCheckStage::verifyCell(const std::string& cell, const CSMWorld::UniversalId& id,
 | |
|     CSMDoc::Messages& messages)
 | |
| {
 | |
|     const std::string specifier = "Cell";
 | |
| 
 | |
|     if (mCellNames.find(cell) == mCellNames.end())
 | |
|     {
 | |
|         writeMissingIdError(specifier, cell, id, messages);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool CSMTools::TopicInfoCheckStage::verifyFactionRank(const std::string& factionName, int rank, const CSMWorld::UniversalId& id,
 | |
|     CSMDoc::Messages& messages)
 | |
| {
 | |
|     if (rank < -1)
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         stream << "Rank or PC Rank is set to " << rank << ", but should be set to -1 if no rank is required";
 | |
| 
 | |
|         messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Error);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     int index = mFactions.searchId(factionName);
 | |
| 
 | |
|     const ESM::Faction &faction = mFactions.getRecord(index).get();
 | |
| 
 | |
|     int limit = 0;
 | |
|     for (; limit < 10; ++limit)
 | |
|     {
 | |
|         if (faction.mRanks[limit].empty())
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     if (rank >= limit)
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         stream << "Rank or PC Rank is set to " << rank << " which is more than the maximum of " << limit - 1
 | |
|                << " for the " << factionName << " faction";
 | |
| 
 | |
|         messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Error);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool CSMTools::TopicInfoCheckStage::verifyItem(const std::string& item, const CSMWorld::UniversalId& id,
 | |
|     CSMDoc::Messages& messages)
 | |
| {
 | |
|     const std::string specifier = "Item";
 | |
| 
 | |
|     CSMWorld::RefIdData::LocalIndex index = mReferencables.searchId(item);
 | |
| 
 | |
|     if (index.first == -1)
 | |
|     {
 | |
|         writeMissingIdError(specifier, item, id, messages);
 | |
|         return false;
 | |
|     }
 | |
|     else if (mReferencables.getRecord(index).isDeleted())
 | |
|     {
 | |
|         writeDeletedRecordError(specifier, item, id, messages);
 | |
|         return false;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         switch (index.second)
 | |
|         {
 | |
|             case CSMWorld::UniversalId::Type_Potion:
 | |
|             case CSMWorld::UniversalId::Type_Apparatus:
 | |
|             case CSMWorld::UniversalId::Type_Armor:
 | |
|             case CSMWorld::UniversalId::Type_Book:
 | |
|             case CSMWorld::UniversalId::Type_Clothing:
 | |
|             case CSMWorld::UniversalId::Type_Ingredient:
 | |
|             case CSMWorld::UniversalId::Type_Light:
 | |
|             case CSMWorld::UniversalId::Type_Lockpick:
 | |
|             case CSMWorld::UniversalId::Type_Miscellaneous:
 | |
|             case CSMWorld::UniversalId::Type_Probe:
 | |
|             case CSMWorld::UniversalId::Type_Repair:
 | |
|             case CSMWorld::UniversalId::Type_Weapon:
 | |
|             case CSMWorld::UniversalId::Type_ItemLevelledList:
 | |
|                 break;
 | |
| 
 | |
|             default:
 | |
|                 writeInvalidTypeError(specifier, item, index.second, "Potion, Armor, Book, etc.", id, messages);
 | |
|                 return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool CSMTools::TopicInfoCheckStage::verifySelectStruct(const ESM::DialInfo::SelectStruct& select,
 | |
|     const CSMWorld::UniversalId& id, CSMDoc::Messages& messages)
 | |
| {
 | |
|     CSMWorld::ConstInfoSelectWrapper infoCondition(select);
 | |
| 
 | |
|     if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_None)
 | |
|     {
 | |
|         messages.add(id, "Invalid Info Condition: " + infoCondition.toString(), "", CSMDoc::Message::Severity_Error);
 | |
|         return false;
 | |
|     }
 | |
|     else if (!infoCondition.variantTypeIsValid())
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         stream << "Info Condition: Value for \"" << infoCondition.toString() << "\" has a type of ";
 | |
| 
 | |
|         switch (select.mValue.getType())
 | |
|         {
 | |
|             case ESM::VT_None:   stream << "None"; break;
 | |
|             case ESM::VT_Short:  stream << "Short"; break;
 | |
|             case ESM::VT_Int:    stream << "Int"; break;
 | |
|             case ESM::VT_Long:   stream << "Long"; break;
 | |
|             case ESM::VT_Float:  stream << "Float"; break;
 | |
|             case ESM::VT_String: stream << "String"; break;
 | |
|             default:             stream << "Unknown"; break;
 | |
|         }
 | |
| 
 | |
|         messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Error);
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.conditionIsAlwaysTrue())
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         stream << "Info Condition: " << infoCondition.toString() << " is always true";
 | |
| 
 | |
|         messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Warning);
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.conditionIsNeverTrue())
 | |
|     {
 | |
|         std::ostringstream stream;
 | |
|         stream << "Info Condition: " << infoCondition.toString() << " is never true";
 | |
| 
 | |
|         messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Warning);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     // Id checks
 | |
|     if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_Global &&
 | |
|         !verifyId(infoCondition.getVariableName(), mGlobals, id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_Journal &&
 | |
|         !verifyId(infoCondition.getVariableName(), mJournals, id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_Item &&
 | |
|         !verifyItem(infoCondition.getVariableName(), id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_Dead &&
 | |
|         !verifyActor(infoCondition.getVariableName(), id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_NotId &&
 | |
|         !verifyActor(infoCondition.getVariableName(), id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_NotFaction &&
 | |
|         !verifyId(infoCondition.getVariableName(), mFactions, id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_NotClass &&
 | |
|         !verifyId(infoCondition.getVariableName(), mClasses, id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_NotRace &&
 | |
|         !verifyId(infoCondition.getVariableName(), mRaces, id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
|     else if (infoCondition.getFunctionName() == CSMWorld::ConstInfoSelectWrapper::Function_NotCell &&
 | |
|         !verifyCell(infoCondition.getVariableName(), id, messages))
 | |
|     {
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool CSMTools::TopicInfoCheckStage::verifySound(const std::string& sound, const CSMWorld::UniversalId& id,
 | |
|     CSMDoc::Messages& messages)
 | |
| {
 | |
|     const std::string specifier = "Sound File";
 | |
| 
 | |
|     if (mSoundFiles.searchId(sound) == -1)
 | |
|     {
 | |
|         writeMissingIdError(specifier, sound, id, messages);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| template <typename T>
 | |
| bool CSMTools::TopicInfoCheckStage::verifyId(const std::string& name, const CSMWorld::IdCollection<T>& collection,
 | |
|     const CSMWorld::UniversalId& id, CSMDoc::Messages& messages)
 | |
| {
 | |
|     int index = collection.searchId(name);
 | |
| 
 | |
|     if (index == -1)
 | |
|     {
 | |
|         writeMissingIdError(T::getRecordType(), name, id, messages);
 | |
|         return false;
 | |
|     }
 | |
|     else if (collection.getRecord(index).isDeleted())
 | |
|     {
 | |
|         writeDeletedRecordError(T::getRecordType(), name, id, messages);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| // Error functions
 | |
| 
 | |
| void CSMTools::TopicInfoCheckStage::writeMissingIdError(const std::string& specifier, const std::string& missingId,
 | |
|     const CSMWorld::UniversalId& id, CSMDoc::Messages& messages)
 | |
| {
 | |
|     std::ostringstream stream;
 | |
|     stream << specifier << ": ID or name \"" << missingId << "\" could not be found";
 | |
| 
 | |
|     messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Error);
 | |
| }
 | |
| 
 | |
| void CSMTools::TopicInfoCheckStage::writeDeletedRecordError(const std::string& specifier, const std::string& recordId,
 | |
|     const CSMWorld::UniversalId& id, CSMDoc::Messages& messages)
 | |
| {
 | |
|     std::ostringstream stream;
 | |
|     stream << specifier << ": Deleted record with ID \"" << recordId << "\" is being referenced";
 | |
| 
 | |
|     messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Error);
 | |
| }
 | |
| 
 | |
| void CSMTools::TopicInfoCheckStage::writeInvalidTypeError(const std::string& specifier, const std::string& invalidId,
 | |
|     CSMWorld::UniversalId::Type invalidType, const std::string& expectedType, const CSMWorld::UniversalId& id,
 | |
|     CSMDoc::Messages& messages)
 | |
| {
 | |
|     CSMWorld::UniversalId tempId(invalidType, invalidId);
 | |
| 
 | |
|     std::ostringstream stream;
 | |
|     stream << specifier << ": invalid type of " << tempId.getTypeName() << " was found for referencable \""
 | |
|            << invalidId << "\" (can be of type " << expectedType << ")";
 | |
| 
 | |
|     messages.add(id, stream.str(), "", CSMDoc::Message::Severity_Error);
 | |
| }
 |