You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
331 lines
10 KiB
331 lines
10 KiB
/* |
|
This file is part of Konsole, an X terminal. |
|
|
|
Copyright 2006-2008 by Robert Knight <robertknight@gmail.com> |
|
|
|
This program is free software; you can redistribute it and/or modify |
|
it under the terms of the GNU Lesser General Public License as published by |
|
the Free Software Foundation; either version 2 of the License, or |
|
(at your option) any later version. |
|
|
|
This program is distributed in the hope that it will be useful, |
|
but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
GNU General Public License for more details. |
|
|
|
You should have received a copy of the GNU Lesser General Public License |
|
along with this program; if not, write to the Free Software |
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA |
|
02110-1301 USA. |
|
*/ |
|
|
|
// Own |
|
#include "TerminalCharacterDecoder.h" |
|
|
|
// Qt |
|
#include <QTextStream> |
|
|
|
// Konsole |
|
#include "ExtendedCharTable.h" |
|
#include "ColorScheme.h" |
|
#include "ColorSchemeManager.h" |
|
#include "Profile.h" |
|
|
|
using namespace Konsole; |
|
PlainTextDecoder::PlainTextDecoder() |
|
: _output(nullptr) |
|
, _includeLeadingWhitespace(true) |
|
, _includeTrailingWhitespace(true) |
|
, _recordLinePositions(false) |
|
, _linePositions(QList<int>()) |
|
{ |
|
} |
|
void PlainTextDecoder::setLeadingWhitespace(bool enable) |
|
{ |
|
_includeLeadingWhitespace = enable; |
|
} |
|
void PlainTextDecoder::setTrailingWhitespace(bool enable) |
|
{ |
|
_includeTrailingWhitespace = enable; |
|
} |
|
void PlainTextDecoder::begin(QTextStream* output) |
|
{ |
|
_output = output; |
|
if (!_linePositions.isEmpty()) { |
|
_linePositions.clear(); |
|
} |
|
} |
|
void PlainTextDecoder::end() |
|
{ |
|
_output = nullptr; |
|
} |
|
|
|
void PlainTextDecoder::setRecordLinePositions(bool record) |
|
{ |
|
_recordLinePositions = record; |
|
} |
|
QList<int> PlainTextDecoder::linePositions() const |
|
{ |
|
return _linePositions; |
|
} |
|
void PlainTextDecoder::decodeLine(const Character* const characters, int count, LineProperty /*properties*/ |
|
) |
|
{ |
|
Q_ASSERT(_output); |
|
|
|
if (_recordLinePositions && (_output->string() != nullptr)) { |
|
int pos = _output->string()->count(); |
|
_linePositions << pos; |
|
} |
|
|
|
//TODO should we ignore or respect the LINE_WRAPPED line property? |
|
|
|
//note: we build up a QString and send it to the text stream rather writing into the text |
|
//stream a character at a time because it is more efficient. |
|
//(since QTextStream always deals with QStrings internally anyway) |
|
QString plainText; |
|
plainText.reserve(count); |
|
|
|
// If we should remove leading whitespace find the first non-space character |
|
int start = 0; |
|
if (!_includeLeadingWhitespace) { |
|
for (start = 0; start < count; start++) { |
|
if (!characters[start].isSpace()) { |
|
break; |
|
} |
|
} |
|
} |
|
|
|
int outputCount = count - start; |
|
|
|
if (outputCount <= 0) { |
|
return; |
|
} |
|
|
|
// if inclusion of trailing whitespace is disabled then find the end of the |
|
// line |
|
if (!_includeTrailingWhitespace) { |
|
for (int i = count - 1 ; i >= start ; i--) { |
|
if (!characters[i].isSpace()) { |
|
break; |
|
} else { |
|
outputCount--; |
|
} |
|
} |
|
} |
|
|
|
// find out the last technically real character in the line |
|
int realCharacterGuard = -1; |
|
for (int i = count - 1 ; i >= start ; i--) { |
|
// FIXME: the special case of '\n' here is really ugly |
|
// Maybe the '\n' should be added after calling this method in |
|
// Screen::copyLineToStream() |
|
if (characters[i].isRealCharacter && characters[i].character != '\n') { |
|
realCharacterGuard = i; |
|
break; |
|
} |
|
} |
|
|
|
for (int i = start; i < outputCount;) { |
|
if ((characters[i].rendition & RE_EXTENDED_CHAR) != 0) { |
|
ushort extendedCharLength = 0; |
|
const uint* chars = ExtendedCharTable::instance.lookupExtendedChar(characters[i].character, extendedCharLength); |
|
if (chars != nullptr) { |
|
const QString s = QString::fromUcs4(chars, extendedCharLength); |
|
plainText.append(s); |
|
i += qMax(1, Character::stringWidth(s)); |
|
} else { |
|
++i; |
|
} |
|
} else { |
|
// All characters which appear before the last real character are |
|
// seen as real characters, even when they are technically marked as |
|
// non-real. |
|
// |
|
// This feels tricky, but otherwise leading "whitespaces" may be |
|
// lost in some situation. One typical example is copying the result |
|
// of `dialog --infobox "qwe" 10 10` . |
|
if (characters[i].isRealCharacter || i <= realCharacterGuard) { |
|
plainText.append(QString::fromUcs4(&characters[i].character, 1)); |
|
i += qMax(1, characters[i].width()); |
|
} else { |
|
++i; // should we 'break' directly here? |
|
} |
|
} |
|
} |
|
*_output << plainText; |
|
} |
|
|
|
HTMLDecoder::HTMLDecoder(const QExplicitlySharedDataPointer<Profile> &profile) : |
|
_output(nullptr) |
|
, _profile(profile) |
|
, _innerSpanOpen(false) |
|
, _lastRendition(DEFAULT_RENDITION) |
|
, _lastForeColor(CharacterColor()) |
|
, _lastBackColor(CharacterColor()) |
|
{ |
|
const ColorScheme *colorScheme = nullptr; |
|
|
|
if (profile) { |
|
colorScheme = ColorSchemeManager::instance()->findColorScheme(profile->colorScheme()); |
|
|
|
if (colorScheme == nullptr) { |
|
colorScheme = ColorSchemeManager::instance()->defaultColorScheme(); |
|
} |
|
|
|
} |
|
|
|
if (colorScheme != nullptr) { |
|
colorScheme->getColorTable(_colorTable); |
|
} else { |
|
for (int i = 0; i < TABLE_COLORS; i++) { |
|
_colorTable[i] = ColorScheme::defaultTable[i]; |
|
} |
|
} |
|
} |
|
|
|
void HTMLDecoder::begin(QTextStream* output) |
|
{ |
|
_output = output; |
|
|
|
|
|
if (_profile) { |
|
QString style; |
|
|
|
QFont font = _profile->font(); |
|
style.append(QStringLiteral("font-family:'%1',monospace;").arg(font.family())); |
|
|
|
// Prefer point size if set |
|
if (font.pointSizeF() > 0) { |
|
style.append(QStringLiteral("font-size:%1pt;").arg(font.pointSizeF())); |
|
} else { |
|
style.append(QStringLiteral("font-size:%1px;").arg(font.pixelSize())); |
|
} |
|
|
|
|
|
style.append(QStringLiteral("color:%1;").arg(_colorTable[DEFAULT_FORE_COLOR].name())); |
|
style.append(QStringLiteral("background-color:%1;").arg(_colorTable[DEFAULT_BACK_COLOR].name())); |
|
|
|
*output << QStringLiteral("<body style=\"%1\">").arg(style); |
|
} else { |
|
QString text; |
|
openSpan(text, QStringLiteral("font-family:monospace")); |
|
*output << text; |
|
} |
|
} |
|
|
|
void HTMLDecoder::end() |
|
{ |
|
Q_ASSERT(_output); |
|
|
|
if (_profile) { |
|
*_output << QStringLiteral("</body>"); |
|
} else { |
|
QString text; |
|
closeSpan(text); |
|
*_output << text; |
|
} |
|
|
|
_output = nullptr; |
|
} |
|
|
|
//TODO: Support for LineProperty (mainly double width , double height) |
|
void HTMLDecoder::decodeLine(const Character* const characters, int count, LineProperty /*properties*/ |
|
) |
|
{ |
|
Q_ASSERT(_output); |
|
|
|
QString text; |
|
|
|
int spaceCount = 0; |
|
|
|
for (int i = 0; i < count; i++) { |
|
//check if appearance of character is different from previous char |
|
if (characters[i].rendition != _lastRendition || |
|
characters[i].foregroundColor != _lastForeColor || |
|
characters[i].backgroundColor != _lastBackColor) { |
|
if (_innerSpanOpen) { |
|
closeSpan(text); |
|
_innerSpanOpen = false; |
|
} |
|
|
|
_lastRendition = characters[i].rendition; |
|
_lastForeColor = characters[i].foregroundColor; |
|
_lastBackColor = characters[i].backgroundColor; |
|
|
|
//build up style string |
|
QString style; |
|
|
|
bool useBold = (_lastRendition & RE_BOLD) != 0; |
|
if (useBold) { |
|
style.append(QLatin1String("font-weight:bold;")); |
|
} |
|
|
|
if ((_lastRendition & RE_UNDERLINE) != 0) { |
|
style.append(QLatin1String("font-decoration:underline;")); |
|
} |
|
|
|
style.append(QStringLiteral("color:%1;").arg(_lastForeColor.color(_colorTable).name())); |
|
|
|
style.append(QStringLiteral("background-color:%1;").arg(_lastBackColor.color(_colorTable).name())); |
|
|
|
//open the span with the current style |
|
openSpan(text, style); |
|
_innerSpanOpen = true; |
|
} |
|
|
|
//handle whitespace |
|
if (characters[i].isSpace()) { |
|
spaceCount++; |
|
} else { |
|
spaceCount = 0; |
|
} |
|
|
|
//output current character |
|
if (spaceCount < 2) { |
|
if ((characters[i].rendition & RE_EXTENDED_CHAR) != 0) { |
|
ushort extendedCharLength = 0; |
|
const uint* chars = ExtendedCharTable::instance.lookupExtendedChar(characters[i].character, extendedCharLength); |
|
if (chars != nullptr) { |
|
text.append(QString::fromUcs4(chars, extendedCharLength)); |
|
} |
|
} else { |
|
//escape HTML tag characters and just display others as they are |
|
const QChar ch = characters[i].character; |
|
if (ch == QLatin1Char('<')) { |
|
text.append(QLatin1String("<")); |
|
} else if (ch == QLatin1Char('>')) { |
|
text.append(QLatin1String(">")); |
|
} else if (ch == QLatin1Char('&')) { |
|
text.append(QLatin1String("&")); |
|
} else { |
|
text.append(ch); |
|
} |
|
} |
|
} else { |
|
// HTML truncates multiple spaces, so use a space marker instead |
|
// Use   instead of   so xmllint will work. |
|
text.append(QLatin1String(" ")); |
|
} |
|
} |
|
|
|
//close any remaining open inner spans |
|
if (_innerSpanOpen) { |
|
closeSpan(text); |
|
_innerSpanOpen = false; |
|
} |
|
|
|
//start new line |
|
text.append(QLatin1String("<br>")); |
|
|
|
*_output << text; |
|
} |
|
void HTMLDecoder::openSpan(QString& text , const QString& style) |
|
{ |
|
text.append(QStringLiteral("<span style=\"%1\">").arg(style)); |
|
} |
|
|
|
void HTMLDecoder::closeSpan(QString& text) |
|
{ |
|
text.append(QLatin1String("</span>")); |
|
}
|
|
|