2009-02-25 09:15:00 +01:00
|
|
|
/**************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
|
|
|
|
** This file is part of Qt Creator
|
|
|
|
|
**
|
2010-03-05 11:25:49 +01:00
|
|
|
** Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-06-17 00:01:27 +10:00
|
|
|
** Contact: Nokia Corporation (qt-info@nokia.com)
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** No Commercial Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** This file contains pre-release code and may not be distributed.
|
|
|
|
|
** You may use this file in accordance with the terms and conditions
|
|
|
|
|
** contained in the Technology Preview License Agreement accompanying
|
|
|
|
|
** this package.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** GNU Lesser General Public License Usage
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
** Alternatively, this file may be used under the terms of the GNU Lesser
|
|
|
|
|
** General Public License version 2.1 as published by the Free Software
|
|
|
|
|
** Foundation and appearing in the file LICENSE.LGPL included in the
|
|
|
|
|
** packaging of this file. Please review the following information to
|
|
|
|
|
** ensure the GNU Lesser General Public License version 2.1 requirements
|
|
|
|
|
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2010-12-17 16:01:08 +01:00
|
|
|
** In addition, as a special exception, Nokia gives you certain additional
|
|
|
|
|
** rights. These rights are described in the Nokia Qt LGPL Exception
|
|
|
|
|
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
|
|
|
|
**
|
|
|
|
|
** If you have questions regarding the use of this file, please contact
|
|
|
|
|
** Nokia at qt-info@nokia.com.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2009-02-25 09:15:00 +01:00
|
|
|
**************************************************************************/
|
2008-12-02 16:19:05 +01:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include "diffhighlighter.h"
|
|
|
|
|
|
2010-11-21 10:15:12 +01:00
|
|
|
#include <texteditor/basetextdocumentlayout.h>
|
|
|
|
|
|
2008-12-09 15:25:01 +01:00
|
|
|
#include <utils/qtcassert.h>
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
#include <QtCore/QDebug>
|
|
|
|
|
#include <QtCore/QtAlgorithms>
|
|
|
|
|
#include <QtCore/QRegExp>
|
2009-12-04 14:23:34 +01:00
|
|
|
#include <QtGui/QBrush>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
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;
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
namespace VCSBase {
|
2010-11-21 10:15:12 +01:00
|
|
|
namespace Internal {
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
// Formats used by DiffHighlighter
|
|
|
|
|
enum DiffFormats {
|
|
|
|
|
DiffTextFormat,
|
|
|
|
|
DiffInFormat,
|
|
|
|
|
DiffOutFormat,
|
|
|
|
|
DiffFileFormat,
|
|
|
|
|
DiffLocationFormat,
|
|
|
|
|
NumDiffFormats
|
|
|
|
|
};
|
|
|
|
|
|
2010-11-21 10:15:12 +01:00
|
|
|
enum FoldingState {
|
|
|
|
|
StartOfFile,
|
|
|
|
|
Header,
|
|
|
|
|
File,
|
|
|
|
|
Location
|
|
|
|
|
};
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
// --- DiffHighlighterPrivate
|
2010-11-21 10:15:12 +01:00
|
|
|
class DiffHighlighterPrivate {
|
|
|
|
|
public:
|
2008-12-02 12:01:29 +01:00
|
|
|
DiffHighlighterPrivate(const QRegExp &filePattern);
|
|
|
|
|
inline DiffFormats analyzeLine(const QString &block) const;
|
|
|
|
|
|
|
|
|
|
const QRegExp m_filePattern;
|
|
|
|
|
const QString m_locationIndicator;
|
|
|
|
|
const QChar m_diffInIndicator;
|
|
|
|
|
const QChar m_diffOutIndicator;
|
|
|
|
|
QTextCharFormat m_formats[NumDiffFormats];
|
2009-12-04 14:23:34 +01:00
|
|
|
QTextCharFormat m_addedTrailingWhiteSpaceFormat;
|
2010-11-21 10:15:12 +01:00
|
|
|
|
|
|
|
|
FoldingState m_foldingState;
|
2008-12-02 12:01:29 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
DiffHighlighterPrivate::DiffHighlighterPrivate(const QRegExp &filePattern) :
|
|
|
|
|
m_filePattern(filePattern),
|
|
|
|
|
m_locationIndicator(QLatin1String("@@")),
|
|
|
|
|
m_diffInIndicator(QLatin1Char('+')),
|
2010-11-21 10:15:12 +01:00
|
|
|
m_diffOutIndicator(QLatin1Char('-')),
|
|
|
|
|
m_foldingState(StartOfFile)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
2008-12-09 15:25:01 +01:00
|
|
|
QTC_ASSERT(filePattern.isValid(), /**/);
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DiffFormats DiffHighlighterPrivate::analyzeLine(const QString &text) const
|
|
|
|
|
{
|
|
|
|
|
// Do not match on git "--- a/" as a deleted line, check
|
|
|
|
|
// file first
|
|
|
|
|
if (m_filePattern.exactMatch(text))
|
|
|
|
|
return DiffFileFormat;
|
|
|
|
|
if (text.startsWith(m_diffInIndicator))
|
|
|
|
|
return DiffInFormat;
|
|
|
|
|
if (text.startsWith(m_diffOutIndicator))
|
|
|
|
|
return DiffOutFormat;
|
|
|
|
|
if (text.startsWith(m_locationIndicator))
|
|
|
|
|
return DiffLocationFormat;
|
|
|
|
|
return DiffTextFormat;
|
|
|
|
|
}
|
|
|
|
|
|
2010-11-21 10:15:12 +01:00
|
|
|
} // namespace Internal
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
// --- DiffHighlighter
|
|
|
|
|
DiffHighlighter::DiffHighlighter(const QRegExp &filePattern,
|
|
|
|
|
QTextDocument *document) :
|
2010-07-09 14:47:18 +02:00
|
|
|
TextEditor::SyntaxHighlighter(document),
|
2010-11-21 10:15:12 +01:00
|
|
|
m_d(new Internal::DiffHighlighterPrivate(filePattern))
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DiffHighlighter::~DiffHighlighter()
|
|
|
|
|
{
|
|
|
|
|
delete m_d;
|
|
|
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
void DiffHighlighter::highlightBlock(const QString &text)
|
|
|
|
|
{
|
|
|
|
|
if (text.isEmpty())
|
|
|
|
|
return;
|
|
|
|
|
|
2009-12-04 14:23:34 +01:00
|
|
|
const int length = text.length();
|
2010-11-21 10:15:12 +01:00
|
|
|
const Internal::DiffFormats format = m_d->analyzeLine(text);
|
2009-12-04 14:23:34 +01:00
|
|
|
switch (format) {
|
2010-11-21 10:15:12 +01:00
|
|
|
case Internal::DiffTextFormat:
|
2009-12-04 14:23:34 +01:00
|
|
|
break;
|
2010-11-21 10:15:12 +01:00
|
|
|
case Internal::DiffInFormat: {
|
2009-12-04 14:23:34 +01:00
|
|
|
// Mark trailing whitespace.
|
|
|
|
|
const int trimmedLen = trimmedLength(text);
|
|
|
|
|
setFormat(0, trimmedLen, m_d->m_formats[format]);
|
|
|
|
|
if (trimmedLen != length)
|
|
|
|
|
setFormat(trimmedLen, length - trimmedLen, m_d->m_addedTrailingWhiteSpaceFormat);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
setFormat(0, length, m_d->m_formats[format]);
|
|
|
|
|
break;
|
|
|
|
|
}
|
2010-11-21 10:15:12 +01:00
|
|
|
|
|
|
|
|
// codefolding:
|
|
|
|
|
TextEditor::TextBlockUserData *data =
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::userData(currentBlock());
|
|
|
|
|
Q_ASSERT(data);
|
|
|
|
|
if (!TextEditor::BaseTextDocumentLayout::testUserData(currentBlock().previous()))
|
|
|
|
|
m_d->m_foldingState = Internal::StartOfFile;
|
|
|
|
|
|
|
|
|
|
switch (m_d->m_foldingState) {
|
|
|
|
|
case Internal::StartOfFile:
|
|
|
|
|
case Internal::Header:
|
|
|
|
|
switch (format) {
|
|
|
|
|
case Internal::DiffFileFormat:
|
|
|
|
|
m_d->m_foldingState = Internal::File;
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), BASE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
case Internal::DiffLocationFormat:
|
|
|
|
|
m_d->m_foldingState = Internal::Location;
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
m_d->m_foldingState = Internal::Header;
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), BASE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case Internal::File:
|
|
|
|
|
switch (format) {
|
|
|
|
|
case Internal::DiffFileFormat:
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
case Internal::DiffLocationFormat:
|
|
|
|
|
m_d->m_foldingState = Internal::Location;
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case Internal::Location:
|
|
|
|
|
switch (format) {
|
|
|
|
|
case Internal::DiffFileFormat:
|
|
|
|
|
m_d->m_foldingState = Internal::File;
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), BASE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
case Internal::DiffLocationFormat:
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), FILE_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
TextEditor::BaseTextDocumentLayout::setFoldingIndent(currentBlock(), LOCATION_LEVEL);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2009-12-04 14:23:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static inline QTextCharFormat invertedColorFormat(const QTextCharFormat &in)
|
|
|
|
|
{
|
|
|
|
|
QTextCharFormat rc = in;
|
|
|
|
|
rc.setForeground(in.background());
|
|
|
|
|
rc.setBackground(in.foreground());
|
|
|
|
|
return rc;
|
2008-12-02 12:01:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void DiffHighlighter::setFormats(const QVector<QTextCharFormat> &s)
|
|
|
|
|
{
|
2010-11-21 10:15:12 +01:00
|
|
|
if (s.size() == Internal::NumDiffFormats) {
|
2008-12-02 12:01:29 +01:00
|
|
|
qCopy(s.constBegin(), s.constEnd(), m_d->m_formats);
|
2009-12-04 14:23:34 +01:00
|
|
|
// Display trailing blanks with colors swapped
|
2010-11-21 10:15:12 +01:00
|
|
|
m_d->m_addedTrailingWhiteSpaceFormat =
|
|
|
|
|
invertedColorFormat(m_d->m_formats[Internal::DiffInFormat]);
|
2008-12-02 12:01:29 +01:00
|
|
|
} else {
|
|
|
|
|
qWarning("%s: insufficient setting size: %d", Q_FUNC_INFO, s.size());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-10 11:02:12 +01:00
|
|
|
QRegExp DiffHighlighter::filePattern() const
|
|
|
|
|
{
|
|
|
|
|
return m_d->m_filePattern;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
} // namespace VCSBase
|