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/ansiescapecodehandler.cpp | |
parent | 7e29974e4ee3e1402104e33333baf4f58e269d77 (diff) |
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/ansiescapecodehandler.cpp')
-rw-r--r-- | src/libs/utils/ansiescapecodehandler.cpp | 200 |
1 files changed, 200 insertions, 0 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 |