diff options
author | Petar Perisin <petar.perisin@gmail.com> | 2013-09-03 06:47:17 +0300 |
---|---|---|
committer | hjk <hjk121@nokiamail.com> | 2013-09-09 13:05:16 +0200 |
commit | e8f5502b78d516b7b4bc47a53d21759416e318b2 (patch) | |
tree | 00280bee6c4e717f364de2e6f075e52cc3fdaede /src/libs/utils | |
parent | 7e29974e4ee3e1402104e33333baf4f58e269d77 (diff) | |
download | qt-creator-e8f5502b78d516b7b4bc47a53d21759416e318b2.tar.gz |
Added color support of ANSI escape codes
in compile and application output pane. Only simple font and background
controles added.
Task-number: QTCREATORBUG-9592
Task-number: QTCREATORBUG-5956
Change-Id: Ida010ed17d34bb73ae1364a77073ff435a03a060
Reviewed-by: André Hartmann <aha_1980@gmx.de>
Reviewed-by: Leena Miettinen <riitta-leena.miettinen@digia.com>
Reviewed-by: Orgad Shaneh <orgads@gmail.com>
Reviewed-by: hjk <hjk121@nokiamail.com>
Diffstat (limited to 'src/libs/utils')
-rw-r--r-- | src/libs/utils/ansiescapecodehandler.cpp | 200 | ||||
-rw-r--r-- | src/libs/utils/ansiescapecodehandler.h | 72 | ||||
-rw-r--r-- | src/libs/utils/outputformatter.cpp | 13 | ||||
-rw-r--r-- | src/libs/utils/outputformatter.h | 4 | ||||
-rw-r--r-- | src/libs/utils/utils-lib.pri | 6 | ||||
-rw-r--r-- | src/libs/utils/utils.qbs | 2 |
6 files changed, 294 insertions, 3 deletions
diff --git a/src/libs/utils/ansiescapecodehandler.cpp b/src/libs/utils/ansiescapecodehandler.cpp new file mode 100644 index 0000000000..66b5f2c186 --- /dev/null +++ b/src/libs/utils/ansiescapecodehandler.cpp @@ -0,0 +1,200 @@ +/**************************************************************************** +** +** Copyright (C) 2013 Petar Perisin <petar.perisin@gmail.com> +** Contact: http://www.qt-project.org/legal +** +** This file is part of Qt Creator. +** +** Commercial License Usage +** Licensees holding valid commercial Qt licenses may use this file in +** accordance with the commercial license agreement provided with the +** Software or, alternatively, in accordance with the terms contained in +** a written agreement between you and Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** 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. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +****************************************************************************/ + +#include "ansiescapecodehandler.h" +#include <utils/qtcassert.h> + +namespace Utils { + +/*! + \class Utils::AnsiEscapeCodeHandler + + \brief The AnsiEscapeCodeHandler class parses text and extracts ANSI escape codes from it. + + In order to preserve color information across text segments, an instance of this class + must be stored for the lifetime of a stream. + Also, one instance of this class should not handle multiple streams (at least not + at the same time). + + Its main method is parseText(), which accepts text and default QTextCharFormat. + This method is designed to parse text and split colored text to smaller strings, + with their appropriate formatting information set inside QTextCharFormat. + + Usage: + \list + \li Create new instance of AnsiEscapeCodeHandler for a stream. + \li To add new text, call parseText() with the text and a default QTextCharFormat. + The result of this method is a list of strings with formats set in appropriate + QTextCharFormat. + \endlist +*/ + +AnsiEscapeCodeHandler::AnsiEscapeCodeHandler() : + m_previousFormatClosed(true) +{ +} + +static QColor ansiColor(uint code) +{ + QTC_ASSERT(code < 8, return QColor()); + + const int red = code & 1 ? 170 : 0; + const int green = code & 2 ? 170 : 0; + const int blue = code & 4 ? 170 : 0; + return QColor(red, green, blue); +} + +QList<StringFormatPair> AnsiEscapeCodeHandler::parseText(const QString &text, + const QTextCharFormat &defaultFormat) +{ + QList<StringFormatPair> outputData; + + QTextCharFormat charFormat = m_previousFormatClosed ? defaultFormat : m_previousFormat; + + const QString escape = QLatin1String("\e["); + if (!text.contains(escape)) { + outputData << StringFormatPair(text, charFormat); + return outputData; + } else if (!text.startsWith(escape)) { + outputData << StringFormatPair(text.left(text.indexOf(escape)), charFormat); + } + + const QChar semicolon = QLatin1Char(';'); + const QChar colorTerminator = QLatin1Char('m'); + // strippedText always starts with "\e[" + QString strippedText = text.mid(text.indexOf(escape)); + while (!strippedText.isEmpty()) { + while (strippedText.startsWith(escape)) { + strippedText.remove(0, 2); + + // get the number + QString strNumber; + QStringList numbers; + while (strippedText.at(0).isDigit() || strippedText.at(0) == semicolon) { + if (strippedText.at(0).isDigit()) { + strNumber += strippedText.at(0); + } else { + numbers << strNumber; + strNumber.clear(); + } + strippedText.remove(0, 1); + } + if (!strNumber.isEmpty()) + numbers << strNumber; + + // remove terminating char + if (!strippedText.startsWith(colorTerminator)) { + strippedText.remove(0, 1); + continue; + } + strippedText.remove(0, 1); + + for (int i = 0; i < numbers.size(); ++i) { + const int code = numbers.at(i).toInt(); + + if (code >= TextColorStart && code <= TextColorEnd) { + charFormat.setForeground(ansiColor(code - TextColorStart)); + setFormatScope(charFormat); + } else if (code >= BackgroundColorStart && code <= BackgroundColorEnd) { + charFormat.setBackground(ansiColor(code - BackgroundColorStart)); + setFormatScope(charFormat); + } else { + switch (code) { + case ResetFormat: + charFormat = defaultFormat; + endFormatScope(); + break; + case BoldText: + charFormat.setFontWeight(QFont::Bold); + setFormatScope(charFormat); + break; + case DefaultTextColor: + charFormat.setForeground(defaultFormat.foreground()); + setFormatScope(charFormat); + break; + case DefaultBackgroundColor: + charFormat.setBackground(defaultFormat.background()); + setFormatScope(charFormat); + break; + case RgbTextColor: + case RgbBackgroundColor: + if (++i >= numbers.size()) + break; + if (numbers.at(i).toInt() == 2) { + // RGB set with format: 38;2;<r>;<g>;<b> + if ((i + 3) < numbers.size()) { + (code == RgbTextColor) ? + charFormat.setForeground(QColor(numbers.at(i + 1).toInt(), + numbers.at(i + 2).toInt(), + numbers.at(i + 3).toInt())) : + charFormat.setBackground(QColor(numbers.at(i + 1).toInt(), + numbers.at(i + 2).toInt(), + numbers.at(i + 3).toInt())); + setFormatScope(charFormat); + } + i += 3; + } else if (numbers.at(i).toInt() == 5) { + // rgb set with format: 38;5;<i> + // unsupported because of unclear documentation, so we just skip <i> + ++i; + } + break; + default: + break; + } + } + } + } + + if (strippedText.isEmpty()) + break; + int index = strippedText.indexOf(escape); + if (index > 0) { + outputData << StringFormatPair(strippedText.left(index), charFormat); + strippedText.remove(0, index); + } else if (index == -1) { + outputData << StringFormatPair(strippedText, charFormat); + break; + } + } + return outputData; +} + +void AnsiEscapeCodeHandler::endFormatScope() +{ + m_previousFormatClosed = true; +} + +void AnsiEscapeCodeHandler::setFormatScope(const QTextCharFormat &charFormat) +{ + m_previousFormat = charFormat; + m_previousFormatClosed = false; +} + +} // namespace Utils diff --git a/src/libs/utils/ansiescapecodehandler.h b/src/libs/utils/ansiescapecodehandler.h new file mode 100644 index 0000000000..d68490c58e --- /dev/null +++ b/src/libs/utils/ansiescapecodehandler.h @@ -0,0 +1,72 @@ +/**************************************************************************** +** +** Copyright (C) 2013 Petar Perisin <petar.perisin@gmail.com> +** Contact: http://www.qt-project.org/legal +** +** This file is part of Qt Creator. +** +** Commercial License Usage +** Licensees holding valid commercial Qt licenses may use this file in +** accordance with the commercial license agreement provided with the +** Software or, alternatively, in accordance with the terms contained in +** a written agreement between you and Digia. For licensing terms and +** conditions see http://qt.digia.com/licensing. For further information +** use the contact form at http://qt.digia.com/contact-us. +** +** GNU Lesser General Public License Usage +** 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. +** +** In addition, as a special exception, Digia gives you certain additional +** rights. These rights are described in the Digia Qt LGPL Exception +** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. +** +****************************************************************************/ + + +#ifndef UTILS_ANSIESCAPECODEHANDLER_H +#define UTILS_ANSIESCAPECODEHANDLER_H + +#include "utils_global.h" + +#include <QTextCharFormat> + +namespace Utils { + +typedef QPair<QString, QTextCharFormat> StringFormatPair; + +class QTCREATOR_UTILS_EXPORT AnsiEscapeCodeHandler +{ + +enum AnsiEscapeCodes { + ResetFormat = 0, + BoldText = 1, + TextColorStart = 30, + TextColorEnd = 37, + RgbTextColor = 38, + DefaultTextColor = 39, + BackgroundColorStart = 40, + BackgroundColorEnd = 47, + RgbBackgroundColor = 48, + DefaultBackgroundColor = 49 +}; + +public: + AnsiEscapeCodeHandler(); + QList<StringFormatPair> parseText(const QString &text, const QTextCharFormat &defaultFormat); + void endFormatScope(); + +private: + void setFormatScope(const QTextCharFormat &charFormat); + + bool m_previousFormatClosed; + QTextCharFormat m_previousFormat; +}; + +} // namespace Utils + +#endif // UTILS_ANSIESCAPECODEHANDLER_H diff --git a/src/libs/utils/outputformatter.cpp b/src/libs/utils/outputformatter.cpp index 81649e80be..d2f631d426 100644 --- a/src/libs/utils/outputformatter.cpp +++ b/src/libs/utils/outputformatter.cpp @@ -30,6 +30,7 @@ #include "outputformatter.h" #include <QPlainTextEdit> +#include <utils/ansiescapecodehandler.h> using namespace Utils; @@ -37,6 +38,7 @@ OutputFormatter::OutputFormatter() : QObject() , m_plainTextEdit(0) , m_formats(0) + , m_escapeCodeHandler(new AnsiEscapeCodeHandler) { } @@ -44,6 +46,7 @@ OutputFormatter::OutputFormatter() OutputFormatter::~OutputFormatter() { delete[] m_formats; + delete m_escapeCodeHandler; } QPlainTextEdit *OutputFormatter::plainTextEdit() const @@ -61,7 +64,9 @@ void OutputFormatter::appendMessage(const QString &text, OutputFormat format) { QTextCursor cursor(m_plainTextEdit->document()); cursor.movePosition(QTextCursor::End); - cursor.insertText(text, m_formats[format]); + + foreach (const StringFormatPair &pair, m_escapeCodeHandler->parseText(text, m_formats[format])) + cursor.insertText(pair.first, pair.second); } QTextCharFormat OutputFormatter::charFormat(OutputFormat format) const @@ -131,3 +136,9 @@ void OutputFormatter::setFont(const QFont &font) m_font = font; initFormats(); } + +void OutputFormatter::flush() +{ + if (m_escapeCodeHandler) + m_escapeCodeHandler->endFormatScope(); +} diff --git a/src/libs/utils/outputformatter.h b/src/libs/utils/outputformatter.h index 789e0da18d..733e7cd52a 100644 --- a/src/libs/utils/outputformatter.h +++ b/src/libs/utils/outputformatter.h @@ -44,6 +44,8 @@ QT_END_NAMESPACE namespace Utils { +class AnsiEscapeCodeHandler; + class QTCREATOR_UTILS_EXPORT OutputFormatter : public QObject { Q_OBJECT @@ -57,6 +59,7 @@ public: QFont font() const; void setFont(const QFont &font); + void flush(); virtual void appendMessage(const QString &text, OutputFormat format); virtual void handleLink(const QString &href); @@ -72,6 +75,7 @@ private: QPlainTextEdit *m_plainTextEdit; QTextCharFormat *m_formats; QFont m_font; + AnsiEscapeCodeHandler *m_escapeCodeHandler; }; } // namespace Utils diff --git a/src/libs/utils/utils-lib.pri b/src/libs/utils/utils-lib.pri index 15ef919407..3899a45c19 100644 --- a/src/libs/utils/utils-lib.pri +++ b/src/libs/utils/utils-lib.pri @@ -84,7 +84,8 @@ SOURCES += $$PWD/environment.cpp \ $$PWD/tooltip/tips.cpp \ $$PWD/tooltip/tipcontents.cpp \ $$PWD/unixutils.cpp \ - $$PWD/function.cpp + $$PWD/function.cpp \ + $$PWD/ansiescapecodehandler.cpp win32 { SOURCES += \ @@ -176,7 +177,8 @@ HEADERS += \ $$PWD/tooltip/effects.h \ $$PWD/unixutils.h \ $$PWD/qtcoverride.h \ - $$PWD/function.h + $$PWD/function.h \ + $$PWD/ansiescapecodehandler.h FORMS += $$PWD/filewizardpage.ui \ $$PWD/projectintropage.ui \ diff --git a/src/libs/utils/utils.qbs b/src/libs/utils/utils.qbs index afc229096f..2a55db5bbc 100644 --- a/src/libs/utils/utils.qbs +++ b/src/libs/utils/utils.qbs @@ -25,6 +25,8 @@ QtcLibrary { files: [ "annotateditemdelegate.cpp", "annotateditemdelegate.h", + "ansiescapecodehandler.cpp", + "ansiescapecodehandler.h", "appmainwindow.cpp", "appmainwindow.h", "basetreeview.cpp", |