230 lines
7.3 KiB
C++
230 lines
7.3 KiB
C++
#include <QtWidgets>
|
|
|
|
#include "codeeditor.h"
|
|
|
|
|
|
Highlighter::Highlighter(QTextDocument *parent)
|
|
: QSyntaxHighlighter(parent)
|
|
{
|
|
HighlightingRule rule;
|
|
|
|
keywordFormat.setForeground(Qt::darkBlue);
|
|
keywordFormat.setFontWeight(QFont::Bold);
|
|
QStringList keywordPatterns;
|
|
keywordPatterns << "\\bchar\\b" << "\\bclass\\b" << "\\bconst\\b"
|
|
<< "\\bdouble\\b" << "\\benum\\b" << "\\bexplicit\\b"
|
|
<< "\\bfriend\\b" << "\\binline\\b" << "\\bint\\b"
|
|
<< "\\blong\\b" << "\\bnamespace\\b" << "\\boperator\\b"
|
|
<< "\\bprivate\\b" << "\\bprotected\\b" << "\\bpublic\\b"
|
|
<< "\\bshort\\b" << "\\bsignals\\b" << "\\bsigned\\b"
|
|
<< "\\bslots\\b" << "\\bstatic\\b" << "\\bstruct\\b"
|
|
<< "\\btemplate\\b" << "\\btypedef\\b" << "\\btypename\\b"
|
|
<< "\\bunion\\b" << "\\bunsigned\\b" << "\\bvirtual\\b"
|
|
<< "\\bvoid\\b" << "\\bvolatile\\b" << "\\bbool\\b";
|
|
foreach (const QString &pattern, keywordPatterns) {
|
|
rule.pattern = QRegularExpression(pattern);
|
|
rule.format = keywordFormat;
|
|
highlightingRules.append(rule);
|
|
}
|
|
|
|
classFormat.setFontWeight(QFont::Bold);
|
|
classFormat.setForeground(Qt::darkMagenta);
|
|
rule.pattern = QRegularExpression("\\bQ[A-Za-z]+\\b");
|
|
rule.format = classFormat;
|
|
highlightingRules.append(rule);
|
|
|
|
quotationFormat.setForeground(Qt::darkGreen);
|
|
rule.pattern = QRegularExpression("\".*\"");
|
|
rule.format = quotationFormat;
|
|
highlightingRules.append(rule);
|
|
|
|
functionFormat.setFontItalic(true);
|
|
functionFormat.setForeground(Qt::blue);
|
|
rule.pattern = QRegularExpression("\\b[A-Za-z0-9_]+(?=\\()");
|
|
rule.format = functionFormat;
|
|
highlightingRules.append(rule);
|
|
|
|
singleLineCommentFormat.setForeground(Qt::gray);
|
|
rule.pattern = QRegularExpression("//[^\n]*");
|
|
rule.format = singleLineCommentFormat;
|
|
highlightingRules.append(rule);
|
|
|
|
highlightingRulesWithSymbols = highlightingRules;
|
|
|
|
multiLineCommentFormat.setForeground(Qt::gray);
|
|
|
|
symbolFormat.setForeground(Qt::red);
|
|
symbolFormat.setBackground(QColor(220,220,255));
|
|
|
|
commentStartExpression = QRegularExpression("/\\*");
|
|
commentEndExpression = QRegularExpression("\\*/");
|
|
}
|
|
|
|
void Highlighter::setSymbols(const QStringList &symbols)
|
|
{
|
|
highlightingRulesWithSymbols = highlightingRules;
|
|
foreach (const QString &sym, symbols) {
|
|
HighlightingRule rule;
|
|
rule.pattern = QRegularExpression("\\b" + sym + "\\b");
|
|
rule.format = symbolFormat;
|
|
highlightingRulesWithSymbols.append(rule);
|
|
}
|
|
}
|
|
|
|
void Highlighter::highlightBlock(const QString &text)
|
|
{
|
|
foreach (const HighlightingRule &rule, highlightingRulesWithSymbols) {
|
|
QRegularExpressionMatchIterator matchIterator = rule.pattern.globalMatch(text);
|
|
while (matchIterator.hasNext()) {
|
|
QRegularExpressionMatch match = matchIterator.next();
|
|
setFormat(match.capturedStart(), match.capturedLength(), rule.format);
|
|
}
|
|
}
|
|
|
|
setCurrentBlockState(0);
|
|
|
|
int startIndex = 0;
|
|
if (previousBlockState() != 1)
|
|
startIndex = text.indexOf(commentStartExpression);
|
|
|
|
while (startIndex >= 0) {
|
|
QRegularExpressionMatch match = commentEndExpression.match(text, startIndex);
|
|
int endIndex = match.capturedStart();
|
|
int commentLength = 0;
|
|
if (endIndex == -1) {
|
|
setCurrentBlockState(1);
|
|
commentLength = text.length() - startIndex;
|
|
} else {
|
|
commentLength = endIndex - startIndex
|
|
+ match.capturedLength();
|
|
}
|
|
setFormat(startIndex, commentLength, multiLineCommentFormat);
|
|
startIndex = text.indexOf(commentStartExpression, startIndex + commentLength);
|
|
}
|
|
}
|
|
|
|
|
|
CodeEditor::CodeEditor(QWidget *parent) : QPlainTextEdit(parent)
|
|
{
|
|
lineNumberArea = new LineNumberArea(this);
|
|
highlighter = new Highlighter(this->document());
|
|
mErrorPosition = -1;
|
|
|
|
setFont(QFontDatabase::systemFont(QFontDatabase::FixedFont));
|
|
|
|
connect(this, SIGNAL(blockCountChanged(int)), this, SLOT(updateLineNumberAreaWidth(int)));
|
|
connect(this, SIGNAL(updateRequest(QRect,int)), this, SLOT(updateLineNumberArea(QRect,int)));
|
|
connect(this, SIGNAL(cursorPositionChanged()), this, SLOT(highlightCurrentLine()));
|
|
|
|
updateLineNumberAreaWidth(0);
|
|
}
|
|
|
|
static int getPos(const QString &fileData, int lineNumber)
|
|
{
|
|
if (lineNumber <= 1)
|
|
return 0;
|
|
for (int pos = 0, line = 1; pos < fileData.size(); ++pos) {
|
|
if (fileData[pos] != '\n')
|
|
continue;
|
|
++line;
|
|
if (line >= lineNumber)
|
|
return pos + 1;
|
|
}
|
|
return fileData.size();
|
|
}
|
|
|
|
void CodeEditor::setError(const QString &code, int errorLine, const QStringList &symbols)
|
|
{
|
|
highlighter->setSymbols(symbols);
|
|
|
|
setPlainText(code);
|
|
|
|
mErrorPosition = getPos(code, errorLine);
|
|
QTextCursor tc = textCursor();
|
|
tc.setPosition(mErrorPosition);
|
|
setTextCursor(tc);
|
|
centerCursor();
|
|
}
|
|
|
|
int CodeEditor::lineNumberAreaWidth()
|
|
{
|
|
int digits = 1;
|
|
int max = qMax(1, blockCount());
|
|
while (max >= 10) {
|
|
max /= 10;
|
|
++digits;
|
|
}
|
|
|
|
int space = 3 + fontMetrics().width(QLatin1Char('9')) * digits;
|
|
return space;
|
|
}
|
|
|
|
void CodeEditor::updateLineNumberAreaWidth(int /* newBlockCount */)
|
|
{
|
|
setViewportMargins(lineNumberAreaWidth(), 0, 0, 0);
|
|
}
|
|
|
|
void CodeEditor::updateLineNumberArea(const QRect &rect, int dy)
|
|
{
|
|
if (dy)
|
|
lineNumberArea->scroll(0, dy);
|
|
else
|
|
lineNumberArea->update(0, rect.y(), lineNumberArea->width(), rect.height());
|
|
|
|
if (rect.contains(viewport()->rect()))
|
|
updateLineNumberAreaWidth(0);
|
|
}
|
|
|
|
void CodeEditor::resizeEvent(QResizeEvent *event)
|
|
{
|
|
QPlainTextEdit::resizeEvent(event);
|
|
QRect cr = contentsRect();
|
|
lineNumberArea->setGeometry(QRect(cr.left(), cr.top(), lineNumberAreaWidth(), cr.height()));
|
|
}
|
|
|
|
void CodeEditor::highlightCurrentLine()
|
|
{
|
|
QTextCursor tc = textCursor();
|
|
tc.setPosition(mErrorPosition);
|
|
setTextCursor(tc);
|
|
|
|
QList<QTextEdit::ExtraSelection> extraSelections;
|
|
|
|
QTextEdit::ExtraSelection selection;
|
|
|
|
QColor lineColor = QColor(255,220,220);
|
|
|
|
selection.format.setBackground(lineColor);
|
|
selection.format.setProperty(QTextFormat::FullWidthSelection, true);
|
|
selection.cursor = textCursor();
|
|
selection.cursor.clearSelection();
|
|
extraSelections.append(selection);
|
|
|
|
setExtraSelections(extraSelections);
|
|
}
|
|
|
|
void CodeEditor::lineNumberAreaPaintEvent(QPaintEvent *event)
|
|
{
|
|
QPainter painter(lineNumberArea);
|
|
painter.fillRect(event->rect(), QColor(240,240,240));
|
|
|
|
QTextBlock block = firstVisibleBlock();
|
|
int blockNumber = block.blockNumber();
|
|
int top = (int) blockBoundingGeometry(block).translated(contentOffset()).top();
|
|
int bottom = top + (int) blockBoundingRect(block).height();
|
|
|
|
while (block.isValid() && top <= event->rect().bottom()) {
|
|
if (block.isVisible() && bottom >= event->rect().top()) {
|
|
QString number = QString::number(blockNumber + 1);
|
|
painter.setPen(Qt::black);
|
|
painter.drawText(0, top, lineNumberArea->width(), fontMetrics().height(),
|
|
Qt::AlignRight, number);
|
|
}
|
|
|
|
block = block.next();
|
|
top = bottom;
|
|
bottom = top + (int) blockBoundingRect(block).height();
|
|
++blockNumber;
|
|
}
|
|
}
|