2013-04-10 20:49:22 +00:00
|
|
|
#include "scripthighlighter.hpp"
|
|
|
|
|
|
|
|
#include <sstream>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <utility>
|
2013-04-10 20:49:22 +00:00
|
|
|
|
2022-10-18 07:26:55 +00:00
|
|
|
#include <apps/opencs/model/prefs/category.hpp>
|
|
|
|
#include <apps/opencs/model/prefs/setting.hpp>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <apps/opencs/model/world/scriptcontext.hpp>
|
2013-09-19 06:58:27 +00:00
|
|
|
#include <components/compiler/extensions0.hpp>
|
2013-04-10 20:49:22 +00:00
|
|
|
#include <components/compiler/scanner.hpp>
|
2022-10-19 17:02:00 +00:00
|
|
|
#include <components/compiler/tokenloc.hpp>
|
2022-10-06 17:39:46 +00:00
|
|
|
#include <components/esm/refid.hpp>
|
2015-05-29 12:20:43 +00:00
|
|
|
|
2022-10-19 17:02:00 +00:00
|
|
|
class QTextDocument;
|
|
|
|
|
|
|
|
namespace CSMWorld
|
|
|
|
{
|
|
|
|
class Data;
|
|
|
|
}
|
|
|
|
|
2013-04-10 20:49:22 +00:00
|
|
|
bool CSVWorld::ScriptHighlighter::parseInt(int value, const Compiler::TokenLoc& loc, Compiler::Scanner& scanner)
|
|
|
|
{
|
|
|
|
highlight(loc, Type_Int);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVWorld::ScriptHighlighter::parseFloat(float value, const Compiler::TokenLoc& loc, Compiler::Scanner& scanner)
|
|
|
|
{
|
|
|
|
highlight(loc, Type_Float);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVWorld::ScriptHighlighter::parseName(
|
|
|
|
const std::string& name, const Compiler::TokenLoc& loc, Compiler::Scanner& scanner)
|
|
|
|
{
|
2022-10-06 17:39:46 +00:00
|
|
|
highlight(loc, mContext.isId(ESM::RefId::stringRefId(name)) ? Type_Id : Type_Name);
|
2013-04-10 20:49:22 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVWorld::ScriptHighlighter::parseKeyword(int keyword, const Compiler::TokenLoc& loc, Compiler::Scanner& scanner)
|
|
|
|
{
|
2014-08-24 15:56:20 +00:00
|
|
|
if (((mMode == Mode_Console || mMode == Mode_Dialogue)
|
|
|
|
&& (keyword == Compiler::Scanner::K_begin || keyword == Compiler::Scanner::K_end
|
|
|
|
|| keyword == Compiler::Scanner::K_short || keyword == Compiler::Scanner::K_long
|
|
|
|
|| keyword == Compiler::Scanner::K_float))
|
|
|
|
|| (mMode == Mode_Console
|
|
|
|
&& (keyword == Compiler::Scanner::K_if || keyword == Compiler::Scanner::K_endif
|
|
|
|
|| keyword == Compiler::Scanner::K_else || keyword == Compiler::Scanner::K_elseif
|
|
|
|
|| keyword == Compiler::Scanner::K_while || keyword == Compiler::Scanner::K_endwhile)))
|
|
|
|
return parseName(loc.mLiteral, loc, scanner);
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2013-04-10 20:49:22 +00:00
|
|
|
highlight(loc, Type_Keyword);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CSVWorld::ScriptHighlighter::parseSpecial(int code, const Compiler::TokenLoc& loc, Compiler::Scanner& scanner)
|
|
|
|
{
|
|
|
|
highlight(loc, Type_Special);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-04-11 08:50:35 +00:00
|
|
|
bool CSVWorld::ScriptHighlighter::parseComment(
|
|
|
|
const std::string& comment, const Compiler::TokenLoc& loc, Compiler::Scanner& scanner)
|
|
|
|
{
|
|
|
|
highlight(loc, Type_Comment);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-04-10 20:49:22 +00:00
|
|
|
void CSVWorld::ScriptHighlighter::parseEOF(Compiler::Scanner& scanner) {}
|
|
|
|
|
|
|
|
void CSVWorld::ScriptHighlighter::highlight(const Compiler::TokenLoc& loc, Type type)
|
|
|
|
{
|
2019-10-29 07:05:18 +00:00
|
|
|
// We should take in account multibyte characters
|
|
|
|
int length = 0;
|
|
|
|
const char* token = loc.mLiteral.c_str();
|
|
|
|
while (*token)
|
|
|
|
length += (*token++ & 0xc0) != 0x80;
|
2013-04-10 20:49:22 +00:00
|
|
|
|
|
|
|
int index = loc.mColumn;
|
|
|
|
|
|
|
|
// compensate for bug in Compiler::Scanner (position of token is the character after the token)
|
|
|
|
index -= length;
|
|
|
|
|
2017-04-28 07:28:05 +00:00
|
|
|
QTextCharFormat scheme = mScheme[type];
|
2017-05-02 05:28:42 +00:00
|
|
|
if (mMarkOccurrences && type == Type_Name && loc.mLiteral == mMarkedWord)
|
2017-04-28 07:28:05 +00:00
|
|
|
scheme.merge(mScheme[Type_Highlight]);
|
|
|
|
|
|
|
|
setFormat(index, length, scheme);
|
2013-04-10 20:49:22 +00:00
|
|
|
}
|
|
|
|
|
2014-08-24 15:56:20 +00:00
|
|
|
CSVWorld::ScriptHighlighter::ScriptHighlighter(const CSMWorld::Data& data, Mode mode, QTextDocument* parent)
|
2018-11-13 19:07:01 +00:00
|
|
|
: QSyntaxHighlighter(parent)
|
|
|
|
, Compiler::Parser(mErrorHandler, mContext)
|
|
|
|
, mContext(data)
|
|
|
|
, mMode(mode)
|
|
|
|
, mMarkOccurrences(false)
|
2013-04-10 20:49:22 +00:00
|
|
|
{
|
2015-12-12 13:49:16 +00:00
|
|
|
QColor color("black");
|
|
|
|
QTextCharFormat format;
|
|
|
|
format.setForeground(color);
|
2015-05-29 12:20:43 +00:00
|
|
|
|
2015-12-12 13:49:16 +00:00
|
|
|
for (int i = 0; i <= Type_Id; ++i)
|
|
|
|
mScheme.insert(std::make_pair(static_cast<Type>(i), format));
|
2013-09-19 10:30:42 +00:00
|
|
|
|
2013-09-19 06:58:27 +00:00
|
|
|
// configure compiler
|
|
|
|
Compiler::registerExtensions(mExtensions);
|
|
|
|
mContext.setExtensions(&mExtensions);
|
2013-04-10 20:49:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CSVWorld::ScriptHighlighter::highlightBlock(const QString& text)
|
|
|
|
{
|
|
|
|
std::istringstream stream(text.toUtf8().constData());
|
|
|
|
|
|
|
|
Compiler::Scanner scanner(mErrorHandler, stream, mContext.getExtensions());
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
scanner.scan(*this);
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
} // ignore syntax errors
|
2013-09-19 12:26:09 +00:00
|
|
|
}
|
2013-04-10 20:49:22 +00:00
|
|
|
|
2017-04-28 07:57:49 +00:00
|
|
|
void CSVWorld::ScriptHighlighter::setMarkOccurrences(bool flag)
|
|
|
|
{
|
|
|
|
mMarkOccurrences = flag;
|
|
|
|
}
|
|
|
|
|
2017-04-28 07:28:05 +00:00
|
|
|
void CSVWorld::ScriptHighlighter::setMarkedWord(const std::string& name)
|
|
|
|
{
|
|
|
|
mMarkedWord = name;
|
|
|
|
}
|
|
|
|
|
2013-09-19 12:26:09 +00:00
|
|
|
void CSVWorld::ScriptHighlighter::invalidateIds()
|
|
|
|
{
|
|
|
|
mContext.invalidateIds();
|
2015-03-11 14:54:45 +00:00
|
|
|
}
|
2015-05-29 12:20:43 +00:00
|
|
|
|
2015-12-12 13:49:16 +00:00
|
|
|
bool CSVWorld::ScriptHighlighter::settingChanged(const CSMPrefs::Setting* setting)
|
2015-05-29 12:20:43 +00:00
|
|
|
{
|
2015-12-12 13:49:16 +00:00
|
|
|
if (setting->getParent()->getKey() == "Scripts")
|
2015-05-29 12:20:43 +00:00
|
|
|
{
|
2015-12-12 13:49:16 +00:00
|
|
|
static const char* const colours[Type_Id + 2] = { "colour-int", "colour-float", "colour-name", "colour-keyword",
|
2017-04-28 07:28:05 +00:00
|
|
|
"colour-special", "colour-comment", "colour-highlight", "colour-id", 0 };
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2015-12-12 13:49:16 +00:00
|
|
|
for (int i = 0; colours[i]; ++i)
|
|
|
|
if (setting->getKey() == colours[i])
|
|
|
|
{
|
|
|
|
QTextCharFormat format;
|
2017-04-28 07:28:05 +00:00
|
|
|
if (i == Type_Highlight)
|
|
|
|
format.setBackground(setting->toColor());
|
|
|
|
else
|
|
|
|
format.setForeground(setting->toColor());
|
2015-12-12 13:49:16 +00:00
|
|
|
mScheme[static_cast<Type>(i)] = format;
|
|
|
|
return true;
|
|
|
|
}
|
2015-05-29 12:20:43 +00:00
|
|
|
}
|
|
|
|
|
2015-12-12 13:49:16 +00:00
|
|
|
return false;
|
2015-05-29 12:20:43 +00:00
|
|
|
}
|