2015-02-13 03:11:36 +00:00
|
|
|
#include "referencecheck.hpp"
|
|
|
|
|
2018-06-19 22:20:03 +00:00
|
|
|
#include "../prefs/state.hpp"
|
|
|
|
|
2022-08-19 17:19:42 +00:00
|
|
|
#include "../../model/world/cell.hpp"
|
|
|
|
|
|
|
|
#include <components/esm3/loadfact.hpp>
|
|
|
|
|
2015-02-13 03:11:36 +00:00
|
|
|
CSMTools::ReferenceCheckStage::ReferenceCheckStage(
|
|
|
|
const CSMWorld::RefCollection& references,
|
|
|
|
const CSMWorld::RefIdCollection& referencables,
|
|
|
|
const CSMWorld::IdCollection<CSMWorld::Cell>& cells,
|
|
|
|
const CSMWorld::IdCollection<ESM::Faction>& factions)
|
|
|
|
:
|
|
|
|
mReferences(references),
|
2018-08-24 23:03:55 +00:00
|
|
|
mObjects(referencables),
|
2015-02-13 11:13:40 +00:00
|
|
|
mDataSet(referencables.getDataSet()),
|
2015-02-13 03:11:36 +00:00
|
|
|
mCells(cells),
|
|
|
|
mFactions(factions)
|
|
|
|
{
|
2018-06-20 09:29:38 +00:00
|
|
|
mIgnoreBaseRecords = false;
|
2015-02-13 03:11:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSMTools::ReferenceCheckStage::perform(int stage, CSMDoc::Messages &messages)
|
|
|
|
{
|
|
|
|
const CSMWorld::Record<CSMWorld::CellRef>& record = mReferences.getRecord(stage);
|
|
|
|
|
2018-06-19 22:20:03 +00:00
|
|
|
// Skip "Base" records (setting!) and "Deleted" records
|
2018-06-20 09:29:38 +00:00
|
|
|
if ((mIgnoreBaseRecords && record.mState == CSMWorld::RecordBase::State_BaseOnly) || record.isDeleted())
|
2015-02-13 03:11:36 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
const CSMWorld::CellRef& cellRef = record.get();
|
|
|
|
const CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Reference, cellRef.mId);
|
|
|
|
|
2018-08-26 19:16:50 +00:00
|
|
|
// Check reference id
|
|
|
|
if (cellRef.mRefID.empty())
|
|
|
|
messages.add(id, "Instance is not based on an object", "", CSMDoc::Message::Severity_Error);
|
|
|
|
else
|
|
|
|
{
|
2015-02-13 11:13:40 +00:00
|
|
|
// Check for non existing referenced object
|
2018-08-26 19:16:50 +00:00
|
|
|
if (mObjects.searchId(cellRef.mRefID) == -1)
|
|
|
|
messages.add(id, "Instance of a non-existent object '" + cellRef.mRefID + "'", "", CSMDoc::Message::Severity_Error);
|
|
|
|
else
|
|
|
|
{
|
2015-02-13 11:16:46 +00:00
|
|
|
// Check if reference charge is valid for it's proper referenced type
|
2015-02-13 11:13:40 +00:00
|
|
|
CSMWorld::RefIdData::LocalIndex localIndex = mDataSet.searchId(cellRef.mRefID);
|
2015-02-13 12:11:10 +00:00
|
|
|
bool isLight = localIndex.second == CSMWorld::UniversalId::Type_Light;
|
2018-08-26 19:16:50 +00:00
|
|
|
if ((isLight && cellRef.mChargeFloat < -1) || (!isLight && cellRef.mChargeInt < -1))
|
|
|
|
messages.add(id, "Invalid charge", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 11:13:40 +00:00
|
|
|
}
|
|
|
|
}
|
2015-02-13 03:11:36 +00:00
|
|
|
|
|
|
|
// If object have owner, check if that owner reference is valid
|
2018-08-24 23:03:55 +00:00
|
|
|
if (!cellRef.mOwner.empty() && mObjects.searchId(cellRef.mOwner) == -1)
|
2018-08-26 19:16:50 +00:00
|
|
|
messages.add(id, "Owner object '" + cellRef.mOwner + "' does not exist", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 03:11:36 +00:00
|
|
|
|
|
|
|
// If object have creature soul trapped, check if that creature reference is valid
|
|
|
|
if (!cellRef.mSoul.empty())
|
2018-08-24 23:03:55 +00:00
|
|
|
if (mObjects.searchId(cellRef.mSoul) == -1)
|
2019-05-28 23:54:19 +00:00
|
|
|
messages.add(id, "Trapped soul object '" + cellRef.mSoul + "' does not exist", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 03:11:36 +00:00
|
|
|
|
2018-08-26 19:16:50 +00:00
|
|
|
if (cellRef.mFaction.empty())
|
|
|
|
{
|
|
|
|
if (cellRef.mFactionRank != -2)
|
|
|
|
messages.add(id, "Reference without a faction has a faction rank", "", CSMDoc::Message::Severity_Error);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-02-13 03:11:36 +00:00
|
|
|
if (mFactions.searchId(cellRef.mFaction) == -1)
|
2018-08-26 19:16:50 +00:00
|
|
|
messages.add(id, "Faction '" + cellRef.mFaction + "' does not exist", "", CSMDoc::Message::Severity_Error);
|
|
|
|
else if (cellRef.mFactionRank < -1)
|
|
|
|
messages.add(id, "Invalid faction rank", "", CSMDoc::Message::Severity_Error);
|
|
|
|
}
|
2015-02-13 03:11:36 +00:00
|
|
|
|
2018-08-26 19:16:50 +00:00
|
|
|
if (!cellRef.mDestCell.empty() && mCells.searchId(cellRef.mDestCell) == -1)
|
|
|
|
messages.add(id, "Destination cell '" + cellRef.mDestCell + "' does not exist", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 03:11:36 +00:00
|
|
|
|
|
|
|
if (cellRef.mScale < 0)
|
2018-08-26 19:16:50 +00:00
|
|
|
messages.add(id, "Negative scale", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 03:11:36 +00:00
|
|
|
|
|
|
|
// Check if enchantement points aren't negative or are at full (-1)
|
2018-08-26 19:16:50 +00:00
|
|
|
if (cellRef.mEnchantmentCharge < -1)
|
|
|
|
messages.add(id, "Negative number of enchantment points", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 03:11:36 +00:00
|
|
|
|
|
|
|
// Check if gold value isn't negative
|
|
|
|
if (cellRef.mGoldValue < 0)
|
2018-08-26 19:16:50 +00:00
|
|
|
messages.add(id, "Negative gold value", "", CSMDoc::Message::Severity_Error);
|
2015-02-13 03:11:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int CSMTools::ReferenceCheckStage::setup()
|
|
|
|
{
|
2018-06-19 22:20:03 +00:00
|
|
|
mIgnoreBaseRecords = CSMPrefs::get()["Reports"]["ignore-base-records"].isTrue();
|
|
|
|
|
2015-02-13 03:11:36 +00:00
|
|
|
return mReferences.getSize();
|
2015-02-15 03:47:25 +00:00
|
|
|
}
|