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

/*
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("&lt;"));
} else if (ch == QLatin1Char('>')) {
text.append(QLatin1String("&gt;"));
} else if (ch == QLatin1Char('&')) {
text.append(QLatin1String("&amp;"));
} else {
text.append(ch);
}
}
} else {
// HTML truncates multiple spaces, so use a space marker instead
// Use &#160 instead of &nbsp so xmllint will work.
text.append(QLatin1String("&#160;"));
}
}
//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>"));
}