1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-17 19:10:24 +00:00
OpenMW/apps/opencs/model/tools/scriptcheck.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

158 lines
4.3 KiB
C++
Raw Normal View History

2014-02-14 12:38:30 +00:00
#include "scriptcheck.hpp"
2022-10-19 17:02:00 +00:00
#include <exception>
#include <sstream>
2022-10-19 17:02:00 +00:00
#include <string>
#include <apps/opencs/model/doc/messages.hpp>
#include <apps/opencs/model/prefs/category.hpp>
#include <apps/opencs/model/prefs/setting.hpp>
#include <apps/opencs/model/world/idcollection.hpp>
#include <apps/opencs/model/world/record.hpp>
#include <apps/opencs/model/world/scriptcontext.hpp>
#include <apps/opencs/model/world/universalid.hpp>
2014-02-14 12:38:30 +00:00
#include <components/compiler/exception.hpp>
#include <components/compiler/extensions0.hpp>
#include <components/compiler/fileparser.hpp>
#include <components/compiler/scanner.hpp>
#include <components/compiler/tokenloc.hpp>
2022-10-19 17:02:00 +00:00
#include <components/esm3/loadscpt.hpp>
2014-02-14 12:38:30 +00:00
#include "../doc/document.hpp"
2014-02-14 12:38:30 +00:00
#include "../world/data.hpp"
#include "../prefs/state.hpp"
CSMDoc::Message::Severity CSMTools::ScriptCheckStage::getSeverity(Type type)
{
switch (type)
{
case WarningMessage:
return CSMDoc::Message::Severity_Warning;
case ErrorMessage:
return CSMDoc::Message::Severity_Error;
}
return CSMDoc::Message::Severity_SeriousError;
}
2014-02-14 12:38:30 +00:00
void CSMTools::ScriptCheckStage::report(const std::string& message, const Compiler::TokenLoc& loc, Type type)
{
std::ostringstream stream;
CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Script, mId);
2019-03-05 17:47:19 +00:00
stream << message << " (" << loc.mLiteral << ")"
<< " @ line " << loc.mLine + 1 << ", column " << loc.mColumn;
2014-02-14 12:38:30 +00:00
std::ostringstream hintStream;
2019-03-05 17:47:19 +00:00
hintStream << "l:" << loc.mLine + 1 << " " << loc.mColumn;
mMessages->add(id, stream.str(), hintStream.str(), getSeverity(type));
2014-02-14 12:38:30 +00:00
}
void CSMTools::ScriptCheckStage::report(const std::string& message, Type type)
{
CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Script, mId);
std::ostringstream stream;
stream << message;
mMessages->add(id, stream.str(), "", getSeverity(type));
2014-02-14 12:38:30 +00:00
}
CSMTools::ScriptCheckStage::ScriptCheckStage(const CSMDoc::Document& document)
2020-11-13 07:39:47 +00:00
: mDocument(document)
, mContext(document.getData())
, mMessages(nullptr)
, mWarningMode(Mode_Ignore)
2014-02-14 12:38:30 +00:00
{
/// \todo add an option to configure warning mode
setWarningsMode(0);
2014-02-14 12:38:30 +00:00
Compiler::registerExtensions(mExtensions);
mContext.setExtensions(&mExtensions);
mIgnoreBaseRecords = false;
2014-02-14 12:38:30 +00:00
}
int CSMTools::ScriptCheckStage::setup()
{
std::string warnings = CSMPrefs::get()["Scripts"]["warnings"].toString();
if (warnings == "Ignore")
mWarningMode = Mode_Ignore;
else if (warnings == "Normal")
mWarningMode = Mode_Normal;
else if (warnings == "Strict")
mWarningMode = Mode_Strict;
2014-02-14 12:38:30 +00:00
mContext.clear();
2020-11-13 07:39:47 +00:00
mMessages = nullptr;
mId = ESM::RefId();
Compiler::ErrorHandler::reset();
2014-02-14 12:38:30 +00:00
mIgnoreBaseRecords = CSMPrefs::get()["Reports"]["ignore-base-records"].isTrue();
return mDocument.getData().getScripts().getSize();
2014-02-14 12:38:30 +00:00
}
void CSMTools::ScriptCheckStage::perform(int stage, CSMDoc::Messages& messages)
2014-02-14 12:38:30 +00:00
{
const CSMWorld::Record<ESM::Script>& record = mDocument.getData().getScripts().getRecord(stage);
mId = mDocument.getData().getScripts().getId(stage);
if (mDocument.isBlacklisted(CSMWorld::UniversalId(CSMWorld::UniversalId::Type_Script, mId)))
return;
// Skip "Base" records (setting!) and "Deleted" records
if ((mIgnoreBaseRecords && record.mState == CSMWorld::RecordBase::State_BaseOnly) || record.isDeleted())
return;
2014-02-14 12:38:30 +00:00
mMessages = &messages;
switch (mWarningMode)
{
case Mode_Ignore:
setWarningsMode(0);
break;
case Mode_Normal:
setWarningsMode(1);
break;
case Mode_Strict:
setWarningsMode(2);
break;
}
2014-02-14 12:38:30 +00:00
try
{
mFile = record.get().mId.getRefIdString();
std::istringstream input(record.get().mScriptText);
2014-02-14 12:38:30 +00:00
Compiler::Scanner scanner(*this, input, mContext.getExtensions());
Compiler::FileParser parser(*this, mContext);
scanner.scan(parser);
}
catch (const Compiler::SourceException&)
{
// error has already been reported via error handler
}
catch (const std::exception& error)
{
CSMWorld::UniversalId id(CSMWorld::UniversalId::Type_Script, mId);
std::ostringstream stream;
stream << error.what();
messages.add(id, stream.str(), "", CSMDoc::Message::Severity_SeriousError);
2014-02-14 12:38:30 +00:00
}
2020-11-13 07:39:47 +00:00
mMessages = nullptr;
2015-03-11 14:54:45 +00:00
}