mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-26 09:35:28 +00:00
9d61d76e92
Adds a boolean setting to the user preferences. This setting is locally saved to all OpenMW-CS check stages. When a verification is done, the setting is updated on setup for each check stage. If set to true, the boolean value is then used to skip the verification process for every base record - minus some special cases where, e.g., counters are to be set first. Related issue: - Fixes #4466: Editor: Add option to ignore base records when running verifier (https://gitlab.com/OpenMW/openmw/issues/4466) Tests: The changes were successfully tested in OpenMW-CS by creating faulty "Base" and "Modified" records for every record type (if possible) and, then, running the verifier with and without the option respectively.
145 lines
5.5 KiB
C++
145 lines
5.5 KiB
C++
#include "pathgridcheck.hpp"
|
|
|
|
#include <sstream>
|
|
#include <algorithm>
|
|
|
|
#include "../prefs/state.hpp"
|
|
|
|
#include "../world/universalid.hpp"
|
|
#include "../world/idcollection.hpp"
|
|
#include "../world/subcellcollection.hpp"
|
|
#include "../world/pathgrid.hpp"
|
|
|
|
CSMTools::PathgridCheckStage::PathgridCheckStage (const CSMWorld::SubCellCollection<CSMWorld::Pathgrid>& pathgrids)
|
|
: mPathgrids (pathgrids)
|
|
{
|
|
mIgnoreBaseRecords = CSMPrefs::get()["Reports"]["ignore-base-records"].isTrue();
|
|
}
|
|
|
|
int CSMTools::PathgridCheckStage::setup()
|
|
{
|
|
mIgnoreBaseRecords = CSMPrefs::get()["Reports"]["ignore-base-records"].isTrue();
|
|
|
|
return mPathgrids.getSize();
|
|
}
|
|
|
|
void CSMTools::PathgridCheckStage::perform (int stage, CSMDoc::Messages& messages)
|
|
{
|
|
const CSMWorld::Record<CSMWorld::Pathgrid>& record = mPathgrids.getRecord (stage);
|
|
|
|
// Skip "Base" records (setting!) and "Deleted" records
|
|
if ((mIgnoreBaseRecords && record.isBaseOnly()) || record.isDeleted())
|
|
return;
|
|
|
|
const CSMWorld::Pathgrid& pathgrid = record.get();
|
|
|
|
CSMWorld::UniversalId id (CSMWorld::UniversalId::Type_Pathgrid, pathgrid.mId);
|
|
|
|
// check the number of pathgrid points
|
|
if (pathgrid.mData.mS2 < static_cast<int>(pathgrid.mPoints.size()))
|
|
messages.add (id, pathgrid.mId + " has less points than expected", "", CSMDoc::Message::Severity_Error);
|
|
else if (pathgrid.mData.mS2 > static_cast<int>(pathgrid.mPoints.size()))
|
|
messages.add (id, pathgrid.mId + " has more points than expected", "", CSMDoc::Message::Severity_Error);
|
|
|
|
std::vector<CSMTools::Point> pointList(pathgrid.mPoints.size());
|
|
std::vector<int> duplList;
|
|
|
|
for (unsigned int i = 0; i < pathgrid.mEdges.size(); ++i)
|
|
{
|
|
if (pathgrid.mEdges[i].mV0 < static_cast<int>(pathgrid.mPoints.size()) && pathgrid.mEdges[i].mV0 >= 0)
|
|
{
|
|
pointList[pathgrid.mEdges[i].mV0].mConnectionNum++;
|
|
// first check for duplicate edges
|
|
unsigned int j = 0;
|
|
for (; j < pointList[pathgrid.mEdges[i].mV0].mOtherIndex.size(); ++j)
|
|
{
|
|
if (pointList[pathgrid.mEdges[i].mV0].mOtherIndex[j] == pathgrid.mEdges[i].mV1)
|
|
{
|
|
std::ostringstream ss;
|
|
ss << "has a duplicate edge between points" << pathgrid.mEdges[i].mV0
|
|
<< " and " << pathgrid.mEdges[i].mV1;
|
|
messages.add (id, pathgrid.mId + ss.str(), "", CSMDoc::Message::Severity_Error);
|
|
break;
|
|
}
|
|
}
|
|
|
|
// only add if not a duplicate
|
|
if (j == pointList[pathgrid.mEdges[i].mV0].mOtherIndex.size())
|
|
pointList[pathgrid.mEdges[i].mV0].mOtherIndex.push_back(pathgrid.mEdges[i].mV1);
|
|
}
|
|
else
|
|
{
|
|
std::ostringstream ss;
|
|
ss << " has an edge connecting a non-existent point " << pathgrid.mEdges[i].mV0;
|
|
messages.add (id, pathgrid.mId + ss.str(), "", CSMDoc::Message::Severity_Error);
|
|
}
|
|
}
|
|
|
|
for (unsigned int i = 0; i < pathgrid.mPoints.size(); ++i)
|
|
{
|
|
// check that edges are bidirectional
|
|
bool foundReverse = false;
|
|
for (unsigned int j = 0; j < pointList[i].mOtherIndex.size(); ++j)
|
|
{
|
|
for (unsigned int k = 0; k < pointList[pointList[i].mOtherIndex[j]].mOtherIndex.size(); ++k)
|
|
{
|
|
if (pointList[pointList[i].mOtherIndex[j]].mOtherIndex[k] == static_cast<int>(i))
|
|
{
|
|
foundReverse = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!foundReverse)
|
|
{
|
|
std::ostringstream ss;
|
|
ss << " has a missing edge between points " << i << " and " << pointList[i].mOtherIndex[j];
|
|
messages.add (id, pathgrid.mId + ss.str(), "", CSMDoc::Message::Severity_Error);
|
|
}
|
|
}
|
|
|
|
// check duplicate points
|
|
// FIXME: how to do this efficiently?
|
|
for (unsigned int j = 0; j < pathgrid.mPoints.size(); ++j)
|
|
{
|
|
if (j == i)
|
|
continue;
|
|
|
|
if (pathgrid.mPoints[i].mX == pathgrid.mPoints[j].mX &&
|
|
pathgrid.mPoints[i].mY == pathgrid.mPoints[j].mY &&
|
|
pathgrid.mPoints[i].mZ == pathgrid.mPoints[j].mZ)
|
|
{
|
|
std::vector<int>::const_iterator it = find(duplList.begin(), duplList.end(), i);
|
|
if (it == duplList.end())
|
|
{
|
|
std::ostringstream ss;
|
|
ss << " has a duplicated point (" << i
|
|
<< ") x=" << pathgrid.mPoints[i].mX
|
|
<< ", y=" << pathgrid.mPoints[i].mY
|
|
<< ", z=" << pathgrid.mPoints[i].mZ;
|
|
messages.add (id, pathgrid.mId + ss.str(), "", CSMDoc::Message::Severity_Warning);
|
|
|
|
duplList.push_back(i);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// check pathgrid points that are not connected to anything
|
|
for (unsigned int i = 0; i < pointList.size(); ++i)
|
|
{
|
|
if (pointList[i].mConnectionNum == 0)
|
|
{
|
|
std::ostringstream ss;
|
|
ss << " has an orphaned point (" << i
|
|
<< ") x=" << pathgrid.mPoints[i].mX
|
|
<< ", y=" << pathgrid.mPoints[i].mY
|
|
<< ", z=" << pathgrid.mPoints[i].mZ;
|
|
messages.add (id, pathgrid.mId + ss.str(), "", CSMDoc::Message::Severity_Warning);
|
|
}
|
|
}
|
|
|
|
// TODO: check whether there are disconnected graphs
|
|
}
|