2022-08-19 15:59:36 +02:00
|
|
|
// Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0+ OR GPL-3.0 WITH Qt-GPL-exception-1.0
|
2008-12-02 16:19:05 +01:00
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
#include "diffandloghighlighter.h"
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2014-09-26 09:14:03 +02:00
|
|
|
#include <texteditor/textdocumentlayout.h>
|
2010-11-21 10:15:12 +01:00
|
|
|
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2012-02-15 10:42:41 +01:00
|
|
|
#include <QDebug>
|
2020-02-19 23:26:24 +02:00
|
|
|
#include <QRegularExpression>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2011-03-28 14:19:17 +02:00
|
|
|
/*!
|
2015-03-25 10:31:55 +02:00
|
|
|
\class VcsBase::DiffAndLogHighlighter
|
2011-03-28 14:19:17 +02:00
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
\brief The DiffAndLogHighlighter class provides a highlighter for diffs and log editors.
|
2011-03-28 14:19:17 +02:00
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
Diff is parametrizable by the file indicator, which is for example '^====' in case of p4:
|
2011-03-28 14:19:17 +02:00
|
|
|
\code
|
|
|
|
|
==== //depot/research/main/qdynamicmainwindow3/qdynamicdockwidgetlayout_p.h#34 (text) ====
|
|
|
|
|
\endcode
|
|
|
|
|
|
|
|
|
|
Or '--- a/|'+++ b/' in case of git:
|
|
|
|
|
\code
|
|
|
|
|
diff --git a/src/plugins/plugins.pro b/src/plugins/plugins.pro
|
|
|
|
|
index 9401ee7..ef35c3b 100644
|
|
|
|
|
--- a/src/plugins/plugins.pro
|
|
|
|
|
+++ b/src/plugins/plugins.pro
|
|
|
|
|
@@ -10,6 +10,7 @@ SUBDIRS = plugin_coreplugin
|
|
|
|
|
\endcode
|
|
|
|
|
|
2015-03-25 10:30:24 +02:00
|
|
|
Log is parametrizable by change indicator. For example '^commit ([0-9a-f]{8})[0-9a-f]{32}'
|
|
|
|
|
in Git:
|
|
|
|
|
\code
|
|
|
|
|
commit a3398841a24b24c73b47759c4bffdc8b78a34936 (HEAD, master)
|
|
|
|
|
\code
|
|
|
|
|
|
2011-03-28 14:19:17 +02:00
|
|
|
Also highlights trailing blanks.
|
|
|
|
|
*/
|
|
|
|
|
|
2010-11-21 10:15:12 +01:00
|
|
|
static const int BASE_LEVEL = 0;
|
|
|
|
|
static const int FILE_LEVEL = 1;
|
|
|
|
|
static const int LOCATION_LEVEL = 2;
|
|
|
|
|
|
2012-01-07 12:31:48 +01:00
|
|
|
namespace VcsBase {
|
2010-11-21 10:15:12 +01:00
|
|
|
namespace Internal {
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2010-11-21 10:15:12 +01:00
|
|
|
enum FoldingState {
|
|
|
|
|
StartOfFile,
|
|
|
|
|
Header,
|
|
|
|
|
File,
|
|
|
|
|
Location
|
|
|
|
|
};
|
|
|
|
|
|
2014-06-16 21:34:13 +04:00
|
|
|
} // namespace Internal;
|
2013-08-13 12:57:31 +02:00
|
|
|
|
|
|
|
|
static inline QTextCharFormat invertedColorFormat(const QTextCharFormat &in)
|
|
|
|
|
{
|
|
|
|
|
QTextCharFormat rc = in;
|
|
|
|
|
rc.setForeground(in.background());
|
|
|
|
|
rc.setBackground(in.foreground());
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
// --- DiffAndLogHighlighterPrivate
|
|
|
|
|
class DiffAndLogHighlighterPrivate
|
2012-01-07 03:35:54 +01:00
|
|
|
{
|
2010-11-21 10:15:12 +01:00
|
|
|
public:
|
2020-02-19 23:26:24 +02:00
|
|
|
DiffAndLogHighlighterPrivate(DiffAndLogHighlighter *q_,
|
|
|
|
|
const QRegularExpression &filePattern,
|
|
|
|
|
const QRegularExpression &changePattern) :
|
2016-02-03 12:15:21 +01:00
|
|
|
q(q_),
|
|
|
|
|
m_filePattern(filePattern),
|
|
|
|
|
m_changePattern(changePattern),
|
|
|
|
|
m_locationIndicator(QLatin1String("@@")),
|
|
|
|
|
m_diffInIndicator(QLatin1Char('+')),
|
|
|
|
|
m_diffOutIndicator(QLatin1Char('-')),
|
|
|
|
|
m_foldingState(Internal::StartOfFile)
|
|
|
|
|
{
|
|
|
|
|
QTC_CHECK(filePattern.isValid());
|
|
|
|
|
}
|
2012-01-07 03:35:54 +01:00
|
|
|
|
2017-05-05 20:34:29 +02:00
|
|
|
TextEditor::TextStyle analyzeLine(const QString &block) const;
|
2013-08-13 12:57:31 +02:00
|
|
|
void updateOtherFormats();
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2016-02-03 12:15:21 +01:00
|
|
|
DiffAndLogHighlighter *const q;
|
|
|
|
|
|
2020-02-19 23:26:24 +02:00
|
|
|
const QRegularExpression m_filePattern;
|
|
|
|
|
const QRegularExpression m_changePattern;
|
2008-12-02 12:01:29 +01:00
|
|
|
const QString m_locationIndicator;
|
|
|
|
|
const QChar m_diffInIndicator;
|
|
|
|
|
const QChar m_diffOutIndicator;
|
2009-12-04 14:23:34 +01:00
|
|
|
QTextCharFormat m_addedTrailingWhiteSpaceFormat;
|
2010-11-21 10:15:12 +01:00
|
|
|
|
2013-08-13 12:57:31 +02:00
|
|
|
Internal::FoldingState m_foldingState;
|
2020-05-16 17:37:33 +02:00
|
|
|
bool m_enabled = true;
|
2008-12-02 12:01:29 +01:00
|
|
|
};
|
|
|
|
|
|
2017-05-05 20:34:29 +02:00
|
|
|
TextEditor::TextStyle DiffAndLogHighlighterPrivate::analyzeLine(const QString &text) const
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
// Do not match on git "--- a/" as a deleted line, check
|
|
|
|
|
// file first
|
2020-02-19 23:26:24 +02:00
|
|
|
if (m_filePattern.match(text).capturedStart() == 0)
|
2017-05-05 20:34:29 +02:00
|
|
|
return TextEditor::C_DIFF_FILE;
|
2020-02-19 23:26:24 +02:00
|
|
|
if (m_changePattern.match(text).capturedStart() == 0)
|
2017-05-05 20:34:29 +02:00
|
|
|
return TextEditor::C_LOG_CHANGE_LINE;
|
2008-12-02 12:01:29 +01:00
|
|
|
if (text.startsWith(m_diffInIndicator))
|
2017-05-05 20:34:29 +02:00
|
|
|
return TextEditor::C_ADDED_LINE;
|
2008-12-02 12:01:29 +01:00
|
|
|
if (text.startsWith(m_diffOutIndicator))
|
2017-05-05 20:34:29 +02:00
|
|
|
return TextEditor::C_REMOVED_LINE;
|
2008-12-02 12:01:29 +01:00
|
|
|
if (text.startsWith(m_locationIndicator))
|
2017-05-05 20:34:29 +02:00
|
|
|
return TextEditor::C_DIFF_LOCATION;
|
|
|
|
|
return TextEditor::C_TEXT;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
void DiffAndLogHighlighterPrivate::updateOtherFormats()
|
2013-08-13 12:57:31 +02:00
|
|
|
{
|
|
|
|
|
m_addedTrailingWhiteSpaceFormat =
|
2017-05-05 20:34:29 +02:00
|
|
|
invertedColorFormat(q->formatForCategory(TextEditor::C_ADDED_LINE));
|
2013-08-13 12:57:31 +02:00
|
|
|
|
|
|
|
|
}
|
2010-11-21 10:15:12 +01:00
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
// --- DiffAndLogHighlighter
|
2020-02-19 23:26:24 +02:00
|
|
|
DiffAndLogHighlighter::DiffAndLogHighlighter(const QRegularExpression &filePattern,
|
|
|
|
|
const QRegularExpression &changePattern) :
|
2018-09-20 01:32:36 +03:00
|
|
|
TextEditor::SyntaxHighlighter(static_cast<QTextDocument *>(nullptr)),
|
2016-02-03 12:15:21 +01:00
|
|
|
d(new DiffAndLogHighlighterPrivate(this, filePattern, changePattern))
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2017-05-05 20:34:29 +02:00
|
|
|
setDefaultTextFormatCategories();
|
2013-08-13 12:57:31 +02:00
|
|
|
d->updateOtherFormats();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
DiffAndLogHighlighter::~DiffAndLogHighlighter()
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2016-02-03 12:15:21 +01:00
|
|
|
delete d;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2009-12-04 14:23:34 +01:00
|
|
|
// Check trailing spaces
|
|
|
|
|
static inline int trimmedLength(const QString &in)
|
|
|
|
|
{
|
|
|
|
|
for (int pos = in.length() - 1; pos >= 0; pos--)
|
|
|
|
|
if (!in.at(pos).isSpace())
|
|
|
|
|
return pos + 1;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 10:56:09 +02:00
|
|
|
/*
|
|
|
|
|
* This sets the folding indent:
|
|
|
|
|
* 0 for the first line of the diff header.
|
|
|
|
|
* 1 for all the following lines of the diff header and all @@ lines.
|
|
|
|
|
* 2 for everything else
|
|
|
|
|
*/
|
2015-03-25 10:31:55 +02:00
|
|
|
void DiffAndLogHighlighter::highlightBlock(const QString &text)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
if (text.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
2009-12-04 14:23:34 +01:00
|
|
|
const int length = text.length();
|
2017-05-05 20:34:29 +02:00
|
|
|
const TextEditor::TextStyle format = d->analyzeLine(text);
|
|
|
|
|
|
2020-05-16 17:37:33 +02:00
|
|
|
if (d->m_enabled) {
|
|
|
|
|
if (format == TextEditor::C_ADDED_LINE) {
|
2009-12-04 14:23:34 +01:00
|
|
|
// Mark trailing whitespace.
|
|
|
|
|
const int trimmedLen = trimmedLength(text);
|
2017-05-08 10:52:28 +02:00
|
|
|
setFormatWithSpaces(text, 0, trimmedLen, formatForCategory(format));
|
2009-12-04 14:23:34 +01:00
|
|
|
if (trimmedLen != length)
|
2011-09-15 13:42:38 +02:00
|
|
|
setFormat(trimmedLen, length - trimmedLen, d->m_addedTrailingWhiteSpaceFormat);
|
2020-05-16 17:37:33 +02:00
|
|
|
} else if (format != TextEditor::C_TEXT) {
|
|
|
|
|
setFormatWithSpaces(text, 0, length, formatForCategory(format));
|
|
|
|
|
} else {
|
|
|
|
|
formatSpaces(text);
|
|
|
|
|
}
|
2009-12-04 14:23:34 +01:00
|
|
|
}
|
2010-11-21 10:15:12 +01:00
|
|
|
|
|
|
|
|
// codefolding:
|
|
|
|
|
TextEditor::TextBlockUserData *data =
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::userData(currentBlock());
|
2011-07-21 09:12:10 +02:00
|
|
|
QTC_ASSERT(data, return; );
|
2019-12-04 08:19:53 +01:00
|
|
|
if (!TextEditor::TextDocumentLayout::textUserData(currentBlock().previous()))
|
2011-09-15 13:42:38 +02:00
|
|
|
d->m_foldingState = Internal::StartOfFile;
|
2010-11-21 10:15:12 +01:00
|
|
|
|
2011-09-15 13:42:38 +02:00
|
|
|
switch (d->m_foldingState) {
|
2010-11-21 10:15:12 +01:00
|
|
|
case Internal::StartOfFile:
|
|
|
|
|
case Internal::Header:
|
2017-05-05 20:34:29 +02:00
|
|
|
if (format == TextEditor::C_DIFF_FILE) {
|
2011-09-15 13:42:38 +02:00
|
|
|
d->m_foldingState = Internal::File;
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), BASE_LEVEL);
|
2017-05-05 20:34:29 +02:00
|
|
|
} else if (format == TextEditor::C_DIFF_LOCATION) {
|
2011-09-15 13:42:38 +02:00
|
|
|
d->m_foldingState = Internal::Location;
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
2017-05-05 20:34:29 +02:00
|
|
|
} else {
|
2011-09-15 13:42:38 +02:00
|
|
|
d->m_foldingState = Internal::Header;
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), BASE_LEVEL);
|
2010-11-21 10:15:12 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case Internal::File:
|
2017-05-05 20:34:29 +02:00
|
|
|
if (format == TextEditor::C_DIFF_FILE) {
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
2017-05-05 20:34:29 +02:00
|
|
|
} else if (format == TextEditor::C_DIFF_LOCATION) {
|
2011-09-15 13:42:38 +02:00
|
|
|
d->m_foldingState = Internal::Location;
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
2017-05-05 20:34:29 +02:00
|
|
|
} else {
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
2010-11-21 10:15:12 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case Internal::Location:
|
2017-05-05 20:34:29 +02:00
|
|
|
if (format == TextEditor::C_DIFF_FILE) {
|
2011-09-15 13:42:38 +02:00
|
|
|
d->m_foldingState = Internal::File;
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), BASE_LEVEL);
|
2017-05-05 20:34:29 +02:00
|
|
|
} else if (format == TextEditor::C_DIFF_LOCATION) {
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
2017-05-05 20:34:29 +02:00
|
|
|
} else {
|
2014-09-26 09:14:03 +02:00
|
|
|
TextEditor::TextDocumentLayout::setFoldingIndent(currentBlock(), LOCATION_LEVEL);
|
2010-11-21 10:15:12 +01:00
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2009-12-04 14:23:34 +01:00
|
|
|
}
|
|
|
|
|
|
2015-03-25 10:31:55 +02:00
|
|
|
void DiffAndLogHighlighter::setFontSettings(const TextEditor::FontSettings &fontSettings)
|
2009-12-04 14:23:34 +01:00
|
|
|
{
|
2013-08-13 12:57:31 +02:00
|
|
|
SyntaxHighlighter::setFontSettings(fontSettings);
|
|
|
|
|
d->updateOtherFormats();
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
2020-05-16 17:37:33 +02:00
|
|
|
void DiffAndLogHighlighter::setEnabled(bool e)
|
|
|
|
|
{
|
|
|
|
|
d->m_enabled = e;
|
|
|
|
|
}
|
|
|
|
|
|
2012-01-07 12:31:48 +01:00
|
|
|
} // namespace VcsBase
|