mirror of
https://github.com/KDE/konsole.git
synced 2025-12-23 23:38:08 -05:00
3117 lines
123 KiB
C++
3117 lines
123 KiB
C++
/*
|
||
SPDX-FileCopyrightText: 2007-2008 Robert Knight <robert.knight@gmail.com>
|
||
SPDX-FileCopyrightText: 1997, 1998 Lars Doelle <lars.doelle@on-line.de>
|
||
|
||
SPDX-License-Identifier: GPL-2.0-or-later
|
||
*/
|
||
// Own
|
||
#include "Vt102Emulation.h"
|
||
#include "config-konsole.h"
|
||
|
||
// Standard
|
||
#include <cstdio>
|
||
#include <unistd.h>
|
||
|
||
// Qt
|
||
#include <QEvent>
|
||
#include <QKeyEvent>
|
||
#include <QTemporaryFile>
|
||
#include <QTimer>
|
||
#include <QtEndian>
|
||
|
||
// KDE
|
||
#include <KLocalizedString>
|
||
#include <KNotification>
|
||
|
||
// Konsole
|
||
#include "EscapeSequenceUrlExtractor.h"
|
||
#include "session/SessionController.h"
|
||
#include "terminalDisplay/TerminalColor.h"
|
||
#include "terminalDisplay/TerminalDisplay.h"
|
||
#include "terminalDisplay/TerminalFonts.h"
|
||
|
||
#include <konsoledebug.h>
|
||
|
||
using Konsole::Vt102Emulation;
|
||
|
||
/*
|
||
The VT100 has 32 special graphical characters. The usual vt100 extended
|
||
xterm fonts have these at 0x00..0x1f.
|
||
|
||
QT's iso mapping leaves 0x00..0x7f without any changes. But the graphicals
|
||
come in here as proper unicode characters.
|
||
|
||
We treat non-iso10646 fonts as VT100 extended and do the required mapping
|
||
from unicode to 0x00..0x1f. The remaining translation is then left to the
|
||
QCodec.
|
||
*/
|
||
|
||
// assert for i in [0..31] : vt100extended(vt100_graphics[i]) == i.
|
||
|
||
/* clang-format off */
|
||
unsigned short Konsole::vt100_graphics[32] = {
|
||
// 0/8 1/9 2/10 3/11 4/12 5/13 6/14 7/15
|
||
0x0020, 0x25C6, 0x2592, 0x2409, 0x240c, 0x240d, 0x240a, 0x00b0,
|
||
0x00b1, 0x2424, 0x240b, 0x2518, 0x2510, 0x250c, 0x2514, 0x253c,
|
||
0x23ba, 0x23bb, 0x2500, 0x23bc, 0x23bd, 0x251c, 0x2524, 0x2534,
|
||
0x252c, 0x2502, 0x2264, 0x2265, 0x03C0, 0x2260, 0x00A3, 0x00b7
|
||
};
|
||
/* clang-format on */
|
||
|
||
enum XTERM_EXTENDED {
|
||
URL_LINK = '8',
|
||
};
|
||
|
||
Vt102Emulation::Vt102Emulation()
|
||
: Emulation()
|
||
, _currentModes(TerminalState())
|
||
, _savedModes(TerminalState())
|
||
, _pendingSessionAttributesUpdates(QHash<int, QString>())
|
||
, _sessionAttributesUpdateTimer(new QTimer(this))
|
||
, _reportFocusEvents(false)
|
||
, player(nullptr)
|
||
{
|
||
_sessionAttributesUpdateTimer->setSingleShot(true);
|
||
QObject::connect(_sessionAttributesUpdateTimer, &QTimer::timeout, this, &Konsole::Vt102Emulation::updateSessionAttributes);
|
||
|
||
initTokenizer();
|
||
imageId = 0;
|
||
savedKeys = QMap<char, qint64>();
|
||
tokenData = QByteArray();
|
||
|
||
for (int i = 0; i < 256; i++) {
|
||
colorTable[i] = QColor();
|
||
}
|
||
}
|
||
|
||
Vt102Emulation::~Vt102Emulation()
|
||
{
|
||
}
|
||
|
||
void Vt102Emulation::clearEntireScreen()
|
||
{
|
||
_currentScreen->clearEntireScreen();
|
||
bufferedUpdate();
|
||
}
|
||
|
||
void Vt102Emulation::clearHistory()
|
||
{
|
||
_graphicsImages.clear();
|
||
|
||
Emulation::clearHistory();
|
||
}
|
||
|
||
void Vt102Emulation::reset(bool softReset, bool preservePrompt)
|
||
{
|
||
// Save the current codec so we can set it later.
|
||
// Ideally we would want to use the profile setting
|
||
const QTextCodec *currentCodec = codec();
|
||
|
||
resetTokenizer();
|
||
if (softReset) {
|
||
resetMode(MODE_AppCuKeys);
|
||
saveMode(MODE_AppCuKeys);
|
||
resetMode(MODE_AppKeyPad);
|
||
saveMode(MODE_AppKeyPad);
|
||
} else {
|
||
resetModes();
|
||
}
|
||
|
||
resetCharset(0);
|
||
_screen[0]->reset(softReset, preservePrompt);
|
||
resetCharset(1);
|
||
_screen[1]->reset(softReset, preservePrompt);
|
||
|
||
if (currentCodec != nullptr) {
|
||
setCodec(currentCodec);
|
||
} else {
|
||
setCodec(LocaleCodec);
|
||
}
|
||
|
||
Q_EMIT resetCursorStyleRequest();
|
||
|
||
bufferedUpdate();
|
||
}
|
||
|
||
/* ------------------------------------------------------------------------- */
|
||
/* */
|
||
/* Processing the incoming byte stream */
|
||
/* */
|
||
/* ------------------------------------------------------------------------- */
|
||
|
||
/* Incoming Bytes Event pipeline
|
||
|
||
This section deals with decoding the incoming character stream.
|
||
Decoding means here, that the stream is first separated into `tokens'
|
||
which are then mapped to a `meaning' provided as operations by the
|
||
`Screen' class or by the emulation class itself.
|
||
|
||
The pipeline proceeds as follows:
|
||
|
||
- Tokenizing the ESC codes (onReceiveChar)
|
||
- VT100 code page translation of plain characters (applyCharset)
|
||
- Interpretation of ESC codes (processToken)
|
||
|
||
The escape codes and their meaning are described in the
|
||
technical reference of this program.
|
||
*/
|
||
|
||
// Tokens ------------------------------------------------------------------ --
|
||
|
||
/*
|
||
Since the tokens are the central notion if this section, we've put them
|
||
in front. They provide the syntactical elements used to represent the
|
||
terminals operations as byte sequences.
|
||
|
||
They are encodes here into a single machine word, so that we can later
|
||
switch over them easily. Depending on the token itself, additional
|
||
argument variables are filled with parameter values.
|
||
|
||
The tokens are defined below:
|
||
|
||
- CHR - Printable characters (32..255 but DEL (=127))
|
||
- CTL - Control characters (0..31 but ESC (= 27), DEL)
|
||
- ESC - Escape codes of the form <ESC><CHR but `[]()+*#'>
|
||
- ESC_DE - Escape codes of the form <ESC><any of `()+*#%'> C
|
||
- CSI_PN - Escape codes of the form <ESC>'[' {Pn} ';' {Pn} C
|
||
- CSI_PS - Escape codes of the form <ESC>'[' {Pn} ';' ... C
|
||
- CSI_PR - Escape codes of the form <ESC>'[' '?' {Pn} ';' ... C
|
||
- CSI_PE - Escape codes of the form <ESC>'[' '!' {Pn} ';' ... C
|
||
- CSI_SP - Escape codes of the form <ESC>'[' ' ' C
|
||
(3rd field is a space)
|
||
- CSI_PSP - Escape codes of the form <ESC>'[' '{Pn}' ' ' C
|
||
(4th field is a space)
|
||
- VT52 - VT52 escape codes
|
||
- <ESC><Chr>
|
||
- <ESC>'Y'{Pc}{Pc}
|
||
- XTE_HA - Xterm window/terminal attribute commands
|
||
of the form <ESC>`]' {Pn} `;' {Text} <BEL>
|
||
(Note that these are handled differently to the other formats)
|
||
|
||
The last two forms allow list of arguments. Since the elements of
|
||
the lists are treated individually the same way, they are passed
|
||
as individual tokens to the interpretation. Further, because the
|
||
meaning of the parameters are names (although represented as numbers),
|
||
they are includes within the token ('N').
|
||
|
||
*/
|
||
constexpr int token_construct(int t, int a, int n)
|
||
{
|
||
return (((n & 0xffff) << 16) | ((a & 0xff) << 8) | (t & 0xff));
|
||
}
|
||
constexpr int token_chr()
|
||
{
|
||
return token_construct(0, 0, 0);
|
||
}
|
||
constexpr int token_ctl(int a)
|
||
{
|
||
return token_construct(1, a, 0);
|
||
}
|
||
constexpr int token_esc(int a)
|
||
{
|
||
return token_construct(2, a, 0);
|
||
}
|
||
constexpr int token_esc_cs(int a, int b)
|
||
{
|
||
return token_construct(3, a, b);
|
||
}
|
||
constexpr int token_esc_de(int a)
|
||
{
|
||
return token_construct(4, a, 0);
|
||
}
|
||
constexpr int token_csi_ps(int a, int n)
|
||
{
|
||
return token_construct(5, a, n);
|
||
}
|
||
constexpr int token_csi_pn(int a)
|
||
{
|
||
return token_construct(6, a, 0);
|
||
}
|
||
constexpr int token_csi_pr(int a, int n)
|
||
{
|
||
return token_construct(7, a, n);
|
||
}
|
||
constexpr int token_vt52(int a)
|
||
{
|
||
return token_construct(8, a, 0);
|
||
}
|
||
constexpr int token_csi_pg(int a)
|
||
{
|
||
return token_construct(9, a, 0);
|
||
}
|
||
constexpr int token_csi_pe(int a)
|
||
{
|
||
return token_construct(10, a, 0);
|
||
}
|
||
constexpr int token_csi_sp(int a)
|
||
{
|
||
return token_construct(11, a, 0);
|
||
}
|
||
constexpr int token_csi_psp(int a, int n)
|
||
{
|
||
return token_construct(12, a, n);
|
||
}
|
||
constexpr int token_csi_pq(int a)
|
||
{
|
||
return token_construct(13, a, 0);
|
||
}
|
||
constexpr int token_osc(int a)
|
||
{
|
||
return token_construct(14, a, 0);
|
||
}
|
||
constexpr int token_apc(int a)
|
||
{
|
||
return token_construct(15, a, 0);
|
||
}
|
||
|
||
const int MAX_ARGUMENT = 40960;
|
||
|
||
// Tokenizer --------------------------------------------------------------- --
|
||
|
||
/* The tokenizer's state
|
||
|
||
The state is represented by the buffer (tokenBuffer, tokenBufferPos),
|
||
and accompanied by decoded arguments kept in params.
|
||
Note that they are kept internal in the tokenizer.
|
||
*/
|
||
|
||
void Vt102Emulation::resetTokenizer()
|
||
{
|
||
tokenBufferPos = 0;
|
||
params.count = 0;
|
||
params.value[0] = 0;
|
||
params.value[1] = 0;
|
||
params.sub[0].value[0] = 0;
|
||
params.sub[0].count = 0;
|
||
params.hasSubParams = false;
|
||
tokenState = -1;
|
||
}
|
||
|
||
void Vt102Emulation::addDigit(int digit)
|
||
{
|
||
if (params.sub[params.count].count == 0) {
|
||
params.value[params.count] = qMin(10 * params.value[params.count] + digit, MAX_ARGUMENT);
|
||
} else {
|
||
struct subParam *sub = ¶ms.sub[params.count];
|
||
sub->value[sub->count] = qMin(10 * sub->value[sub->count] + digit, MAX_ARGUMENT);
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::addArgument()
|
||
{
|
||
params.count = qMin(params.count + 1, MAXARGS - 1);
|
||
params.value[params.count] = 0;
|
||
params.sub[params.count].value[0] = 0;
|
||
params.sub[params.count].count = 0;
|
||
}
|
||
|
||
void Vt102Emulation::addSub()
|
||
{
|
||
struct subParam *sub = ¶ms.sub[params.count];
|
||
sub->count = qMin(sub->count + 1, MAXARGS - 1);
|
||
sub->value[sub->count] = 0;
|
||
params.hasSubParams = true;
|
||
}
|
||
|
||
void Vt102Emulation::addToCurrentToken(uint cc)
|
||
{
|
||
tokenBufferPos = qMin(tokenBufferPos, MAX_TOKEN_LENGTH - 1);
|
||
tokenBuffer[tokenBufferPos] = cc;
|
||
tokenBufferPos++;
|
||
}
|
||
|
||
// Character Class flags used while decoding
|
||
const int CTL = 1; // Control character
|
||
const int CHR = 2; // Printable character
|
||
const int CPN = 4; // TODO: Document me
|
||
const int DIG = 8; // Digit
|
||
const int SCS = 16; // Select Character Set
|
||
const int GRP = 32; // TODO: Document me
|
||
const int CPS = 64; // Character which indicates end of window resize
|
||
const int INT = 128; // Intermediate Byte (ECMA 48 5.4 -> CSI P..P I..I F)
|
||
|
||
void Vt102Emulation::initTokenizer()
|
||
{
|
||
int i;
|
||
quint8 *s;
|
||
for (i = 0; i < 256; ++i) {
|
||
charClass[i] = 0;
|
||
}
|
||
for (i = 0; i < 32; ++i) {
|
||
charClass[i] |= CTL;
|
||
}
|
||
for (i = 32; i < 256; ++i) {
|
||
charClass[i] |= CHR;
|
||
}
|
||
for (i = 0x20; i < 0x30; ++i) {
|
||
charClass[i] |= INT;
|
||
}
|
||
for (s = (quint8 *)"@ABCDEFGHILMPSTXZbcdfry"; *s != 0U; ++s) {
|
||
charClass[*s] |= CPN;
|
||
}
|
||
// resize = \e[8;<row>;<col>t
|
||
for (s = (quint8 *)"t"; *s != 0U; ++s) {
|
||
charClass[*s] |= CPS;
|
||
}
|
||
for (s = (quint8 *)"0123456789"; *s != 0U; ++s) {
|
||
charClass[*s] |= DIG;
|
||
}
|
||
for (s = (quint8 *)"()+*%"; *s != 0U; ++s) {
|
||
charClass[*s] |= SCS;
|
||
}
|
||
for (s = (quint8 *)"()+*#[]%"; *s != 0U; ++s) {
|
||
charClass[*s] |= GRP;
|
||
}
|
||
|
||
resetTokenizer();
|
||
}
|
||
|
||
/* Ok, here comes the nasty part of the decoder.
|
||
|
||
Instead of keeping an explicit state, we deduce it from the
|
||
token scanned so far. It is then immediately combined with
|
||
the current character to form a scanning decision.
|
||
|
||
This is done by the following defines.
|
||
|
||
- P is the length of the token scanned so far.
|
||
- L (often P-1) is the position on which contents we base a decision.
|
||
- C is a character or a group of characters (taken from 'charClass').
|
||
|
||
- 'cc' is the current character
|
||
- 's' is a pointer to the start of the token buffer
|
||
- 'p' is the current position within the token buffer
|
||
|
||
Note that they need to applied in proper order.
|
||
*/
|
||
|
||
/* clang-format off */
|
||
#define lec(P,L,C) (p == (P) && s[(L)] == (C))
|
||
#define les(P,L,C) (p == (P) && s[L] < 256 && (charClass[s[(L)]] & (C)) == (C))
|
||
#define eec(C) (p >= 3 && cc == (C))
|
||
#define ccc(C) (cc < 256 && (charClass[cc] & (C)) == (C))
|
||
#define ees(C) (p >= 3 && cc < 256 && (charClass[cc] & (C)) == (C))
|
||
#define eps(C) (p >= 3 && s[2] != '?' && s[2] != '!' && s[2] != '=' && s[2] != '>' && cc < 256 && (charClass[cc] & (C)) == (C) && (charClass[s[p-2]] & (INT)) != (INT) )
|
||
#define epp( ) (p >= 3 && s[2] == '?')
|
||
#define epe( ) (p >= 3 && s[2] == '!')
|
||
#define eeq( ) (p >= 3 && s[2] == '=')
|
||
#define egt( ) (p >= 3 && s[2] == '>')
|
||
#define esp( ) (p >= 4 && s[2] == SP )
|
||
#define epsp( ) (p >= 5 && s[3] == SP )
|
||
#define osc (tokenBufferPos >= 2 && tokenBuffer[1] == ']')
|
||
//#define apc(C) (p >= 3 && s[1] == '_' && s[2] == C)
|
||
#define pm (tokenBufferPos >= 2 && tokenBuffer[1] == '^')
|
||
#define apc (tokenBufferPos >= 2 && tokenBuffer[1] == '_')
|
||
#define ces(C) (cc < 256 && (charClass[cc] & (C)) == (C))
|
||
#define dcs (p >= 2 && s[0] == ESC && s[1] == 'P')
|
||
#define sixel( ) (p == 1 && cc >= '?' && cc <= '~')
|
||
|
||
/* clang-format on */
|
||
|
||
#define CNTL(c) ((c) - '@')
|
||
const int ESC = 27;
|
||
const int DEL = 127;
|
||
const int SP = 32;
|
||
|
||
// process an incoming unicode character
|
||
//
|
||
// Parser based on the vt100.net diagram:
|
||
// Williams, Paul Flo. “A parser for DEC’s ANSI-compatible video
|
||
// terminals.” VT100.net. https://vt100.net/emu/dec_ansi_parser
|
||
|
||
void Vt102Emulation::switchState(const ParserStates newState, const uint cc)
|
||
{
|
||
switch (_state) {
|
||
case DcsPassthrough:
|
||
unhook();
|
||
break;
|
||
case OscString:
|
||
osc_end(cc);
|
||
break;
|
||
case SosPmApcString:
|
||
apc_end();
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
|
||
_state = newState;
|
||
}
|
||
|
||
void Vt102Emulation::esc_dispatch(const uint cc)
|
||
{
|
||
if (_ignore)
|
||
return;
|
||
if (_nIntermediate == 0) {
|
||
processToken(token_esc(cc), 0, 0);
|
||
} else if (_nIntermediate == 1) {
|
||
const uint intermediate = _intermediate[0];
|
||
if ((charClass[intermediate] & SCS) == SCS) {
|
||
processToken(token_esc_cs(intermediate, cc), 0, 0);
|
||
} else if (intermediate == '#') {
|
||
processToken(token_esc_de(cc), 0, 0);
|
||
}
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::clear()
|
||
{
|
||
_nIntermediate = 0;
|
||
_ignore = false;
|
||
resetTokenizer();
|
||
}
|
||
|
||
#define MAX_INTERMEDIATES 1
|
||
void Vt102Emulation::collect(const uint cc)
|
||
{
|
||
addToCurrentToken(cc);
|
||
if (cc > 0x30)
|
||
return;
|
||
if (_nIntermediate >= MAX_INTERMEDIATES) {
|
||
_ignore = true;
|
||
return;
|
||
}
|
||
_intermediate[_nIntermediate++] = cc;
|
||
}
|
||
|
||
void Vt102Emulation::param(const uint cc)
|
||
{
|
||
addToCurrentToken(cc);
|
||
if ((charClass[cc] & DIG) == DIG) {
|
||
addDigit(cc - '0');
|
||
} else if (cc == ';') {
|
||
addArgument();
|
||
} else if (cc == ':') {
|
||
addSub();
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::csi_dispatch(const uint cc)
|
||
{
|
||
if (_ignore || (params.hasSubParams && cc != 'm')) // Be conservative for now
|
||
return;
|
||
if ((tokenBufferPos == 0 || (tokenBuffer[0] != '?' && tokenBuffer[0] != '!' && tokenBuffer[0] != '=' && tokenBuffer[0] != '>')) && cc < 256
|
||
&& (charClass[cc] & CPN) == CPN && _nIntermediate == 0) {
|
||
processToken(token_csi_pn(cc), params.value[0], params.value[1]);
|
||
} else if ((tokenBufferPos == 0 || (tokenBuffer[0] != '?' && tokenBuffer[0] != '!' && tokenBuffer[0] != '=' && tokenBuffer[0] != '>')) && cc < 256
|
||
&& (charClass[cc] & CPS) == CPS && _nIntermediate == 0) {
|
||
processToken(token_csi_ps(cc, params.value[0]), params.value[1], params.value[2]);
|
||
} else if (tokenBufferPos != 0 && tokenBuffer[0] == '!') {
|
||
processToken(token_csi_pe(cc), 0, 0);
|
||
} else if (_nIntermediate == 1 && _intermediate[0] == ' ') {
|
||
if (tokenBufferPos == 1) {
|
||
processToken(token_csi_sp(cc), 0, 0);
|
||
} else {
|
||
processToken(token_csi_psp(cc, params.value[0]), 0, 0);
|
||
}
|
||
} else if (cc == 'y' && _nIntermediate == 1 && _intermediate[0] == '*') {
|
||
processChecksumRequest(params.count, params.value);
|
||
} else {
|
||
for (int i = 0; i <= params.count; i++) {
|
||
if (tokenBufferPos != 0 && tokenBuffer[0] == '?') {
|
||
processToken(token_csi_pr(cc, params.value[i]), i, 0);
|
||
} else if (tokenBufferPos != 0 && tokenBuffer[0] == '=') {
|
||
processToken(token_csi_pq(cc), 0, 0);
|
||
} else if (tokenBufferPos != 0 && tokenBuffer[0] == '>') {
|
||
processToken(token_csi_pg(cc), 0, 0);
|
||
} else if (cc == 'm' && !params.sub[i].count && params.count - i >= 4 && (params.value[i] == 38 || params.value[i] == 48 || params.value[i] == 58)
|
||
&& params.value[i + 1] == 2) {
|
||
// ESC[ ... 48;2;<red>;<green>;<blue> ... m -or- ESC[ ... 38;2;<red>;<green>;<blue> ... m
|
||
i += 2;
|
||
processToken(token_csi_ps(cc, params.value[i - 2]),
|
||
COLOR_SPACE_RGB,
|
||
(params.value[i] << 16) | (params.value[i + 1] << 8) | params.value[i + 2]);
|
||
i += 2;
|
||
} else if (cc == 'm' && params.sub[i].count >= 5 && (params.value[i] == 38 || params.value[i] == 48 || params.value[i] == 58)
|
||
&& params.sub[i].value[1] == 2) {
|
||
// ESC[ ... 48:2:<id>:<red>:<green>:<blue> ... m -or- ESC[ ... 38:2:<id>:<red>:<green>:<blue> ... m
|
||
processToken(token_csi_ps(cc, params.value[i]),
|
||
COLOR_SPACE_RGB,
|
||
(params.sub[i].value[3] << 16) | (params.sub[i].value[4] << 8) | params.sub[i].value[5]);
|
||
} else if (cc == 'm' && params.sub[i].count == 4 && (params.value[i] == 38 || params.value[i] == 48 || params.value[i] == 58)
|
||
&& params.sub[i].value[1] == 2) {
|
||
// ESC[ ... 48:2:<red>:<green>:<blue> ... m -or- ESC[ ... 38:2:<red>:<green>:<blue> ... m
|
||
processToken(token_csi_ps(cc, params.value[i]),
|
||
COLOR_SPACE_RGB,
|
||
(params.sub[i].value[2] << 16) | (params.sub[i].value[3] << 8) | params.sub[i].value[4]);
|
||
} else if (cc == 'm' && params.sub[i].count == 1 && params.value[i] == 4) {
|
||
processToken(token_csi_ps(cc, params.value[i]), params.sub[i].value[1], 1);
|
||
} else if (cc == 'm' && !params.sub[i].count && params.count - i >= 2 && (params.value[i] == 38 || params.value[i] == 48 || params.value[i] == 58)
|
||
&& params.value[i + 1] == 5) {
|
||
// ESC[ ... 48;5;<index> ... m -or- ESC[ ... 38;5;<index> ... m
|
||
i += 2;
|
||
processToken(token_csi_ps(cc, params.value[i - 2]), COLOR_SPACE_256, params.value[i]);
|
||
} else if (cc == 'm' && params.sub[i].count >= 2 && (params.value[i] == 38 || params.value[i] == 48 || params.value[i] == 58)
|
||
&& params.sub[i].value[1] == 5) {
|
||
// ESC[ ... 48:5:<index> ... m -or- ESC[ ... 38:5:<index> ... m
|
||
processToken(token_csi_ps(cc, params.value[i]), COLOR_SPACE_256, params.sub[i].value[2]);
|
||
} else if (_nIntermediate == 0) {
|
||
processToken(token_csi_ps(cc, params.value[i]), 0, 0);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::osc_start()
|
||
{
|
||
tokenBufferPos = 0;
|
||
}
|
||
|
||
void Vt102Emulation::osc_put(const uint cc)
|
||
{
|
||
addToCurrentToken(cc);
|
||
|
||
// Special case: iterm file protocol is a long escape sequence
|
||
if (tokenState == -1) {
|
||
tokenStateChange = "1337;File=:";
|
||
tokenState = 0;
|
||
}
|
||
if (tokenState >= 0) {
|
||
if ((uint)tokenStateChange[tokenState] == tokenBuffer[tokenBufferPos - 1]) {
|
||
tokenState++;
|
||
tokenPos = tokenBufferPos;
|
||
if ((uint)tokenState == strlen(tokenStateChange)) {
|
||
tokenState = -2;
|
||
tokenData.clear();
|
||
}
|
||
return;
|
||
}
|
||
} else if (tokenState == -2) {
|
||
if (tokenBufferPos - tokenPos == 4) {
|
||
tokenData.append(QByteArray::fromBase64(QString::fromUcs4(&tokenBuffer[tokenPos], 4).toLocal8Bit()));
|
||
tokenBufferPos -= 4;
|
||
return;
|
||
}
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::osc_end(const uint cc)
|
||
{
|
||
// This runs two times per link, the first prepares the link to be read,
|
||
// the second finalizes it. The escape sequence is in two parts
|
||
// start: '\e ] 8 ; <id-path> ; <url-part> \e \\'
|
||
// end: '\e ] 8 ; ; \e \\'
|
||
// GNU libtextstyle inserts the IDs, for instance; many examples
|
||
// do not.
|
||
if (tokenBuffer[0] == XTERM_EXTENDED::URL_LINK) {
|
||
// printf '\e]8;;https://example.com\e\\This is a link\e]8;;\e\\\n'
|
||
emit toggleUrlExtractionRequest();
|
||
}
|
||
|
||
processSessionAttributeRequest(tokenBufferPos, cc);
|
||
}
|
||
|
||
void Vt102Emulation::put(const uint cc)
|
||
{
|
||
if (m_SixelPictureDefinition && cc >= 0x21) {
|
||
addToCurrentToken(cc);
|
||
processSixel(cc);
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::hook(const uint cc)
|
||
{
|
||
if (cc == 'q' && _nIntermediate == 0) {
|
||
m_SixelPictureDefinition = true;
|
||
resetTokenizer();
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::unhook()
|
||
{
|
||
m_SixelPictureDefinition = false;
|
||
SixelModeDisable();
|
||
resetTokenizer();
|
||
}
|
||
|
||
void Vt102Emulation::apc_start(const uint cc)
|
||
{
|
||
tokenBufferPos = 0;
|
||
if (cc == 0x9F || cc == 0x5F) {
|
||
_sosPmApc = Apc;
|
||
} else if (cc == 0x9E || cc == 0x5E) {
|
||
_sosPmApc = Pm;
|
||
} else {
|
||
// 0x98, 0x58
|
||
_sosPmApc = Sos;
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::apc_put(const uint cc)
|
||
{
|
||
if (_sosPmApc != Apc) {
|
||
return;
|
||
}
|
||
|
||
addToCurrentToken(cc);
|
||
|
||
// <ESC> '_' ... <ESC> '\'
|
||
if (tokenBufferPos > 1 && tokenBuffer[0] == 'G') {
|
||
if (tokenState == -1) {
|
||
tokenStateChange = ";";
|
||
tokenState = 0;
|
||
} else if (tokenState >= 0) {
|
||
if ((uint)tokenStateChange[tokenState] == tokenBuffer[tokenBufferPos - 1]) {
|
||
tokenState++;
|
||
tokenPos = tokenBufferPos;
|
||
if ((uint)tokenState == strlen(tokenStateChange)) {
|
||
tokenState = -2;
|
||
tokenData.clear();
|
||
}
|
||
}
|
||
} else if (tokenState == -2) {
|
||
if (tokenBufferPos - tokenPos == 4) {
|
||
tokenData.append(QByteArray::fromBase64(QString::fromUcs4(&tokenBuffer[tokenPos], 4).toLocal8Bit()));
|
||
tokenBufferPos -= 4;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::apc_end()
|
||
{
|
||
if (_sosPmApc == Apc && tokenBuffer[0] == 'G') {
|
||
// Graphics command
|
||
processGraphicsToken(tokenBufferPos);
|
||
resetTokenizer();
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::receiveChars(const QVector<uint> &chars)
|
||
{
|
||
for (uint cc : chars) {
|
||
// early out for displayable characters
|
||
if (_state == Ground && ((cc >= 0x20 && cc <= 0x7E) || cc >= 0xA0)) {
|
||
_currentScreen->displayCharacter(applyCharset(cc));
|
||
continue;
|
||
}
|
||
|
||
if (getMode(MODE_Ansi)) {
|
||
// First, process characters that act the same on all states, i.e.
|
||
// coming from "anywhere" in the VT100.net diagram.
|
||
if (cc == 0x1B) {
|
||
switchState(Escape, cc);
|
||
clear();
|
||
} else if (cc == 0x9B) {
|
||
switchState(CsiEntry, cc);
|
||
clear();
|
||
} else if (cc == 0x90) {
|
||
switchState(DcsEntry, cc);
|
||
clear();
|
||
} else if (cc == 0x9D) {
|
||
osc_start();
|
||
switchState(OscString, cc);
|
||
} else if (cc == 0x98 || cc == 0x9E || cc == 0x9F) {
|
||
apc_start(cc);
|
||
switchState(SosPmApcString, cc);
|
||
} else if (cc == 0x18 || cc == 0x1A || (cc >= 0x80 && cc <= 0x9A)) { // 0x90, 0x98 taken care of just above.
|
||
// konsole has always ignored CAN and SUB in OSC, extend the behavior a bit
|
||
// this differs from VT240, where 7-bit ST, 8-bit ST, ESC + chr, ***CAN, SUB, C1*** terminate and show SIXEL
|
||
if (_state != OscString && _state != SosPmApcString && _state != DcsPassthrough) {
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
switchState(Ground, cc);
|
||
}
|
||
} else if (cc == 0x9C) {
|
||
// no action
|
||
switchState(Ground, cc);
|
||
|
||
} else {
|
||
// Now take the current state into account
|
||
switch (_state) {
|
||
case Ground:
|
||
if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else {
|
||
// 0x7F is ignored by displayCharacter(), since its Character::width() is -1
|
||
_currentScreen->displayCharacter(applyCharset(cc));
|
||
}
|
||
break;
|
||
case Escape:
|
||
if (cc == 0x5B) {
|
||
switchState(CsiEntry, cc);
|
||
clear();
|
||
} else if ((cc >= 0x30 && cc <= 0x4F) || (cc >= 0x51 && cc <= 0x57) || (cc >= 0x59 && cc <= 0x5A) || cc == 0x5C
|
||
|| (cc >= 0x60 && cc <= 0x7E)) {
|
||
esc_dispatch(cc);
|
||
switchState(Ground, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
switchState(EscapeIntermediate, cc);
|
||
} else if (cc == 0x5D) {
|
||
osc_start();
|
||
switchState(OscString, cc);
|
||
} else if (cc == 0x50) {
|
||
switchState(DcsEntry, cc);
|
||
clear();
|
||
} else if (cc == 0x58 || cc == 0x5E || cc == 0x5F) {
|
||
apc_start(cc);
|
||
switchState(SosPmApcString, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case EscapeIntermediate:
|
||
if (cc >= 0x30 && cc <= 0x7E) {
|
||
esc_dispatch(cc);
|
||
switchState(Ground, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case CsiEntry:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
csi_dispatch(cc);
|
||
switchState(Ground, cc);
|
||
} else if (cc >= 0x30 && cc <= 0x3B) { // recognize 0x3A as part of params
|
||
param(cc);
|
||
switchState(CsiParam, cc);
|
||
} else if (cc >= 0x3C && cc <= 0x3F) {
|
||
collect(cc);
|
||
switchState(CsiParam, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
switchState(CsiIntermediate, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case CsiParam:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
csi_dispatch(cc);
|
||
switchState(Ground, cc);
|
||
} else if (cc >= 0x30 && cc <= 0x3B) { // recognize 0x3A as part of params
|
||
param(cc);
|
||
} else if (cc >= 0x3C && cc <= 0x3F) {
|
||
switchState(CsiIgnore, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
switchState(CsiIntermediate, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case CsiIntermediate:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
csi_dispatch(cc);
|
||
switchState(Ground, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
} else if (cc >= 0x30 && cc <= 0x3F) {
|
||
switchState(CsiIgnore, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case CsiIgnore:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
switchState(Ground, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if ((/*cc >= 0x20 && */ cc <= 0x3F) || cc == 0x7F) { // cc <= 0x1F taking care of just above
|
||
// ignore
|
||
}
|
||
break;
|
||
case DcsEntry:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
hook(cc);
|
||
switchState(DcsPassthrough, cc);
|
||
} else if (cc >= 0x30 && cc <= 0x3B) { // recognize 0x3A as part of params
|
||
param(cc);
|
||
switchState(DcsParam, cc);
|
||
} else if (cc >= 0x3C && cc <= 0x3F) {
|
||
collect(cc);
|
||
switchState(DcsParam, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
switchState(DcsIntermediate, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case DcsParam:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
hook(cc);
|
||
switchState(DcsPassthrough, cc);
|
||
} else if (cc >= 0x30 && cc <= 0x3B) { // recognize 0x3A as part of params
|
||
param(cc);
|
||
} else if (cc >= 0x3C && cc <= 0x3F) {
|
||
switchState(DcsIgnore, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
switchState(DcsIntermediate, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case DcsIntermediate:
|
||
if (cc >= 0x40 && cc <= 0x7E) {
|
||
hook(cc);
|
||
switchState(DcsPassthrough, cc);
|
||
} else if (cc >= 0x20 && cc <= 0x2F) {
|
||
collect(cc);
|
||
} else if (cc >= 0x30 && cc <= 0x3F) {
|
||
switchState(DcsIgnore, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case DcsPassthrough:
|
||
if (cc <= 0x7E || cc >= 0xA0) { // 0x18, 0x1A, 0x1B already taken care of
|
||
put(cc);
|
||
// 0x9C already taken care of.
|
||
} else if (cc == 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case DcsIgnore:
|
||
// 0x9C already taken care of.
|
||
if (cc <= 0x7F) {
|
||
// ignore
|
||
}
|
||
break;
|
||
case OscString:
|
||
if ((cc >= 0x20 && cc <= 0x7F) || cc >= 0xA0) {
|
||
osc_put(cc);
|
||
} else if (cc == 0x07 // recognize BEL as OSC terminator
|
||
|| cc == 0x9C) {
|
||
switchState(Ground, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B already taken care of. 0x07 taken care of just above.
|
||
// ignore
|
||
}
|
||
break;
|
||
case SosPmApcString:
|
||
if (cc <= 0x7F || cc >= 0xA0) { // 0x18, 0x1A, 0x1B already taken care of.
|
||
apc_put(cc); // while the vt100.net diagram has ignore here, konsole does process some APCs (kitty images).
|
||
}
|
||
// 0x9C already taken care of.
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
} else {
|
||
// VT52 Mode
|
||
|
||
// First, process characters that act the same on all states
|
||
if (cc == 0x18 || cc == 0x1A) {
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
switchState(Ground, cc);
|
||
} else if (cc == 0x1B) {
|
||
switchState(Vt52Escape, cc);
|
||
} else if (cc <= 0x1F) { // 0x18, 0x1A, 0x1B taken care of just above
|
||
processToken(token_ctl(cc + '@'), 0, 0);
|
||
} else {
|
||
// Now take the current state into account
|
||
switch (_state) {
|
||
case Ground:
|
||
_currentScreen->displayCharacter(applyCharset(cc));
|
||
break;
|
||
case Vt52Escape:
|
||
if (cc == 'Y') {
|
||
switchState(Vt52CupRow, cc);
|
||
} else if ((cc >= 0x20 && cc <= 'X') || (cc >= 'Z' && cc <= 0x7F)) {
|
||
processToken(token_vt52(cc), 0, 0);
|
||
switchState(Ground, cc);
|
||
}
|
||
break;
|
||
case Vt52CupRow:
|
||
tokenBuffer[0] = cc;
|
||
switchState(Vt52CupColumn, cc);
|
||
break;
|
||
case Vt52CupColumn:
|
||
processToken(token_vt52('Y'), tokenBuffer[0], cc);
|
||
switchState(Ground, cc);
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::processChecksumRequest([[maybe_unused]] int crargc, int crargv[])
|
||
{
|
||
int checksum = 0;
|
||
|
||
#if ENABLE_DECRQCRA
|
||
int top, left, bottom, right;
|
||
|
||
/* DEC STD-070 5-179 "If Pp is 0 or omitted, subsequent parameters are ignored
|
||
* and a checksum for all page memory will be reported."
|
||
*/
|
||
if (crargv[1] == 0) {
|
||
crargc = 1;
|
||
}
|
||
/* clang-format off */
|
||
if (crargc >= 2) { top = crargv[2]; } else { top = 1; }
|
||
if (crargc >= 3) { left = crargv[3]; } else { left = 1; }
|
||
if (crargc >= 4) { bottom = crargv[4]; } else { bottom = _currentScreen->getLines(); }
|
||
if (crargc >= 5) { right = crargv[5]; } else { right = _currentScreen->getColumns(); }
|
||
/* clang-format on */
|
||
|
||
if (top > bottom || left > right) {
|
||
return;
|
||
}
|
||
|
||
if (_currentScreen->getMode(MODE_Origin)) {
|
||
top += _currentScreen->topMargin();
|
||
bottom += _currentScreen->topMargin();
|
||
}
|
||
|
||
top = qBound(1, top, _currentScreen->getLines());
|
||
bottom = qBound(1, bottom, _currentScreen->getLines());
|
||
|
||
int imgsize = sizeof(Character) * _currentScreen->getLines() * _currentScreen->getColumns();
|
||
Character *image = (Character *)malloc(imgsize);
|
||
if (image == nullptr) {
|
||
fprintf(stderr, "couldn't alloc mem\n");
|
||
return;
|
||
}
|
||
_currentScreen->getImage(image, imgsize, _currentScreen->getHistLines(), _currentScreen->getHistLines() + _currentScreen->getLines() - 1);
|
||
|
||
for (int y = top - 1; y <= bottom - 1; y++) {
|
||
for (int x = left - 1; x <= right - 1; x++) {
|
||
// XXX: Apparently, VT520 uses 0x00 for uninitialized cells, konsole can't tell uninitialized cells from spaces
|
||
Character c = image[y * _currentScreen->getColumns() + x];
|
||
|
||
if (c.rendition.f.conceal) {
|
||
checksum += 0x20; // don't reveal secrets
|
||
} else {
|
||
checksum += c.character;
|
||
}
|
||
|
||
checksum += c.rendition.f.bold * 0x80;
|
||
checksum += c.rendition.f.blink * 0x40;
|
||
checksum += c.rendition.f.reverse * 0x20;
|
||
checksum += !!(c.rendition.all & RE_UNDERLINE_MASK) * 0x10;
|
||
}
|
||
}
|
||
|
||
free(image);
|
||
#endif
|
||
|
||
char tmp[30];
|
||
checksum = -checksum;
|
||
checksum &= 0xffff;
|
||
snprintf(tmp, sizeof(tmp), "\033P%d!~%04X\033\\", crargv[0], checksum);
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::processSessionAttributeRequest(const int tokenSize, const uint terminator)
|
||
{
|
||
// Describes the window or terminal session attribute to change
|
||
// See Session::SessionAttributes for possible values
|
||
int attribute = 0;
|
||
int i;
|
||
|
||
/* clang-format off */
|
||
for (i = 0; i < tokenSize &&
|
||
tokenBuffer[i] >= '0' &&
|
||
tokenBuffer[i] <= '9'; i++)
|
||
{
|
||
attribute = 10 * attribute + (tokenBuffer[i]-'0');
|
||
}
|
||
/* clang-format on */
|
||
|
||
if (tokenBuffer[i] != ';') {
|
||
reportDecodingError(token_osc(terminator));
|
||
return;
|
||
}
|
||
// skip initial ';'
|
||
++i;
|
||
|
||
QString value = QString::fromUcs4(&tokenBuffer[i], tokenSize - i);
|
||
if (_currentScreen->urlExtractor() && _currentScreen->urlExtractor()->reading()) {
|
||
// To handle '\e ] 8 ; <id-part> ; <url-part>' we discard
|
||
// the <id-part>. Often it is empty, but GNU libtextstyle
|
||
// may output an id here, see e.g.
|
||
// https://www.gnu.org/software/gettext/libtextstyle/manual/libtextstyle.html#index-styled_005fostream_005fset_005fhyperlink
|
||
value.remove(0, value.indexOf(QLatin1Char(';')) + 1);
|
||
_currentScreen->urlExtractor()->setUrl(value);
|
||
return;
|
||
}
|
||
|
||
if (attribute == SemanticPrompts) {
|
||
if (value[0] == QLatin1Char('A') || value[0] == QLatin1Char('N') || value[0] == QLatin1Char('P')) {
|
||
_currentScreen->setReplMode(REPL_PROMPT);
|
||
}
|
||
if (value[0] == QLatin1Char('L') && _currentScreen->getCursorX() > 0) {
|
||
_currentScreen->nextLine();
|
||
}
|
||
if (value[0] == QLatin1Char('B')) {
|
||
_currentScreen->setReplMode(REPL_INPUT);
|
||
}
|
||
if (value[0] == QLatin1Char('C')) {
|
||
_currentScreen->setReplMode(REPL_OUTPUT);
|
||
}
|
||
if (value[0] == QLatin1Char('D')) {
|
||
_currentScreen->setReplMode(REPL_None);
|
||
}
|
||
QMap<QString, QString> params;
|
||
auto list = value.split(QLatin1Char(';'));
|
||
for (int i = 1; i < list.size(); i++) {
|
||
int eq = list.at(i).indexOf(QLatin1Char('='));
|
||
if (i == 1 && value[0] == QLatin1Char('D')) {
|
||
// Special case - exit code without '='
|
||
params[QLatin1String("exit_code")] = list.at(1);
|
||
int exitCode = list.at(1).toInt();
|
||
if (exitCode) {
|
||
_currentScreen->setExitCode(exitCode);
|
||
}
|
||
} else if (eq > 0) {
|
||
params[list.at(i).mid(0, eq)] = list.at(i).mid(eq + 1);
|
||
}
|
||
}
|
||
}
|
||
if (attribute == ReportColors) {
|
||
// RGB colors
|
||
QStringList params = value.split(QLatin1Char(';'));
|
||
for (int j = 0; j < params.length(); j += 2) {
|
||
if (params.length() == j + 1) {
|
||
return;
|
||
}
|
||
int c = params[j].toInt();
|
||
if (params[j + 1] == QLatin1String("?")) {
|
||
QColor color = colorTable[c];
|
||
if (!color.isValid()) {
|
||
color = CharacterColor(COLOR_SPACE_256, c).color(ColorScheme::defaultTable);
|
||
}
|
||
reportColor(c, color);
|
||
return;
|
||
}
|
||
QColor col(params[1]);
|
||
colorTable[c] = col;
|
||
}
|
||
return;
|
||
}
|
||
if (attribute == ResetColors) {
|
||
// RGB colors
|
||
QStringList params = value.split(QLatin1Char(';'));
|
||
for (int k = 0; k < params.length(); k++) {
|
||
int c = params[k].toInt();
|
||
colorTable[c] = QColor();
|
||
}
|
||
}
|
||
if (attribute == Notification) {
|
||
// Notification
|
||
auto params = value.split(QLatin1Char(';'));
|
||
if (params.length() < 1 || params[0] != QLatin1String("notify")) {
|
||
return;
|
||
}
|
||
|
||
const auto hasFocus = _currentScreen->currentTerminalDisplay()->hasFocus();
|
||
KNotification *notification = nullptr;
|
||
if (params.length() >= 3) {
|
||
notification =
|
||
KNotification::event(hasFocus ? QStringLiteral("ProcessNotification") : QStringLiteral("ProcessNotificationHidden"), params[1], params[2]);
|
||
} else {
|
||
notification = KNotification::event(hasFocus ? QStringLiteral("ProcessNotification") : QStringLiteral("ProcessNotificationHidden"), params[1]);
|
||
}
|
||
|
||
notification->setDefaultAction(i18n("Show session"));
|
||
connect(notification, &KNotification::defaultActivated, this, [this, notification]() {
|
||
_currentScreen->currentTerminalDisplay()->notificationClicked(notification->xdgActivationToken());
|
||
});
|
||
|
||
return;
|
||
}
|
||
|
||
if (value == QLatin1String("?")) {
|
||
// pass terminator type indication here, because OSC response terminator
|
||
// should match the terminator of OSC request.
|
||
Q_EMIT sessionAttributeRequest(attribute, terminator);
|
||
return;
|
||
}
|
||
|
||
if (attribute == Session::ProfileChange) {
|
||
if (value.startsWith(QLatin1String("CursorShape="))) {
|
||
const auto numStr = QStringView(value).right(1);
|
||
const Enum::CursorShapeEnum shape = static_cast<Enum::CursorShapeEnum>(numStr.toInt());
|
||
Q_EMIT setCursorStyleRequest(shape);
|
||
return;
|
||
}
|
||
}
|
||
|
||
if (attribute == Image) {
|
||
bool inlineImage = false;
|
||
bool inlineMedia = false;
|
||
if (value.startsWith(QLatin1String("ReportCellSize"))) {
|
||
iTermReportCellSize();
|
||
return;
|
||
}
|
||
if (!value.startsWith(QLatin1String("File="))) {
|
||
return;
|
||
}
|
||
int pos = value.indexOf(QLatin1Char(':'));
|
||
QStringList params = value.mid(5, pos - 5).split(QLatin1Char(';'));
|
||
bool keepAspect = true;
|
||
int scaledWidth = 0;
|
||
int scaledHeight = 0;
|
||
bool moveCursor = true;
|
||
for (const auto &p : params) {
|
||
int eq = p.indexOf(QLatin1Char('='));
|
||
if (eq > 0) {
|
||
QString var = p.mid(0, eq);
|
||
QString val = p.mid(eq + 1);
|
||
if (var == QLatin1String("inline")) {
|
||
inlineImage = val == QLatin1String("1");
|
||
}
|
||
if (var == QLatin1String("inlineMedia")) {
|
||
inlineMedia = val == QLatin1String("1");
|
||
}
|
||
if (var == QLatin1String("preserveAspectRatio")) {
|
||
keepAspect = val == QLatin1String("0");
|
||
}
|
||
if (var == QLatin1String("doNotMoveCursor")) {
|
||
moveCursor = val != QLatin1String("1");
|
||
}
|
||
if (var == QLatin1String("width")) {
|
||
int unitPos = val.toStdString().find_first_not_of("0123456789");
|
||
scaledWidth = QStringView(val).mid(0, unitPos).toInt();
|
||
if (unitPos == -1) {
|
||
scaledWidth *= _currentScreen->currentTerminalDisplay()->terminalFont()->fontWidth();
|
||
} else {
|
||
if (val.mid(unitPos) == QLatin1String("%")) {
|
||
scaledWidth *= _currentScreen->currentTerminalDisplay()->terminalFont()->fontWidth() * _currentScreen->getColumns() / 100;
|
||
}
|
||
}
|
||
}
|
||
if (var == QLatin1String("height")) {
|
||
int unitPos = val.toStdString().find_first_not_of("0123456789");
|
||
scaledHeight = QStringView(val).mid(0, unitPos).toInt();
|
||
if (unitPos == -1) {
|
||
scaledHeight *= _currentScreen->currentTerminalDisplay()->terminalFont()->fontHeight();
|
||
} else {
|
||
if (val.mid(unitPos) == QLatin1String("%")) {
|
||
scaledHeight *= _currentScreen->currentTerminalDisplay()->terminalFont()->fontHeight() * _currentScreen->getLines() / 100;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
if (inlineMedia) {
|
||
QTemporaryFile file;
|
||
file.open();
|
||
file.write(tokenData);
|
||
if (player == nullptr) {
|
||
player = new QMediaPlayer(this);
|
||
connect(player, SIGNAL(mediaStatusChanged(QMediaPlayer::MediaStatus)), this, SLOT(deletePlayer(QMediaPlayer::MediaStatus)));
|
||
}
|
||
player->setMedia(QUrl::fromLocalFile(file.fileName()));
|
||
player->play();
|
||
return;
|
||
}
|
||
if (!inlineImage) {
|
||
return;
|
||
}
|
||
QPixmap pixmap;
|
||
pixmap.loadFromData(tokenData);
|
||
tokenData.clear();
|
||
if (pixmap.isNull()) {
|
||
return;
|
||
}
|
||
if (scaledWidth && scaledHeight) {
|
||
pixmap = pixmap.scaled(scaledWidth, scaledHeight, (Qt::AspectRatioMode)keepAspect);
|
||
} else {
|
||
if (keepAspect && scaledWidth) {
|
||
pixmap = pixmap.scaledToWidth(scaledWidth);
|
||
} else if (keepAspect && scaledHeight) {
|
||
pixmap = pixmap.scaledToHeight(scaledHeight);
|
||
}
|
||
}
|
||
int rows = -1, cols = -1;
|
||
_currentScreen->addPlacement(pixmap, rows, cols, -1, -1, true, moveCursor);
|
||
}
|
||
_pendingSessionAttributesUpdates[attribute] = value;
|
||
_sessionAttributesUpdateTimer->start(20);
|
||
}
|
||
|
||
void Vt102Emulation::deletePlayer(QMediaPlayer::MediaStatus mediaStatus)
|
||
{
|
||
if (mediaStatus == QMediaPlayer::EndOfMedia || mediaStatus == QMediaPlayer::InvalidMedia) {
|
||
player->deleteLater();
|
||
player = nullptr;
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::updateSessionAttributes()
|
||
{
|
||
QListIterator<int> iter(_pendingSessionAttributesUpdates.keys());
|
||
while (iter.hasNext()) {
|
||
int arg = iter.next();
|
||
Q_EMIT sessionAttributeChanged(arg, _pendingSessionAttributesUpdates[arg]);
|
||
}
|
||
_pendingSessionAttributesUpdates.clear();
|
||
}
|
||
|
||
// Interpreting Codes ---------------------------------------------------------
|
||
|
||
/*
|
||
Now that the incoming character stream is properly tokenized,
|
||
meaning is assigned to them. These are either operations of
|
||
the current _screen, or of the emulation class itself.
|
||
|
||
The token to be interpreted comes in as a machine word
|
||
possibly accompanied by two parameters.
|
||
|
||
Likewise, the operations assigned to, come with up to two
|
||
arguments. One could consider to make up a proper table
|
||
from the function below.
|
||
|
||
The technical reference manual provides more information
|
||
about this mapping.
|
||
*/
|
||
|
||
void Vt102Emulation::processToken(int token, int p, int q)
|
||
{
|
||
/* clang-format off */
|
||
switch (token) {
|
||
case token_chr( ) :
|
||
_currentScreen->displayCharacter (p ); break; //UTF16
|
||
|
||
// 127 DEL : ignored on input
|
||
|
||
case token_ctl('@' ) : /* NUL: ignored */ break;
|
||
case token_ctl('A' ) : /* SOH: ignored */ break;
|
||
case token_ctl('B' ) : /* STX: ignored */ break;
|
||
case token_ctl('C' ) : /* ETX: ignored */ break;
|
||
case token_ctl('D' ) : /* EOT: ignored */ break;
|
||
case token_ctl('E' ) : reportAnswerBack ( ); break; //VT100
|
||
case token_ctl('F' ) : /* ACK: ignored */ break;
|
||
case token_ctl('G' ) : Q_EMIT bell();
|
||
break; //VT100
|
||
case token_ctl('H' ) : _currentScreen->backspace ( ); break; //VT100
|
||
case token_ctl('I' ) : _currentScreen->tab ( ); break; //VT100
|
||
case token_ctl('J' ) : _currentScreen->newLine ( ); break; //VT100
|
||
case token_ctl('K' ) : _currentScreen->newLine ( ); break; //VT100
|
||
case token_ctl('L' ) : _currentScreen->newLine ( ); break; //VT100
|
||
case token_ctl('M' ) : _currentScreen->toStartOfLine ( ); break; //VT100
|
||
|
||
case token_ctl('N' ) : useCharset ( 1); break; //VT100
|
||
case token_ctl('O' ) : useCharset ( 0); break; //VT100
|
||
|
||
case token_ctl('P' ) : /* DLE: ignored */ break;
|
||
case token_ctl('Q' ) : /* DC1: XON continue */ break; //VT100
|
||
case token_ctl('R' ) : /* DC2: ignored */ break;
|
||
case token_ctl('S' ) : /* DC3: XOFF halt */ break; //VT100
|
||
case token_ctl('T' ) : /* DC4: ignored */ break;
|
||
case token_ctl('U' ) : /* NAK: ignored */ break;
|
||
case token_ctl('V' ) : /* SYN: ignored */ break;
|
||
case token_ctl('W' ) : /* ETB: ignored */ break;
|
||
case token_ctl('X' ) : _currentScreen->displayCharacter ( 0x2592); break; //VT100
|
||
case token_ctl('Y' ) : /* EM : ignored */ break;
|
||
case token_ctl('Z' ) : _currentScreen->displayCharacter ( 0x2592); break; //VT100
|
||
case token_ctl('[' ) : /* ESC: cannot be seen here. */ break;
|
||
case token_ctl('\\' ) : /* FS : ignored */ break;
|
||
case token_ctl(']' ) : /* GS : ignored */ break;
|
||
case token_ctl('^' ) : /* RS : ignored */ break;
|
||
case token_ctl('_' ) : /* US : ignored */ break;
|
||
|
||
case token_esc('D' ) : _currentScreen->index ( ); break; //VT100
|
||
case token_esc('E' ) : _currentScreen->nextLine ( ); break; //VT100
|
||
case token_esc('H' ) : _currentScreen->changeTabStop (true ); break; //VT100
|
||
case token_esc('M' ) : _currentScreen->reverseIndex ( ); break; //VT100
|
||
case token_esc('Z' ) : reportTerminalType ( ); break;
|
||
case token_esc('c' ) : reset ( ); break;
|
||
|
||
case token_esc('n' ) : useCharset ( 2); break;
|
||
case token_esc('o' ) : useCharset ( 3); break;
|
||
case token_esc('7' ) : saveCursor ( ); break;
|
||
case token_esc('8' ) : restoreCursor ( ); break;
|
||
|
||
case token_esc('=' ) : setMode (MODE_AppKeyPad); break;
|
||
case token_esc('>' ) : resetMode (MODE_AppKeyPad); break;
|
||
case token_esc('<' ) : setMode (MODE_Ansi ); break; //VT100
|
||
|
||
case token_esc('\\' ) : resetMode (MODE_Sixel ); break;
|
||
|
||
case token_esc_cs('(', '0') : setCharset (0, '0'); break; //VT100
|
||
case token_esc_cs('(', 'A') : setCharset (0, 'A'); break; //VT100
|
||
case token_esc_cs('(', 'B') : setCharset (0, 'B'); break; //VT100
|
||
|
||
case token_esc_cs(')', '0') : setCharset (1, '0'); break; //VT100
|
||
case token_esc_cs(')', 'A') : setCharset (1, 'A'); break; //VT100
|
||
case token_esc_cs(')', 'B') : setCharset (1, 'B'); break; //VT100
|
||
|
||
case token_esc_cs('*', '0') : setCharset (2, '0'); break; //VT100
|
||
case token_esc_cs('*', 'A') : setCharset (2, 'A'); break; //VT100
|
||
case token_esc_cs('*', 'B') : setCharset (2, 'B'); break; //VT100
|
||
|
||
case token_esc_cs('+', '0') : setCharset (3, '0'); break; //VT100
|
||
case token_esc_cs('+', 'A') : setCharset (3, 'A'); break; //VT100
|
||
case token_esc_cs('+', 'B') : setCharset (3, 'B'); break; //VT100
|
||
|
||
case token_esc_cs('%', 'G') : setCodec (Utf8Codec ); break; //LINUX
|
||
case token_esc_cs('%', '@') : setCodec (LocaleCodec ); break; //LINUX
|
||
|
||
case token_esc_de('3' ) : /* Double height line, top half */
|
||
_currentScreen->setLineProperty( LINE_DOUBLEWIDTH , true );
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_TOP , true );
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_BOTTOM , false );
|
||
break;
|
||
case token_esc_de('4' ) : /* Double height line, bottom half */
|
||
_currentScreen->setLineProperty( LINE_DOUBLEWIDTH , true );
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_TOP , false );
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_BOTTOM , true );
|
||
break;
|
||
case token_esc_de('5' ) : /* Single width, single height line*/
|
||
_currentScreen->setLineProperty( LINE_DOUBLEWIDTH , false);
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_TOP , false);
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_BOTTOM , false);
|
||
break;
|
||
case token_esc_de('6' ) : /* Double width, single height line*/
|
||
_currentScreen->setLineProperty( LINE_DOUBLEWIDTH , true);
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_TOP , false);
|
||
_currentScreen->setLineProperty( LINE_DOUBLEHEIGHT_BOTTOM , false);
|
||
break;
|
||
case token_esc_de('8' ) : _currentScreen->helpAlign ( ); break;
|
||
|
||
// resize = \e[8;<rows>;<cols>t
|
||
case token_csi_ps('t', 8) : setImageSize( p /* rows */, q /* columns */ );
|
||
Q_EMIT imageResizeRequest(QSize(q, p)); // Note columns (x), rows (y) in QSize
|
||
break;
|
||
|
||
case token_csi_ps('t', 14) : reportPixelSize(); break;
|
||
case token_csi_ps('t', 16) : reportCellSize(); break;
|
||
case token_csi_ps('t', 18) : reportSize(); break;
|
||
// change tab text color : \e[28;<color>t color: 0-16,777,215
|
||
case token_csi_ps('t', 28) : /* IGNORED: konsole-specific KDE3-era extension, not implemented */ break;
|
||
|
||
case token_csi_ps('t', 22) : /* IGNORED: Save icon and window title on stack */ break; //XTERM
|
||
case token_csi_ps('t', 23) : /* IGNORED: Restore icon and window title from stack */ break; //XTERM
|
||
|
||
case token_csi_ps('K', 0) : _currentScreen->clearToEndOfLine ( ); break;
|
||
case token_csi_ps('K', 1) : _currentScreen->clearToBeginOfLine ( ); break;
|
||
case token_csi_ps('K', 2) : _currentScreen->clearEntireLine ( ); break;
|
||
case token_csi_ps('J', 0) : _currentScreen->clearToEndOfScreen ( ); break;
|
||
case token_csi_ps('J', 1) : _currentScreen->clearToBeginOfScreen ( ); break;
|
||
case token_csi_ps('J', 2) : _currentScreen->clearEntireScreen ( ); break;
|
||
case token_csi_ps('J', 3) : clearHistory(); break;
|
||
case token_csi_ps('g', 0) : _currentScreen->changeTabStop (false ); break; //VT100
|
||
case token_csi_ps('g', 3) : _currentScreen->clearTabStops ( ); break; //VT100
|
||
case token_csi_ps('h', 4) : _currentScreen-> setMode (MODE_Insert ); break;
|
||
case token_csi_ps('h', 20) : setMode (MODE_NewLine ); break;
|
||
case token_csi_ps('i', 0) : /* IGNORE: attached printer */ break; //VT100
|
||
case token_csi_ps('l', 4) : _currentScreen-> resetMode (MODE_Insert ); break;
|
||
case token_csi_ps('l', 20) : resetMode (MODE_NewLine ); break;
|
||
case token_csi_ps('s', 0) : saveCursor ( ); break;
|
||
case token_csi_ps('u', 0) : restoreCursor ( ); break;
|
||
|
||
case token_csi_ps('m', 0) : _currentScreen->setDefaultRendition ( ); break;
|
||
case token_csi_ps('m', 1) : _currentScreen-> setRendition (RE_BOLD ); break; //VT100
|
||
case token_csi_ps('m', 2) : _currentScreen-> setRendition (RE_FAINT ); break;
|
||
case token_csi_ps('m', 3) : _currentScreen-> setRendition (RE_ITALIC ); break; //VT100
|
||
case token_csi_ps('m', 4) :
|
||
if (q == 1) {
|
||
_currentScreen->setUnderlineType(p);
|
||
} else {
|
||
_currentScreen->setUnderlineType(RE_UNDERLINE);
|
||
}
|
||
break; //VT100
|
||
case token_csi_ps('m', 5) : _currentScreen-> setRendition (RE_BLINK ); break; //VT100
|
||
case token_csi_ps('m', 7) : _currentScreen-> setRendition (RE_REVERSE ); break;
|
||
case token_csi_ps('m', 8) : _currentScreen-> setRendition (RE_CONCEAL ); break;
|
||
case token_csi_ps('m', 9) : _currentScreen-> setRendition (RE_STRIKEOUT); break;
|
||
case token_csi_ps('m', 53) : _currentScreen-> setRendition (RE_OVERLINE ); break;
|
||
case token_csi_ps('m', 10) : /* IGNORED: mapping related */ break; //LINUX
|
||
case token_csi_ps('m', 11) : /* IGNORED: mapping related */ break; //LINUX
|
||
case token_csi_ps('m', 12) : /* IGNORED: mapping related */ break; //LINUX
|
||
case token_csi_ps('m', 21) : _currentScreen->setUnderlineType(RE_UNDERLINE_DOUBLE); break;
|
||
case token_csi_ps('m', 22) : _currentScreen->resetRendition (RE_BOLD );
|
||
_currentScreen->resetRendition (RE_FAINT ); break;
|
||
case token_csi_ps('m', 23) : _currentScreen->resetRendition (RE_ITALIC ); break; //VT100
|
||
case token_csi_ps('m', 24) : _currentScreen->resetRendition (RE_UNDERLINE_MASK); break;
|
||
case token_csi_ps('m', 25) : _currentScreen->resetRendition (RE_BLINK ); break;
|
||
case token_csi_ps('m', 27) : _currentScreen->resetRendition (RE_REVERSE ); break;
|
||
case token_csi_ps('m', 28) : _currentScreen->resetRendition (RE_CONCEAL ); break;
|
||
case token_csi_ps('m', 29) : _currentScreen->resetRendition (RE_STRIKEOUT); break;
|
||
case token_csi_ps('m', 55) : _currentScreen->resetRendition (RE_OVERLINE ); break;
|
||
|
||
case token_csi_ps('m', 30) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 0); break;
|
||
case token_csi_ps('m', 31) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 1); break;
|
||
case token_csi_ps('m', 32) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 2); break;
|
||
case token_csi_ps('m', 33) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 3); break;
|
||
case token_csi_ps('m', 34) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 4); break;
|
||
case token_csi_ps('m', 35) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 5); break;
|
||
case token_csi_ps('m', 36) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 6); break;
|
||
case token_csi_ps('m', 37) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 7); break;
|
||
|
||
case token_csi_ps('m', 38) : _currentScreen->setForeColor (p, q); break;
|
||
|
||
case token_csi_ps('m', 39) : _currentScreen->setForeColor (COLOR_SPACE_DEFAULT, 0); break;
|
||
|
||
case token_csi_ps('m', 40) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 0); break;
|
||
case token_csi_ps('m', 41) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 1); break;
|
||
case token_csi_ps('m', 42) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 2); break;
|
||
case token_csi_ps('m', 43) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 3); break;
|
||
case token_csi_ps('m', 44) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 4); break;
|
||
case token_csi_ps('m', 45) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 5); break;
|
||
case token_csi_ps('m', 46) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 6); break;
|
||
case token_csi_ps('m', 47) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 7); break;
|
||
|
||
case token_csi_ps('m', 48) : _currentScreen->setBackColor (p, q); break;
|
||
|
||
case token_csi_ps('m', 49) : _currentScreen->setBackColor (COLOR_SPACE_DEFAULT, 1); break;
|
||
|
||
case token_csi_ps('m', 58) : _currentScreen->setULColor (p, q); break;
|
||
|
||
case token_csi_ps('m', 59) : _currentScreen->setULColor (COLOR_SPACE_UNDEFINED, 0); break;
|
||
|
||
case token_csi_ps('m', 90) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 8); break;
|
||
case token_csi_ps('m', 91) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 9); break;
|
||
case token_csi_ps('m', 92) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 10); break;
|
||
case token_csi_ps('m', 93) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 11); break;
|
||
case token_csi_ps('m', 94) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 12); break;
|
||
case token_csi_ps('m', 95) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 13); break;
|
||
case token_csi_ps('m', 96) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 14); break;
|
||
case token_csi_ps('m', 97) : _currentScreen->setForeColor (COLOR_SPACE_SYSTEM, 15); break;
|
||
|
||
case token_csi_ps('m', 100) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 8); break;
|
||
case token_csi_ps('m', 101) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 9); break;
|
||
case token_csi_ps('m', 102) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 10); break;
|
||
case token_csi_ps('m', 103) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 11); break;
|
||
case token_csi_ps('m', 104) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 12); break;
|
||
case token_csi_ps('m', 105) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 13); break;
|
||
case token_csi_ps('m', 106) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 14); break;
|
||
case token_csi_ps('m', 107) : _currentScreen->setBackColor (COLOR_SPACE_SYSTEM, 15); break;
|
||
|
||
case token_csi_ps('n', 5) : reportStatus ( ); break;
|
||
case token_csi_ps('n', 6) : reportCursorPosition ( ); break;
|
||
case token_csi_ps('q', 0) : /* IGNORED: LEDs off */ break; //VT100
|
||
case token_csi_ps('q', 1) : /* IGNORED: LED1 on */ break; //VT100
|
||
case token_csi_ps('q', 2) : /* IGNORED: LED2 on */ break; //VT100
|
||
case token_csi_ps('q', 3) : /* IGNORED: LED3 on */ break; //VT100
|
||
case token_csi_ps('q', 4) : /* IGNORED: LED4 on */ break; //VT100
|
||
case token_csi_ps('x', 0) : reportTerminalParms ( 2); break; //VT100
|
||
case token_csi_ps('x', 1) : reportTerminalParms ( 3); break; //VT100
|
||
|
||
case token_csi_pn('@' ) : _currentScreen->insertChars (p ); break;
|
||
case token_csi_pn('A' ) : _currentScreen->cursorUp (p ); break; //VT100
|
||
case token_csi_pn('B' ) : _currentScreen->cursorDown (p ); break; //VT100
|
||
case token_csi_pn('C' ) : _currentScreen->cursorRight (p ); break; //VT100
|
||
case token_csi_pn('D' ) : _currentScreen->cursorLeft (p ); break; //VT100
|
||
case token_csi_pn('E' ) : _currentScreen->cursorNextLine (p ); break; //VT100
|
||
case token_csi_pn('F' ) : _currentScreen->cursorPreviousLine (p ); break; //VT100
|
||
case token_csi_pn('G' ) : _currentScreen->setCursorX (p ); break; //LINUX
|
||
case token_csi_pn('H' ) : _currentScreen->setCursorYX (p, q); break; //VT100
|
||
case token_csi_pn('I' ) : _currentScreen->tab (p ); break;
|
||
case token_csi_pn('L' ) : _currentScreen->insertLines (p ); break;
|
||
case token_csi_pn('M' ) : _currentScreen->deleteLines (p ); break;
|
||
case token_csi_pn('P' ) : _currentScreen->deleteChars (p ); break;
|
||
case token_csi_pn('S' ) : _currentScreen->scrollUp (p ); break;
|
||
case token_csi_pn('T' ) : _currentScreen->scrollDown (p ); break;
|
||
case token_csi_pn('X' ) : _currentScreen->eraseChars (p ); break;
|
||
case token_csi_pn('Z' ) : _currentScreen->backtab (p ); break;
|
||
case token_csi_pn('b' ) : _currentScreen->repeatChars (p ); break;
|
||
case token_csi_pn('c' ) : reportTerminalType ( ); break; //VT100
|
||
case token_csi_pn('d' ) : _currentScreen->setCursorY (p ); break; //LINUX
|
||
case token_csi_pn('f' ) : _currentScreen->setCursorYX (p, q); break; //VT100
|
||
case token_csi_pn('r' ) : setMargins (p, q); break; //VT100
|
||
case token_csi_pn('y' ) : /* IGNORED: Confidence test */ break; //VT100
|
||
|
||
case token_csi_pr('h', 1) : setMode (MODE_AppCuKeys); break; //VT100
|
||
case token_csi_pr('l', 1) : resetMode (MODE_AppCuKeys); break; //VT100
|
||
case token_csi_pr('s', 1) : saveMode (MODE_AppCuKeys); break; //FIXME
|
||
case token_csi_pr('r', 1) : restoreMode (MODE_AppCuKeys); break; //FIXME
|
||
|
||
case token_csi_pr('l', 2) : resetMode (MODE_Ansi ); break; //VT100
|
||
|
||
case token_csi_pr('h', 3) : setMode (MODE_132Columns); break; //VT100
|
||
case token_csi_pr('l', 3) : resetMode (MODE_132Columns); break; //VT100
|
||
|
||
case token_csi_pr('h', 4) : /* IGNORED: soft scrolling */ break; //VT100
|
||
case token_csi_pr('l', 4) : /* IGNORED: soft scrolling */ break; //VT100
|
||
|
||
case token_csi_pr('h', 5) : _currentScreen-> setMode (MODE_Screen ); break; //VT100
|
||
case token_csi_pr('l', 5) : _currentScreen-> resetMode (MODE_Screen ); break; //VT100
|
||
|
||
case token_csi_pr('h', 6) : _currentScreen-> setMode (MODE_Origin ); break; //VT100
|
||
case token_csi_pr('l', 6) : _currentScreen-> resetMode (MODE_Origin ); break; //VT100
|
||
case token_csi_pr('s', 6) : _currentScreen-> saveMode (MODE_Origin ); break; //FIXME
|
||
case token_csi_pr('r', 6) : _currentScreen->restoreMode (MODE_Origin ); break; //FIXME
|
||
|
||
case token_csi_pr('h', 7) : _currentScreen-> setMode (MODE_Wrap ); break; //VT100
|
||
case token_csi_pr('l', 7) : _currentScreen-> resetMode (MODE_Wrap ); break; //VT100
|
||
case token_csi_pr('s', 7) : _currentScreen-> saveMode (MODE_Wrap ); break; //FIXME
|
||
case token_csi_pr('r', 7) : _currentScreen->restoreMode (MODE_Wrap ); break; //FIXME
|
||
|
||
case token_csi_pr('h', 8) : /* IGNORED: autorepeat on */ break; //VT100
|
||
case token_csi_pr('l', 8) : /* IGNORED: autorepeat off */ break; //VT100
|
||
case token_csi_pr('s', 8) : /* IGNORED: autorepeat on */ break; //VT100
|
||
case token_csi_pr('r', 8) : /* IGNORED: autorepeat off */ break; //VT100
|
||
|
||
case token_csi_pr('h', 9) : /* IGNORED: interlace */ break; //VT100
|
||
case token_csi_pr('l', 9) : /* IGNORED: interlace */ break; //VT100
|
||
case token_csi_pr('s', 9) : /* IGNORED: interlace */ break; //VT100
|
||
case token_csi_pr('r', 9) : /* IGNORED: interlace */ break; //VT100
|
||
|
||
case token_csi_pr('h', 12) : /* IGNORED: Cursor blink */ break; //att610
|
||
case token_csi_pr('l', 12) : /* IGNORED: Cursor blink */ break; //att610
|
||
case token_csi_pr('s', 12) : /* IGNORED: Cursor blink */ break; //att610
|
||
case token_csi_pr('r', 12) : /* IGNORED: Cursor blink */ break; //att610
|
||
|
||
case token_csi_pr('h', 25) : setMode (MODE_Cursor ); break; //VT100
|
||
case token_csi_pr('l', 25) : resetMode (MODE_Cursor ); break; //VT100
|
||
case token_csi_pr('s', 25) : saveMode (MODE_Cursor ); break; //VT100
|
||
case token_csi_pr('r', 25) : restoreMode (MODE_Cursor ); break; //VT100
|
||
|
||
case token_csi_pr('h', 40) : setMode(MODE_Allow132Columns ); break; // XTERM
|
||
case token_csi_pr('l', 40) : resetMode(MODE_Allow132Columns ); break; // XTERM
|
||
|
||
case token_csi_pr('h', 41) : /* IGNORED: obsolete more(1) fix */ break; //XTERM
|
||
case token_csi_pr('l', 41) : /* IGNORED: obsolete more(1) fix */ break; //XTERM
|
||
case token_csi_pr('s', 41) : /* IGNORED: obsolete more(1) fix */ break; //XTERM
|
||
case token_csi_pr('r', 41) : /* IGNORED: obsolete more(1) fix */ break; //XTERM
|
||
|
||
case token_csi_pr('h', 47) : setMode (MODE_AppScreen); break; //VT100
|
||
case token_csi_pr('l', 47) : resetMode (MODE_AppScreen); break; //VT100
|
||
case token_csi_pr('s', 47) : saveMode (MODE_AppScreen); break; //XTERM
|
||
case token_csi_pr('r', 47) : restoreMode (MODE_AppScreen); break; //XTERM
|
||
|
||
case token_csi_pr('h', 67) : /* IGNORED: DECBKM */ break; //XTERM
|
||
case token_csi_pr('l', 67) : /* IGNORED: DECBKM */ break; //XTERM
|
||
case token_csi_pr('s', 67) : /* IGNORED: DECBKM */ break; //XTERM
|
||
case token_csi_pr('r', 67) : /* IGNORED: DECBKM */ break; //XTERM
|
||
|
||
case token_csi_pr('h', 80) : m_SixelScrolling = false; break;
|
||
case token_csi_pr('l', 80) : m_SixelScrolling = true; break;
|
||
|
||
// XTerm defines the following modes:
|
||
// SET_VT200_MOUSE 1000
|
||
// SET_VT200_HIGHLIGHT_MOUSE 1001
|
||
// SET_BTN_EVENT_MOUSE 1002
|
||
// SET_ANY_EVENT_MOUSE 1003
|
||
//
|
||
|
||
//Note about mouse modes:
|
||
//There are four mouse modes which xterm-compatible terminals can support - 1000,1001,1002,1003
|
||
//Konsole currently supports mode 1000 (basic mouse press and release), mode 1002 (dragging the mouse)
|
||
//and mode 1003 (moving the mouse).
|
||
//TODO: Implementation of mouse mode 1001 (something called highlight tracking).
|
||
//
|
||
|
||
case token_csi_pr('h', 1000) : setMode (MODE_Mouse1000); break; //XTERM
|
||
case token_csi_pr('l', 1000) : resetMode (MODE_Mouse1000); break; //XTERM
|
||
case token_csi_pr('s', 1000) : saveMode (MODE_Mouse1000); break; //XTERM
|
||
case token_csi_pr('r', 1000) : restoreMode (MODE_Mouse1000); break; //XTERM
|
||
|
||
case token_csi_pr('h', 1001) : /* IGNORED: hilite mouse tracking */ break; //XTERM
|
||
case token_csi_pr('l', 1001) : resetMode (MODE_Mouse1001); break; //XTERM
|
||
case token_csi_pr('s', 1001) : /* IGNORED: hilite mouse tracking */ break; //XTERM
|
||
case token_csi_pr('r', 1001) : /* IGNORED: hilite mouse tracking */ break; //XTERM
|
||
|
||
case token_csi_pr('h', 1002) : setMode (MODE_Mouse1002); break; //XTERM
|
||
case token_csi_pr('l', 1002) : resetMode (MODE_Mouse1002); break; //XTERM
|
||
case token_csi_pr('s', 1002) : saveMode (MODE_Mouse1002); break; //XTERM
|
||
case token_csi_pr('r', 1002) : restoreMode (MODE_Mouse1002); break; //XTERM
|
||
|
||
case token_csi_pr('h', 1003) : setMode (MODE_Mouse1003); break; //XTERM
|
||
case token_csi_pr('l', 1003) : resetMode (MODE_Mouse1003); break; //XTERM
|
||
case token_csi_pr('s', 1003) : saveMode (MODE_Mouse1003); break; //XTERM
|
||
case token_csi_pr('r', 1003) : restoreMode (MODE_Mouse1003); break; //XTERM
|
||
|
||
case token_csi_pr('h', 1004) : _reportFocusEvents = true; break;
|
||
case token_csi_pr('l', 1004) : _reportFocusEvents = false; break;
|
||
|
||
case token_csi_pr('h', 1005) : setMode (MODE_Mouse1005); break; //XTERM
|
||
case token_csi_pr('l', 1005) : resetMode (MODE_Mouse1005); break; //XTERM
|
||
case token_csi_pr('s', 1005) : saveMode (MODE_Mouse1005); break; //XTERM
|
||
case token_csi_pr('r', 1005) : restoreMode (MODE_Mouse1005); break; //XTERM
|
||
|
||
case token_csi_pr('h', 1006) : setMode (MODE_Mouse1006); break; //XTERM
|
||
case token_csi_pr('l', 1006) : resetMode (MODE_Mouse1006); break; //XTERM
|
||
case token_csi_pr('s', 1006) : saveMode (MODE_Mouse1006); break; //XTERM
|
||
case token_csi_pr('r', 1006) : restoreMode (MODE_Mouse1006); break; //XTERM
|
||
|
||
case token_csi_pr('h', 1007) : setMode (MODE_Mouse1007); break; //XTERM
|
||
case token_csi_pr('l', 1007) : resetMode (MODE_Mouse1007); break; //XTERM
|
||
case token_csi_pr('s', 1007) : saveMode (MODE_Mouse1007); break; //XTERM
|
||
case token_csi_pr('r', 1007) : restoreMode (MODE_Mouse1007); break; //XTERM
|
||
|
||
case token_csi_pr('h', 1015) : setMode (MODE_Mouse1015); break; //URXVT
|
||
case token_csi_pr('l', 1015) : resetMode (MODE_Mouse1015); break; //URXVT
|
||
case token_csi_pr('s', 1015) : saveMode (MODE_Mouse1015); break; //URXVT
|
||
case token_csi_pr('r', 1015) : restoreMode (MODE_Mouse1015); break; //URXVT
|
||
|
||
case token_csi_pr('h', 1034) : /* IGNORED: 8bitinput activation */ break; //XTERM
|
||
|
||
case token_csi_pr('h', 1047) : setMode (MODE_AppScreen); break; //XTERM
|
||
case token_csi_pr('l', 1047) : _screen[1]->clearEntireScreen(); resetMode(MODE_AppScreen); break; //XTERM
|
||
case token_csi_pr('s', 1047) : saveMode (MODE_AppScreen); break; //XTERM
|
||
case token_csi_pr('r', 1047) : restoreMode (MODE_AppScreen); break; //XTERM
|
||
|
||
//FIXME: Unitoken: save translations
|
||
case token_csi_pr('h', 1048) : saveCursor ( ); break; //XTERM
|
||
case token_csi_pr('l', 1048) : restoreCursor ( ); break; //XTERM
|
||
case token_csi_pr('s', 1048) : saveCursor ( ); break; //XTERM
|
||
case token_csi_pr('r', 1048) : restoreCursor ( ); break; //XTERM
|
||
|
||
//FIXME: every once new sequences like this pop up in xterm.
|
||
// Here's a guess of what they could mean.
|
||
case token_csi_pr('h', 1049) : saveCursor(); _screen[1]->clearEntireScreen(); setMode(MODE_AppScreen); break; //XTERM
|
||
case token_csi_pr('l', 1049) : resetMode(MODE_AppScreen); restoreCursor(); break; //XTERM
|
||
|
||
case token_csi_pr('h', 2004) : setMode (MODE_BracketedPaste); break; //XTERM
|
||
case token_csi_pr('l', 2004) : resetMode (MODE_BracketedPaste); break; //XTERM
|
||
case token_csi_pr('s', 2004) : saveMode (MODE_BracketedPaste); break; //XTERM
|
||
case token_csi_pr('r', 2004) : restoreMode (MODE_BracketedPaste); break; //XTERM
|
||
|
||
case token_csi_pr('S', 1) : if(!p) sixelQuery (1 ); break;
|
||
case token_csi_pr('S', 2) : if(!p) sixelQuery (2 ); break;
|
||
// Set Cursor Style (DECSCUSR), VT520, with the extra xterm sequences
|
||
// the first one is a special case, 'ESC[ q', which mimics 'ESC[1 q'
|
||
// Using 0 to reset to default is matching VTE, but not any official standard.
|
||
case token_csi_sp ('q' ) : Q_EMIT setCursorStyleRequest(Enum::BlockCursor, true); break;
|
||
case token_csi_psp('q', 0) : Q_EMIT resetCursorStyleRequest(); break;
|
||
case token_csi_psp('q', 1) : Q_EMIT setCursorStyleRequest(Enum::BlockCursor, true); break;
|
||
case token_csi_psp('q', 2) : Q_EMIT setCursorStyleRequest(Enum::BlockCursor, false); break;
|
||
case token_csi_psp('q', 3) : Q_EMIT setCursorStyleRequest(Enum::UnderlineCursor, true); break;
|
||
case token_csi_psp('q', 4) : Q_EMIT setCursorStyleRequest(Enum::UnderlineCursor, false); break;
|
||
case token_csi_psp('q', 5) : Q_EMIT setCursorStyleRequest(Enum::IBeamCursor, true); break;
|
||
case token_csi_psp('q', 6) : Q_EMIT setCursorStyleRequest(Enum::IBeamCursor, false); break;
|
||
|
||
// DECSTR (Soft Terminal Reset)
|
||
case token_csi_pe('p' ) : reset(true); break; //VT220
|
||
|
||
case token_csi_pq('c' ) : reportTertiaryAttributes( ); break; //VT420
|
||
case token_csi_pg('c' ) : reportSecondaryAttributes( ); break; //VT100
|
||
case token_csi_pg('q' ) : reportVersion( ); break;
|
||
|
||
//FIXME: when changing between vt52 and ansi mode evtl do some resetting.
|
||
case token_vt52('A' ) : _currentScreen->cursorUp ( 1); break; //VT52
|
||
case token_vt52('B' ) : _currentScreen->cursorDown ( 1); break; //VT52
|
||
case token_vt52('C' ) : _currentScreen->cursorRight ( 1); break; //VT52
|
||
case token_vt52('D' ) : _currentScreen->cursorLeft ( 1); break; //VT52
|
||
|
||
case token_vt52('F' ) : setAndUseCharset (0, '0'); break; //VT52
|
||
case token_vt52('G' ) : setAndUseCharset (0, 'B'); break; //VT52
|
||
|
||
case token_vt52('H' ) : _currentScreen->setCursorYX (1,1 ); break; //VT52
|
||
case token_vt52('I' ) : _currentScreen->reverseIndex ( ); break; //VT52
|
||
case token_vt52('J' ) : _currentScreen->clearToEndOfScreen ( ); break; //VT52
|
||
case token_vt52('K' ) : _currentScreen->clearToEndOfLine ( ); break; //VT52
|
||
case token_vt52('Y' ) : _currentScreen->setCursorYX (p-31,q-31 ); break; //VT52
|
||
case token_vt52('Z' ) : reportTerminalType ( ); break; //VT52
|
||
case token_vt52('<' ) : setMode (MODE_Ansi ); break; //VT52
|
||
case token_vt52('=' ) : setMode (MODE_AppKeyPad); break; //VT52
|
||
case token_vt52('>' ) : resetMode (MODE_AppKeyPad); break; //VT52
|
||
|
||
default:
|
||
reportDecodingError(token);
|
||
break;
|
||
}
|
||
/* clang-format on */
|
||
}
|
||
|
||
void delete_func(void *p)
|
||
{
|
||
delete (QByteArray *)p;
|
||
}
|
||
|
||
void Vt102Emulation::processGraphicsToken(int tokenSize)
|
||
{
|
||
QString value = QString::fromUcs4(&tokenBuffer[1], tokenSize - 1);
|
||
QStringList list;
|
||
QPixmap pixmap;
|
||
|
||
int dataPos = value.indexOf(QLatin1Char(';'));
|
||
if (dataPos == -1) {
|
||
dataPos = value.size();
|
||
}
|
||
if (dataPos > 1024) {
|
||
reportDecodingError(token_apc('G'));
|
||
return;
|
||
}
|
||
list = value.mid(0, dataPos).split(QLatin1Char(','));
|
||
|
||
QMap<char, qint64> keys; // Keys may be signed or unsigned 32 bit integers
|
||
|
||
if (savedKeys.empty()) {
|
||
keys['a'] = 't';
|
||
keys['t'] = 'd';
|
||
keys['q'] = 0;
|
||
keys['m'] = 0;
|
||
keys['f'] = 32;
|
||
keys['i'] = 0;
|
||
keys['o'] = 0;
|
||
keys['X'] = 0;
|
||
keys['Y'] = 0;
|
||
keys['x'] = 0;
|
||
keys['y'] = 0;
|
||
keys['z'] = 0;
|
||
keys['C'] = 0;
|
||
keys['c'] = 0;
|
||
keys['r'] = 0;
|
||
keys['A'] = 255;
|
||
keys['I'] = 0;
|
||
keys['d'] = 'a';
|
||
keys['p'] = -1;
|
||
} else {
|
||
keys = QMap<char, qint64>(savedKeys);
|
||
}
|
||
|
||
for (int i = 0; i < list.size(); i++) {
|
||
if (list.at(i).at(1).toLatin1() != '=') {
|
||
reportDecodingError(token_apc('G'));
|
||
return;
|
||
}
|
||
if (list.at(i).at(2).isNumber() || list.at(i).at(2).toLatin1() == '-') {
|
||
keys[list.at(i).at(0).toLatin1()] = QStringView(list.at(i)).mid(2).toInt();
|
||
} else {
|
||
keys[list.at(i).at(0).toLatin1()] = list.at(i).at(2).toLatin1();
|
||
}
|
||
}
|
||
|
||
if (keys['a'] == 't' || keys['a'] == 'T' || keys['a'] == 'q') {
|
||
if (keys['q'] < 2 && keys['t'] != 'd') {
|
||
QString params = QStringLiteral("i=") + QString::number(keys['i']);
|
||
QString error = QStringLiteral("ENOTSUPPORTED:");
|
||
sendGraphicsReply(params, error);
|
||
return;
|
||
}
|
||
if (keys['I']) {
|
||
keys['i'] = getFreeGraphicsImageId();
|
||
}
|
||
if (imageId != keys['i']) {
|
||
imageId = keys['i'];
|
||
imageData.clear();
|
||
}
|
||
imageData.append(tokenData);
|
||
tokenData.clear();
|
||
imageData.append(QByteArray::fromBase64(value.mid(dataPos + 1).toLocal8Bit()));
|
||
if (keys['m'] == 0) {
|
||
imageId = 0;
|
||
savedKeys = QMap<char, qint64>();
|
||
QByteArray out;
|
||
|
||
uint32_t byteCount = 0;
|
||
if (keys['f'] == 24 || keys['f'] == 32) {
|
||
int bpp = keys['f'] / 8;
|
||
byteCount = bpp * keys['s'] * keys['v'];
|
||
} else {
|
||
byteCount = 8 * 1024 * 1024;
|
||
}
|
||
|
||
if (keys['o'] == 'z') {
|
||
char header[sizeof byteCount];
|
||
qToBigEndian(byteCount, header);
|
||
imageData.prepend(header, sizeof header);
|
||
out = qUncompress(imageData);
|
||
|
||
if (keys['f'] != 24 && keys['f'] != 32) {
|
||
imageData = out;
|
||
}
|
||
}
|
||
if (out.isEmpty()) {
|
||
out = imageData;
|
||
}
|
||
|
||
if (keys['f'] == 24 || keys['f'] == 32) {
|
||
if (unsigned(out.size()) < byteCount) {
|
||
qCWarning(KonsoleDebug) << "Not enough image data" << out.size() << "require" << byteCount;
|
||
imageData.clear();
|
||
return;
|
||
}
|
||
QImage::Format format = keys['f'] == 24 ? QImage::Format_RGB888 : QImage::Format_RGBA8888;
|
||
pixmap = QPixmap::fromImage(QImage((const uchar *)out.constData(), 0 + keys['s'], 0 + keys['v'], 0 + keys['s'] * keys['f'] / 8, format));
|
||
pixmap.detach();
|
||
} else {
|
||
pixmap.loadFromData(out);
|
||
}
|
||
|
||
if (keys['a'] == 'q') {
|
||
QString params = QStringLiteral("i=") + QString::number(keys['i']);
|
||
sendGraphicsReply(params, QString());
|
||
} else {
|
||
if (keys['i']) {
|
||
_graphicsImages[keys['i']] = pixmap;
|
||
}
|
||
if (keys['q'] == 0 && keys['a'] == 't') {
|
||
QString params = QStringLiteral("i=") + QString::number(keys['i']);
|
||
if (keys['I']) {
|
||
params = params + QStringLiteral(",I=") + QString::number(keys['I']);
|
||
}
|
||
sendGraphicsReply(params, QString());
|
||
}
|
||
}
|
||
imageData.clear();
|
||
} else {
|
||
if (savedKeys.empty()) {
|
||
savedKeys = QMap<char, qint64>(keys);
|
||
savedKeys.remove('m');
|
||
}
|
||
}
|
||
}
|
||
if (keys['a'] == 'p' || (keys['a'] == 'T' && keys['m'] == 0)) {
|
||
if (keys['a'] == 'p') {
|
||
pixmap = _graphicsImages[keys['i']];
|
||
}
|
||
if (!pixmap.isNull()) {
|
||
if (keys['x'] || keys['y'] || keys['w'] || keys['h']) {
|
||
int w = keys['w'] ? keys['w'] : pixmap.width() - keys['x'];
|
||
int h = keys['h'] ? keys['h'] : pixmap.height() - keys['y'];
|
||
pixmap = pixmap.copy(keys['x'], keys['y'], w, h);
|
||
}
|
||
if (keys['c'] && keys['r']) {
|
||
pixmap = pixmap.scaled(keys['c'] * _currentScreen->currentTerminalDisplay()->terminalFont()->fontWidth(),
|
||
keys['r'] * _currentScreen->currentTerminalDisplay()->terminalFont()->fontHeight());
|
||
}
|
||
int rows = -1, cols = -1;
|
||
_currentScreen->addPlacement(pixmap,
|
||
rows,
|
||
cols,
|
||
-1,
|
||
-1,
|
||
true,
|
||
keys['C'] == 0,
|
||
true,
|
||
keys['z'],
|
||
keys['i'],
|
||
keys['p'],
|
||
keys['A'] / 255.0,
|
||
keys['X'],
|
||
keys['Y']);
|
||
if (keys['q'] == 0 && keys['i']) {
|
||
QString params = QStringLiteral("i=") + QString::number(keys['i']);
|
||
if (keys['I']) {
|
||
params = params + QStringLiteral(",I=") + QString::number(keys['I']);
|
||
}
|
||
if (keys['p'] >= 0) {
|
||
params = params + QStringLiteral(",p=") + QString::number(keys['p']);
|
||
}
|
||
sendGraphicsReply(params, QString());
|
||
}
|
||
} else {
|
||
if (keys['q'] < 2) {
|
||
QString params = QStringLiteral("i=") + QString::number(keys['i']);
|
||
sendGraphicsReply(params, QStringLiteral("ENOENT:No such image"));
|
||
}
|
||
}
|
||
}
|
||
if (keys['a'] == 'd') {
|
||
int action = keys['d'] | 0x20;
|
||
int id = keys['i'];
|
||
int pid = keys['p'];
|
||
int x = keys['x'];
|
||
int y = keys['y'];
|
||
if (action == 'n') {
|
||
} else if (action == 'c') {
|
||
action = 'p';
|
||
x = _currentScreen->getCursorX();
|
||
y = _currentScreen->getCursorY();
|
||
}
|
||
_currentScreen->delPlacements(action, id, pid, x, y, keys['z']);
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::clearScreenAndSetColumns(int columnCount)
|
||
{
|
||
setImageSize(_currentScreen->getLines(), columnCount);
|
||
clearEntireScreen();
|
||
setDefaultMargins();
|
||
_currentScreen->setCursorYX(0, 0);
|
||
}
|
||
|
||
void Vt102Emulation::sendString(const QByteArray &s)
|
||
{
|
||
Q_EMIT sendData(s);
|
||
}
|
||
|
||
void Vt102Emulation::sendGraphicsReply(const QString ¶ms, const QString &error)
|
||
{
|
||
sendString(
|
||
(QStringLiteral("\033_G") + params + QStringLiteral(";") + (error.isEmpty() ? QStringLiteral("OK") : error) + QStringLiteral("\033\\")).toLatin1());
|
||
}
|
||
|
||
void Vt102Emulation::reportCursorPosition()
|
||
{
|
||
char tmp[30];
|
||
int y = _currentScreen->getCursorY() + 1;
|
||
int x = _currentScreen->getCursorX() + 1;
|
||
if (_currentScreen->getMode(MODE_Origin)) {
|
||
y -= _currentScreen->topMargin();
|
||
}
|
||
snprintf(tmp, sizeof(tmp), "\033[%d;%dR", y, x);
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::reportPixelSize()
|
||
{
|
||
char tmp[30];
|
||
snprintf(tmp,
|
||
sizeof(tmp),
|
||
"\033[4;%d;%dt",
|
||
_currentScreen->currentTerminalDisplay()->terminalFont()->fontHeight() * _currentScreen->getLines(),
|
||
_currentScreen->currentTerminalDisplay()->terminalFont()->fontWidth() * _currentScreen->getColumns());
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::iTermReportCellSize()
|
||
{
|
||
char tmp[50];
|
||
snprintf(tmp,
|
||
sizeof(tmp),
|
||
"\033]1337;ReportCellSize=%d.0;%d.0;1.0\007",
|
||
_currentScreen->currentTerminalDisplay()->terminalFont()->fontHeight(),
|
||
_currentScreen->currentTerminalDisplay()->terminalFont()->fontWidth());
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::reportCellSize()
|
||
{
|
||
char tmp[30];
|
||
snprintf(tmp,
|
||
sizeof(tmp),
|
||
"\033[6;%d;%dt",
|
||
_currentScreen->currentTerminalDisplay()->terminalFont()->fontHeight(),
|
||
_currentScreen->currentTerminalDisplay()->terminalFont()->fontWidth());
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::reportColor(int c, QColor color)
|
||
{
|
||
char tmp[60];
|
||
snprintf(tmp,
|
||
sizeof(tmp),
|
||
"\033]4;%i;rgb:%02x%02x/%02x%02x/%02x%02x\007",
|
||
c,
|
||
color.red(),
|
||
color.red(),
|
||
color.green(),
|
||
color.green(),
|
||
color.blue(),
|
||
color.blue());
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::reportSize()
|
||
{
|
||
char tmp[30];
|
||
snprintf(tmp, sizeof(tmp), "\033[8;%d;%dt", _currentScreen->getLines(), _currentScreen->getColumns());
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::reportTerminalType()
|
||
{
|
||
// Primary device attribute response (Request was: ^[[0c or ^[[c (from TT321 Users Guide))
|
||
// VT220: ^[[?63;1;2;3;6;7;8c (list deps on emul. capabilities)
|
||
// VT100: ^[[?1;2c
|
||
// VT101: ^[[?1;0c
|
||
// VT102: ^[[?6v
|
||
if (getMode(MODE_Ansi)) {
|
||
sendString("\033[?62;1;4c"); // I'm a VT2xx with 132 columns and Sixel
|
||
} else {
|
||
sendString("\033/Z"); // I'm a VT52
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::reportTertiaryAttributes()
|
||
{
|
||
// Tertiary device attribute response DECRPTUI (Request was: ^[[=0c or ^[[=c)
|
||
// 7E4B4445 is hex for ASCII "~KDE"
|
||
sendString("\033P!|7E4B4445\033\\");
|
||
}
|
||
|
||
void Vt102Emulation::reportSecondaryAttributes()
|
||
{
|
||
// Secondary device attribute response (Request was: ^[[>0c or ^[[>c)
|
||
if (getMode(MODE_Ansi)) {
|
||
sendString("\033[>1;115;0c"); // Why 115? ;)
|
||
} else {
|
||
sendString("\033/Z"); // FIXME I don't think VT52 knows about it but kept for
|
||
}
|
||
// konsoles backward compatibility.
|
||
}
|
||
|
||
void Vt102Emulation::reportVersion()
|
||
{
|
||
sendString("\033P>|Konsole " KONSOLE_VERSION "\033\\");
|
||
}
|
||
|
||
/* DECREPTPARM – Report Terminal Parameters
|
||
ESC [ <sol>; <par>; <nbits>; <xspeed>; <rspeed>; <clkmul>; <flags> x
|
||
|
||
https://vt100.net/docs/vt100-ug/chapter3.html
|
||
*/
|
||
void Vt102Emulation::reportTerminalParms(int p)
|
||
{
|
||
char tmp[100];
|
||
/*
|
||
sol=1: This message is a request; report in response to a request.
|
||
par=1: No parity set
|
||
nbits=1: 8 bits per character
|
||
xspeed=112: 9600
|
||
rspeed=112: 9600
|
||
clkmul=1: The bit rate multiplier is 16.
|
||
flags=0: None
|
||
*/
|
||
snprintf(tmp, sizeof(tmp), "\033[%d;1;1;112;112;1;0x", p); // not really true.
|
||
sendString(tmp);
|
||
}
|
||
|
||
void Vt102Emulation::reportStatus()
|
||
{
|
||
sendString("\033[0n"); // VT100. Device status report. 0 = Ready.
|
||
}
|
||
|
||
void Vt102Emulation::reportAnswerBack()
|
||
{
|
||
// FIXME - Test this with VTTEST
|
||
// This is really obsolete VT100 stuff.
|
||
const char *ANSWER_BACK = "";
|
||
sendString(ANSWER_BACK);
|
||
}
|
||
|
||
/*!
|
||
`cx',`cy' are 1-based.
|
||
`cb' indicates the button pressed or released (0-2) or scroll event (64-65) plus key modifiers (Alt=8, Control=16).
|
||
|
||
eventType represents the kind of mouse action that occurred:
|
||
0 = Mouse button press
|
||
1 = Mouse drag
|
||
2 = Mouse button release
|
||
3 = Mouse click to move cursor in input field
|
||
*/
|
||
|
||
void Vt102Emulation::sendMouseEvent(int cb, int cx, int cy, int eventType)
|
||
{
|
||
if (cx < 1 || cy < 1) {
|
||
return;
|
||
}
|
||
|
||
if (eventType == 3) {
|
||
// We know we are in input mode
|
||
TerminalDisplay *currentView = _currentScreen->currentTerminalDisplay();
|
||
bool isReadOnly = false;
|
||
if (currentView != nullptr) {
|
||
isReadOnly = currentView->getReadOnly();
|
||
}
|
||
auto point = std::make_pair(cy, cx);
|
||
if (!isReadOnly && _currentScreen->replModeStart() <= point && point <= _currentScreen->replModeEnd()) {
|
||
KeyboardTranslator::States states = KeyboardTranslator::NoState;
|
||
|
||
// get current states
|
||
if (getMode(MODE_NewLine)) {
|
||
states |= KeyboardTranslator::NewLineState;
|
||
}
|
||
if (getMode(MODE_Ansi)) {
|
||
states |= KeyboardTranslator::AnsiState;
|
||
}
|
||
if (getMode(MODE_AppCuKeys)) {
|
||
states |= KeyboardTranslator::CursorKeysState;
|
||
}
|
||
if (getMode(MODE_AppScreen)) {
|
||
states |= KeyboardTranslator::AlternateScreenState;
|
||
}
|
||
KeyboardTranslator::Entry LRKeys[2] = {_keyTranslator->findEntry(Qt::Key_Left, Qt::NoModifier, states),
|
||
_keyTranslator->findEntry(Qt::Key_Right, Qt::NoModifier, states)};
|
||
QVector<LineProperty> lineProperties = _currentScreen->getLineProperties(cy + _currentScreen->getHistLines(), cy + _currentScreen->getHistLines());
|
||
cx = qMin(cx, (int)lineProperties[0].length);
|
||
int cuX = _currentScreen->getCursorX();
|
||
int cuY = _currentScreen->getCursorY();
|
||
QByteArray textToSend;
|
||
if (cuY != cy) {
|
||
for (int i = abs(cy - cuY); i > 0; i--) {
|
||
emulateUpDown(cy < cuY, LRKeys[cy > cuY], textToSend, i == 1 ? cx : -1);
|
||
textToSend += LRKeys[cy > cuY].text();
|
||
}
|
||
} else {
|
||
if (cuX < cx) {
|
||
for (int i = 0; i < cx - cuX; i++) {
|
||
textToSend += LRKeys[1].text();
|
||
}
|
||
} else {
|
||
for (int i = 0; i < cuX - cx; i++) {
|
||
textToSend += LRKeys[0].text();
|
||
}
|
||
}
|
||
}
|
||
Q_EMIT sendData(textToSend);
|
||
}
|
||
return;
|
||
}
|
||
|
||
// Don't send move/drag events if only press and release requested
|
||
if (eventType == 1 && getMode(MODE_Mouse1000)) {
|
||
return;
|
||
}
|
||
|
||
// Don't send move with no button pressed if button-motion requested
|
||
if ((cb & 3) == 3 && getMode(MODE_Mouse1002)) {
|
||
return;
|
||
}
|
||
|
||
// With the exception of the 1006 mode, button release is encoded in cb.
|
||
// Note that if multiple extensions are enabled, the 1006 is used, so it's okay to check for only that.
|
||
if (eventType == 2 && !getMode(MODE_Mouse1006)) {
|
||
cb &= ~3;
|
||
cb |= 3;
|
||
}
|
||
|
||
// Mouse motion handling
|
||
if ((getMode(MODE_Mouse1002) || getMode(MODE_Mouse1003)) && eventType == 1) {
|
||
cb += 0x20; // add 32 to signify motion event
|
||
}
|
||
char command[40];
|
||
command[0] = '\0';
|
||
// Check the extensions in decreasing order of preference. Encoding the release event above assumes that 1006 comes first.
|
||
if (getMode(MODE_Mouse1006)) {
|
||
snprintf(command, sizeof(command), "\033[<%d;%d;%d%c", cb, cx, cy, eventType == 2 ? 'm' : 'M');
|
||
} else if (getMode(MODE_Mouse1015)) {
|
||
snprintf(command, sizeof(command), "\033[%d;%d;%dM", cb + 0x20, cx, cy);
|
||
} else if (getMode(MODE_Mouse1005)) {
|
||
if (cx <= 2015 && cy <= 2015) {
|
||
// The xterm extension uses UTF-8 (up to 2 bytes) to encode
|
||
// coordinate+32, no matter what the locale is. We could easily
|
||
// convert manually, but QString can also do it for us.
|
||
QChar coords[2];
|
||
coords[0] = QChar(cx + 0x20);
|
||
coords[1] = QChar(cy + 0x20);
|
||
QString coordsStr = QString(coords, 2);
|
||
QByteArray utf8 = coordsStr.toUtf8();
|
||
snprintf(command, sizeof(command), "\033[M%c%s", cb + 0x20, utf8.constData());
|
||
}
|
||
} else if (cx <= 223 && cy <= 223) {
|
||
snprintf(command, sizeof(command), "\033[M%c%c%c", cb + 0x20, cx + 0x20, cy + 0x20);
|
||
}
|
||
|
||
sendString(command);
|
||
}
|
||
|
||
void Vt102Emulation::emulateUpDown(bool up, KeyboardTranslator::Entry entry, QByteArray &textToSend, int toCol)
|
||
{
|
||
int cuX = _currentScreen->getCursorX();
|
||
int cuY = _currentScreen->getCursorY();
|
||
int realX = cuX;
|
||
QVector<LineProperty> lineProperties = _currentScreen->getLineProperties(cuY - 1 + _currentScreen->getHistLines(),
|
||
qMin(_currentScreen->getLines() - 1, cuY + 1) + _currentScreen->getHistLines());
|
||
int num = _currentScreen->getColumns();
|
||
if (up) {
|
||
if ((lineProperties[0].flags.f.wrapped) == 0) {
|
||
num = cuX + qMax(0, lineProperties[0].length - cuX) + 1;
|
||
}
|
||
} else {
|
||
if ((lineProperties[1].flags.f.wrapped) == 0 || (lineProperties[2].flags.f.wrapped) == 0) {
|
||
realX = qMin(cuX, lineProperties[2].length + 1);
|
||
num = lineProperties[1].length - cuX + realX;
|
||
}
|
||
}
|
||
if (toCol > -1) {
|
||
num += up ? realX - toCol : toCol - realX;
|
||
}
|
||
for (int i = 1; i < num; i++) {
|
||
// One more will be added by the rest of the code.
|
||
textToSend += entry.text();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* The focus change event can be used by Vim (or other terminal applications)
|
||
* to recognize that the konsole window has changed focus.
|
||
* The escape sequence is also used by iTerm2.
|
||
* Vim needs the following plugin to be installed to convert the escape
|
||
* sequence into the FocusLost/FocusGained autocmd:
|
||
* https://github.com/sjl/vitality.vim
|
||
*/
|
||
void Vt102Emulation::focusChanged(bool focused)
|
||
{
|
||
if (_reportFocusEvents) {
|
||
sendString(focused ? "\033[I" : "\033[O");
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::sendText(const QString &text)
|
||
{
|
||
if (!text.isEmpty()) {
|
||
QKeyEvent event(QEvent::KeyPress, 0, Qt::NoModifier, text);
|
||
sendKeyEvent(&event); // expose as a big fat keypress event
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::sendKeyEvent(QKeyEvent *event)
|
||
{
|
||
const Qt::KeyboardModifiers modifiers = event->modifiers();
|
||
KeyboardTranslator::States states = KeyboardTranslator::NoState;
|
||
|
||
TerminalDisplay *currentView = _currentScreen->currentTerminalDisplay();
|
||
bool isReadOnly = false;
|
||
if (currentView != nullptr) {
|
||
isReadOnly = currentView->getReadOnly();
|
||
}
|
||
|
||
// get current states
|
||
if (getMode(MODE_NewLine)) {
|
||
states |= KeyboardTranslator::NewLineState;
|
||
}
|
||
if (getMode(MODE_Ansi)) {
|
||
states |= KeyboardTranslator::AnsiState;
|
||
}
|
||
if (getMode(MODE_AppCuKeys)) {
|
||
states |= KeyboardTranslator::CursorKeysState;
|
||
}
|
||
if (getMode(MODE_AppScreen)) {
|
||
states |= KeyboardTranslator::AlternateScreenState;
|
||
}
|
||
if (getMode(MODE_AppKeyPad) && ((modifiers & Qt::KeypadModifier) != 0U)) {
|
||
states |= KeyboardTranslator::ApplicationKeypadState;
|
||
}
|
||
|
||
if (!isReadOnly) {
|
||
// check flow control state
|
||
if ((modifiers & Qt::ControlModifier) != 0U) {
|
||
switch (event->key()) {
|
||
case Qt::Key_S:
|
||
Q_EMIT flowControlKeyPressed(true);
|
||
break;
|
||
case Qt::Key_C:
|
||
if (m_SixelStarted) {
|
||
SixelModeAbort();
|
||
}
|
||
|
||
// Allow the user to take back control
|
||
resetTokenizer();
|
||
Q_EMIT flowControlKeyPressed(false);
|
||
break;
|
||
case Qt::Key_Q: // cancel flow control
|
||
Q_EMIT flowControlKeyPressed(false);
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
// look up key binding
|
||
if (_keyTranslator != nullptr) {
|
||
KeyboardTranslator::Entry entry = _keyTranslator->findEntry(event->key(), modifiers, states);
|
||
// send result to terminal
|
||
QByteArray textToSend;
|
||
|
||
int cuX = _currentScreen->getCursorX();
|
||
int cuY = _currentScreen->getCursorY();
|
||
if ((event->key() == Qt::Key_Up || event->key() == Qt::Key_Down) && _currentScreen->replMode() == REPL_INPUT
|
||
&& _currentScreen->currentTerminalDisplay()->semanticUpDown()) {
|
||
bool up = event->key() == Qt::Key_Up;
|
||
if ((up && _currentScreen->replModeStart() <= std::make_pair(cuY - 1, cuX))
|
||
|| (!up && std::make_pair(cuY + 1, cuX) <= _currentScreen->replModeEnd())) {
|
||
entry = _keyTranslator->findEntry(up ? Qt::Key_Left : Qt::Key_Right, Qt::NoModifier, states);
|
||
emulateUpDown(up, entry, textToSend);
|
||
}
|
||
}
|
||
|
||
// special handling for the Alt (aka. Meta) modifier. pressing
|
||
// Alt+[Character] results in Esc+[Character] being sent
|
||
// (unless there is an entry defined for this particular combination
|
||
// in the keyboard modifier)
|
||
const bool wantsAltModifier = ((entry.modifiers() & entry.modifierMask() & Qt::AltModifier) != 0U);
|
||
const bool wantsMetaModifier = ((entry.modifiers() & entry.modifierMask() & Qt::MetaModifier) != 0U);
|
||
const bool wantsAnyModifier = ((entry.state() & entry.stateMask() & KeyboardTranslator::AnyModifierState) != 0);
|
||
|
||
if (((modifiers & Qt::AltModifier) != 0U) && !(wantsAltModifier || wantsAnyModifier) && !event->text().isEmpty()) {
|
||
textToSend.prepend("\033");
|
||
}
|
||
if (((modifiers & Qt::MetaModifier) != 0U) && !(wantsMetaModifier || wantsAnyModifier) && !event->text().isEmpty()) {
|
||
textToSend.prepend("\030@s");
|
||
}
|
||
|
||
if (entry.command() != KeyboardTranslator::NoCommand) {
|
||
if ((entry.command() & KeyboardTranslator::EraseCommand) != 0) {
|
||
textToSend += eraseChar();
|
||
}
|
||
if (currentView != nullptr) {
|
||
if ((entry.command() & KeyboardTranslator::ScrollPageUpCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollPages, -1);
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollPageDownCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollPages, 1);
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollLineUpCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollLines, -1);
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollLineDownCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollLines, 1);
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollUpToTopCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollLines, -currentView->screenWindow()->currentLine());
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollDownToBottomCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollLines, lineCount());
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollPromptUpCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollPrompts, -1);
|
||
} else if ((entry.command() & KeyboardTranslator::ScrollPromptDownCommand) != 0) {
|
||
currentView->scrollScreenWindow(ScreenWindow::ScrollPrompts, 1);
|
||
}
|
||
}
|
||
} else if (!entry.text().isEmpty()) {
|
||
textToSend += entry.text(true, modifiers);
|
||
} else {
|
||
Q_ASSERT(_codec);
|
||
textToSend += _codec->fromUnicode(event->text());
|
||
}
|
||
|
||
if (!isReadOnly) {
|
||
Q_EMIT sendData(textToSend);
|
||
}
|
||
} else {
|
||
if (!isReadOnly) {
|
||
// print an error message to the terminal if no key translator has been
|
||
// set
|
||
QString translatorError = i18n(
|
||
"No keyboard translator available. "
|
||
"The information needed to convert key presses "
|
||
"into characters to send to the terminal "
|
||
"is missing.");
|
||
reset();
|
||
receiveData(translatorError.toLatin1().constData(), translatorError.count());
|
||
}
|
||
}
|
||
}
|
||
|
||
/* ------------------------------------------------------------------------- */
|
||
/* */
|
||
/* VT100 Charsets */
|
||
/* */
|
||
/* ------------------------------------------------------------------------- */
|
||
|
||
// Character Set Conversion ------------------------------------------------ --
|
||
|
||
/*
|
||
The processing contains a VT100 specific code translation layer.
|
||
It's still in use and mainly responsible for the line drawing graphics.
|
||
|
||
These and some other glyphs are assigned to codes (0x5f-0xfe)
|
||
normally occupied by the latin letters. Since this codes also
|
||
appear within control sequences, the extra code conversion
|
||
does not permute with the tokenizer and is placed behind it
|
||
in the pipeline. It only applies to tokens, which represent
|
||
plain characters.
|
||
|
||
This conversion it eventually continued in TerminalDisplay.C, since
|
||
it might involve VT100 enhanced fonts, which have these
|
||
particular glyphs allocated in (0x00-0x1f) in their code page.
|
||
*/
|
||
|
||
#define CHARSET _charset[_currentScreen == _screen[1]]
|
||
|
||
// Apply current character map.
|
||
|
||
unsigned int Vt102Emulation::applyCharset(uint c)
|
||
{
|
||
if (CHARSET.graphic && 0x5f <= c && c <= 0x7e) {
|
||
return vt100_graphics[c - 0x5f];
|
||
}
|
||
if (CHARSET.pound && c == '#') {
|
||
return 0xa3; // This mode is obsolete
|
||
}
|
||
return c;
|
||
}
|
||
|
||
/*
|
||
"Charset" related part of the emulation state.
|
||
This configures the VT100 charset filter.
|
||
|
||
While most operation work on the current _screen,
|
||
the following two are different.
|
||
*/
|
||
|
||
void Vt102Emulation::resetCharset(int scrno)
|
||
{
|
||
_charset[scrno].cu_cs = 0;
|
||
qstrncpy(_charset[scrno].charset, "BBBB", 4);
|
||
_charset[scrno].sa_graphic = false;
|
||
_charset[scrno].sa_pound = false;
|
||
_charset[scrno].graphic = false;
|
||
_charset[scrno].pound = false;
|
||
}
|
||
|
||
void Vt102Emulation::setCharset(int n, int cs) // on both screens.
|
||
{
|
||
_charset[0].charset[n & 3] = cs;
|
||
useCharset(_charset[0].cu_cs);
|
||
_charset[1].charset[n & 3] = cs;
|
||
useCharset(_charset[1].cu_cs);
|
||
}
|
||
|
||
void Vt102Emulation::setAndUseCharset(int n, int cs)
|
||
{
|
||
CHARSET.charset[n & 3] = cs;
|
||
useCharset(n & 3);
|
||
}
|
||
|
||
void Vt102Emulation::useCharset(int n)
|
||
{
|
||
CHARSET.cu_cs = n & 3;
|
||
CHARSET.graphic = (CHARSET.charset[n & 3] == '0');
|
||
CHARSET.pound = (CHARSET.charset[n & 3] == 'A'); // This mode is obsolete
|
||
}
|
||
|
||
void Vt102Emulation::setDefaultMargins()
|
||
{
|
||
_screen[0]->setDefaultMargins();
|
||
_screen[1]->setDefaultMargins();
|
||
}
|
||
|
||
void Vt102Emulation::setMargins(int t, int b)
|
||
{
|
||
_screen[0]->setMargins(t, b);
|
||
_screen[1]->setMargins(t, b);
|
||
}
|
||
|
||
void Vt102Emulation::saveCursor()
|
||
{
|
||
CHARSET.sa_graphic = CHARSET.graphic;
|
||
CHARSET.sa_pound = CHARSET.pound; // This mode is obsolete
|
||
// we are not clear about these
|
||
// sa_charset = charsets[cScreen->_charset];
|
||
// sa_charset_num = cScreen->_charset;
|
||
_currentScreen->saveCursor();
|
||
}
|
||
|
||
void Vt102Emulation::restoreCursor()
|
||
{
|
||
CHARSET.graphic = CHARSET.sa_graphic;
|
||
CHARSET.pound = CHARSET.sa_pound; // This mode is obsolete
|
||
_currentScreen->restoreCursor();
|
||
}
|
||
|
||
/* ------------------------------------------------------------------------- */
|
||
/* */
|
||
/* Mode Operations */
|
||
/* */
|
||
/* ------------------------------------------------------------------------- */
|
||
|
||
/*
|
||
Some of the emulations state is either added to the state of the screens.
|
||
|
||
This causes some scoping problems, since different emulations choose to
|
||
located the mode either to the current _screen or to both.
|
||
|
||
For strange reasons, the extend of the rendition attributes ranges over
|
||
all screens and not over the actual _screen.
|
||
|
||
We decided on the precise precise extend, somehow.
|
||
*/
|
||
|
||
// "Mode" related part of the state. These are all booleans.
|
||
|
||
void Vt102Emulation::resetModes()
|
||
{
|
||
// MODE_Allow132Columns is not reset here
|
||
// to match Xterm's behavior (see Xterm's VTReset() function)
|
||
|
||
// MODE_Mouse1007 (Alternate Scrolling) is not reset here, to maintain
|
||
// the profile alternate scrolling property after reset() is called, which
|
||
// makes more sense; also this matches XTerm behavior.
|
||
|
||
resetMode(MODE_132Columns);
|
||
saveMode(MODE_132Columns);
|
||
resetMode(MODE_Mouse1000);
|
||
saveMode(MODE_Mouse1000);
|
||
resetMode(MODE_Mouse1001);
|
||
saveMode(MODE_Mouse1001);
|
||
resetMode(MODE_Mouse1002);
|
||
saveMode(MODE_Mouse1002);
|
||
resetMode(MODE_Mouse1003);
|
||
saveMode(MODE_Mouse1003);
|
||
resetMode(MODE_Mouse1005);
|
||
saveMode(MODE_Mouse1005);
|
||
resetMode(MODE_Mouse1006);
|
||
saveMode(MODE_Mouse1006);
|
||
resetMode(MODE_Mouse1015);
|
||
saveMode(MODE_Mouse1015);
|
||
resetMode(MODE_BracketedPaste);
|
||
saveMode(MODE_BracketedPaste);
|
||
|
||
resetMode(MODE_AppScreen);
|
||
saveMode(MODE_AppScreen);
|
||
resetMode(MODE_AppCuKeys);
|
||
saveMode(MODE_AppCuKeys);
|
||
resetMode(MODE_AppKeyPad);
|
||
saveMode(MODE_AppKeyPad);
|
||
resetMode(MODE_NewLine);
|
||
setMode(MODE_Ansi);
|
||
}
|
||
|
||
void Vt102Emulation::setMode(int m)
|
||
{
|
||
_currentModes.mode[m] = true;
|
||
switch (m) {
|
||
case MODE_132Columns:
|
||
if (getMode(MODE_Allow132Columns)) {
|
||
clearScreenAndSetColumns(132);
|
||
} else {
|
||
_currentModes.mode[m] = false;
|
||
}
|
||
break;
|
||
case MODE_Mouse1000:
|
||
case MODE_Mouse1001:
|
||
case MODE_Mouse1002:
|
||
case MODE_Mouse1003:
|
||
_currentModes.mode[MODE_Mouse1000] = false;
|
||
_currentModes.mode[MODE_Mouse1001] = false;
|
||
_currentModes.mode[MODE_Mouse1002] = false;
|
||
_currentModes.mode[MODE_Mouse1003] = false;
|
||
_currentModes.mode[m] = true;
|
||
Q_EMIT programRequestsMouseTracking(true);
|
||
break;
|
||
case MODE_Mouse1007:
|
||
Q_EMIT enableAlternateScrolling(true);
|
||
break;
|
||
case MODE_Mouse1005:
|
||
case MODE_Mouse1006:
|
||
case MODE_Mouse1015:
|
||
_currentModes.mode[MODE_Mouse1005] = false;
|
||
_currentModes.mode[MODE_Mouse1006] = false;
|
||
_currentModes.mode[MODE_Mouse1015] = false;
|
||
_currentModes.mode[m] = true;
|
||
break;
|
||
|
||
case MODE_BracketedPaste:
|
||
Q_EMIT programBracketedPasteModeChanged(true);
|
||
break;
|
||
|
||
case MODE_AppScreen:
|
||
_screen[1]->setDefaultRendition();
|
||
_screen[1]->clearSelection();
|
||
setScreen(1);
|
||
if (_currentScreen && _currentScreen->currentTerminalDisplay()) {
|
||
_currentScreen->delPlacements(1);
|
||
_currentScreen->currentTerminalDisplay()->update();
|
||
}
|
||
break;
|
||
}
|
||
// FIXME: Currently this has a redundant condition as MODES_SCREEN is 6
|
||
// and MODE_NewLine is 5
|
||
if (m < MODES_SCREEN || m == MODE_NewLine) {
|
||
_screen[0]->setMode(m);
|
||
_screen[1]->setMode(m);
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::resetMode(int m)
|
||
{
|
||
_currentModes.mode[m] = false;
|
||
switch (m) {
|
||
case MODE_132Columns:
|
||
if (getMode(MODE_Allow132Columns)) {
|
||
clearScreenAndSetColumns(80);
|
||
}
|
||
break;
|
||
case MODE_Mouse1000:
|
||
case MODE_Mouse1001:
|
||
case MODE_Mouse1002:
|
||
case MODE_Mouse1003:
|
||
// Same behavior as xterm, these modes are mutually exclusive,
|
||
// and disabling any disables mouse tracking.
|
||
_currentModes.mode[MODE_Mouse1000] = false;
|
||
_currentModes.mode[MODE_Mouse1001] = false;
|
||
_currentModes.mode[MODE_Mouse1002] = false;
|
||
_currentModes.mode[MODE_Mouse1003] = false;
|
||
Q_EMIT programRequestsMouseTracking(false);
|
||
break;
|
||
case MODE_Mouse1007:
|
||
Q_EMIT enableAlternateScrolling(false);
|
||
break;
|
||
|
||
case MODE_BracketedPaste:
|
||
Q_EMIT programBracketedPasteModeChanged(false);
|
||
break;
|
||
|
||
case MODE_AppScreen:
|
||
_screen[0]->clearSelection();
|
||
setScreen(0);
|
||
if (_currentScreen && _currentScreen->currentTerminalDisplay()) {
|
||
_currentScreen->currentTerminalDisplay()->update();
|
||
}
|
||
break;
|
||
}
|
||
// FIXME: Currently this has a redundant condition as MODES_SCREEN is 7
|
||
// MODE_AppScreen is 6 and MODE_NewLine is 5
|
||
if (m < MODES_SCREEN || m == MODE_NewLine) {
|
||
_screen[0]->resetMode(m);
|
||
_screen[1]->resetMode(m);
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::saveMode(int m)
|
||
{
|
||
_savedModes.mode[m] = _currentModes.mode[m];
|
||
}
|
||
|
||
void Vt102Emulation::restoreMode(int m)
|
||
{
|
||
if (_savedModes.mode[m]) {
|
||
setMode(m);
|
||
} else {
|
||
resetMode(m);
|
||
}
|
||
}
|
||
|
||
bool Vt102Emulation::getMode(int m)
|
||
{
|
||
return _currentModes.mode[m];
|
||
}
|
||
|
||
char Vt102Emulation::eraseChar() const
|
||
{
|
||
KeyboardTranslator::Entry entry = _keyTranslator->findEntry(Qt::Key_Backspace, Qt::NoModifier, KeyboardTranslator::NoState);
|
||
if (entry.text().count() > 0) {
|
||
return entry.text().at(0);
|
||
} else {
|
||
return '\b';
|
||
}
|
||
}
|
||
|
||
// return contents of the scan buffer
|
||
static QString hexdump2(uint *s, int len)
|
||
{
|
||
int i;
|
||
char dump[128];
|
||
QString returnDump;
|
||
|
||
for (i = 0; i < len; i++) {
|
||
if (s[i] == '\\') {
|
||
snprintf(dump, sizeof(dump), "%s", "\\\\");
|
||
} else if ((s[i]) > 32 && s[i] < 127) {
|
||
snprintf(dump, sizeof(dump), "%c", s[i]);
|
||
} else if (s[i] == 0x1b) {
|
||
snprintf(dump, sizeof(dump), "ESC");
|
||
} else {
|
||
snprintf(dump, sizeof(dump), "\\%04x(hex)", s[i]);
|
||
}
|
||
returnDump.append(QLatin1String(dump));
|
||
}
|
||
return returnDump;
|
||
}
|
||
|
||
void Vt102Emulation::reportDecodingError(int token)
|
||
{
|
||
QString outputError = QStringLiteral("Undecodable sequence: ");
|
||
|
||
switch (token & 0xff) {
|
||
case 2:
|
||
case 3:
|
||
case 4:
|
||
outputError.append(QStringLiteral("ESC "));
|
||
break;
|
||
case 5:
|
||
case 6:
|
||
case 7:
|
||
case 9:
|
||
case 10:
|
||
case 11:
|
||
case 12:
|
||
case 13:
|
||
outputError.append(QStringLiteral("CSI "));
|
||
break;
|
||
case 14:
|
||
outputError.append(QStringLiteral("OSC "));
|
||
break;
|
||
case 15:
|
||
outputError.append(QStringLiteral("APC "));
|
||
break;
|
||
}
|
||
if ((token & 0xff) != 8) { // VT52
|
||
outputError.append(hexdump2(tokenBuffer, tokenBufferPos));
|
||
} else {
|
||
outputError.append(QStringLiteral("(VT52) ESC"));
|
||
}
|
||
|
||
if ((token & 0xff) != 3) { // SCS
|
||
outputError.append(QLatin1Char((token >> 8) & 0xff));
|
||
} else {
|
||
outputError.append(QLatin1Char((token >> 16) & 0xff));
|
||
}
|
||
|
||
qCDebug(KonsoleDebug).noquote() << outputError;
|
||
|
||
resetTokenizer();
|
||
|
||
if (m_SixelStarted) {
|
||
SixelModeAbort();
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::sixelQuery(int q)
|
||
{
|
||
char tmp[30];
|
||
if (q == 1) {
|
||
if (params.value[1] == 1 || params.value[1] == 4) {
|
||
snprintf(tmp, sizeof(tmp), "\033[?1;0;%dS", MAX_SIXEL_COLORS);
|
||
sendString(tmp);
|
||
}
|
||
}
|
||
if (q == 2) {
|
||
if (params.value[1] == 1 || params.value[1] == 4) {
|
||
snprintf(tmp, sizeof(tmp), "\033[?2;0;%d;%dS", MAX_IMAGE_DIM, MAX_IMAGE_DIM);
|
||
sendString(tmp);
|
||
}
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::SixelModeEnable(int width, int height)
|
||
{
|
||
if (width <= 0 || height <= 0) {
|
||
return;
|
||
}
|
||
if (width > MAX_IMAGE_DIM) {
|
||
width = MAX_IMAGE_DIM;
|
||
}
|
||
if (height > MAX_IMAGE_DIM) {
|
||
height = MAX_IMAGE_DIM;
|
||
}
|
||
m_actualSize = QSize(width, height);
|
||
|
||
// We assume square pixels because I'm lazy and didn't implement the stuff in vt102emulation.
|
||
int charactersHeight = height / 6 + 1;
|
||
|
||
m_currentImage = QImage(width, charactersHeight * 6 + 1, QImage::Format_Indexed8);
|
||
m_currentColor = 3;
|
||
m_currentX = 0;
|
||
m_verticalPosition = 0;
|
||
if (!m_currentImage.isNull()) {
|
||
m_SixelStarted = true;
|
||
}
|
||
m_currentImage.fill(0);
|
||
std::string initial_colors[16] = {"#000000",
|
||
"#3333CC",
|
||
"#CC2323",
|
||
"#33CC33",
|
||
"#CC33CC",
|
||
"#33CCCC",
|
||
"#CCCC33",
|
||
"#777777",
|
||
"#444444",
|
||
"#565699",
|
||
"#994444",
|
||
"#569956",
|
||
"#995699",
|
||
"#569999",
|
||
"#999956",
|
||
"#CCCCCC"};
|
||
for (int i = 0; i < 16; i++) {
|
||
m_currentImage.setColor(i, QColor(initial_colors[i].c_str()).rgb());
|
||
}
|
||
}
|
||
|
||
void Vt102Emulation::SixelModeAbort()
|
||
{
|
||
if (!m_SixelStarted) {
|
||
return;
|
||
}
|
||
resetMode(MODE_Sixel);
|
||
resetTokenizer();
|
||
m_SixelStarted = false;
|
||
m_currentImage = QImage();
|
||
}
|
||
|
||
void Vt102Emulation::SixelModeDisable()
|
||
{
|
||
if (!m_SixelStarted) {
|
||
return;
|
||
}
|
||
m_SixelStarted = false;
|
||
int col, row;
|
||
if (m_SixelScrolling) {
|
||
col = _currentScreen->getCursorX();
|
||
row = _currentScreen->getCursorY();
|
||
} else {
|
||
col = 0;
|
||
row = 0;
|
||
}
|
||
QPixmap pixmap = QPixmap::fromImage(m_currentImage.copy(QRect(0, 0, m_actualSize.width(), m_actualSize.height())));
|
||
if (m_aspect.first != m_aspect.second) {
|
||
pixmap = pixmap.scaled(pixmap.width(), m_aspect.first * pixmap.height() / m_aspect.second);
|
||
}
|
||
int rows = -1, cols = -1;
|
||
_currentScreen->addPlacement(pixmap, rows, cols, row, col, m_SixelScrolling, m_SixelScrolling * 2, false);
|
||
}
|
||
|
||
void Vt102Emulation::SixelColorChangeRGB(const int index, int red, int green, int blue)
|
||
{
|
||
if (index < 0 || index >= MAX_SIXEL_COLORS) {
|
||
return;
|
||
}
|
||
|
||
red = red * 255 / 100;
|
||
green = green * 255 / 100;
|
||
blue = blue * 255 / 100;
|
||
|
||
// QImage automatically handles the size of the color table
|
||
m_currentImage.setColor(index, qRgb(red, green, blue));
|
||
m_currentColor = index;
|
||
}
|
||
|
||
void Vt102Emulation::SixelColorChangeHSL(const int index, int hue, int saturation, int value)
|
||
{
|
||
if (index < 0 || index >= MAX_SIXEL_COLORS) {
|
||
return;
|
||
}
|
||
|
||
hue = qBound(0, hue, 360);
|
||
saturation = qBound(0, saturation, 100);
|
||
value = qBound(0, value, 100);
|
||
|
||
// libsixel is offset by 240 degrees, so we assume that is correct
|
||
hue = (hue + 240) % 360;
|
||
|
||
saturation = saturation * 255 / 100;
|
||
value = value * 255 / 100;
|
||
|
||
m_currentImage.setColor(index, QColor::fromHsl(hue, saturation, value).rgb());
|
||
m_currentColor = index;
|
||
}
|
||
|
||
void Vt102Emulation::SixelCharacterAdd(uint8_t character, int repeat)
|
||
{
|
||
if (!m_SixelStarted) {
|
||
return;
|
||
}
|
||
|
||
switch (character) {
|
||
case '\r':
|
||
m_currentX = 0;
|
||
return;
|
||
case '\n':
|
||
m_verticalPosition++;
|
||
return;
|
||
default:
|
||
break;
|
||
}
|
||
character -= '?';
|
||
const int top = m_verticalPosition * 6;
|
||
const int bottom = (m_verticalPosition + 1) * 6;
|
||
if (bottom > MAX_IMAGE_DIM) { // Ignore lines below MAX_IMAGE_DIM
|
||
return;
|
||
}
|
||
repeat = std::clamp(repeat, 1, MAX_IMAGE_DIM - m_currentX); // Won't repeat beyond MAX_IMAGE_DIM
|
||
|
||
if (bottom >= m_currentImage.height() - 1 || m_currentX + repeat >= m_currentImage.width()) {
|
||
// If we copy out of bounds it gets filled with 0
|
||
int extraWidth = 255 + repeat; // Increase size by at least 256, to avoid increasing for every pixel
|
||
int newWidth = qMax(m_currentX + extraWidth, m_currentImage.width() + extraWidth);
|
||
int newHeight = (qMax(bottom + 256, m_currentImage.height() + 256) / 6 + 1) * 6;
|
||
newWidth = qMin(newWidth, MAX_IMAGE_DIM);
|
||
newHeight = qMin(newHeight, MAX_IMAGE_DIM);
|
||
if (newWidth != m_currentImage.width() || newHeight != m_currentImage.height()) {
|
||
m_currentImage = m_currentImage.copy(0, 0, newWidth, newHeight);
|
||
}
|
||
if (m_currentImage.isNull()) {
|
||
m_SixelStarted = false;
|
||
return;
|
||
}
|
||
}
|
||
|
||
const ptrdiff_t bpl = m_currentImage.bytesPerLine();
|
||
uchar *data = m_currentImage.bits() + top * bpl + m_currentX;
|
||
|
||
if (repeat == 1) {
|
||
if (m_preserveBackground) {
|
||
// Konsole is built without _any_ optimizations by default, so a little
|
||
// manual unrolling to avoid calling shift for every loop iteration.
|
||
if (character & (1 << 0)) {
|
||
data[0 * bpl] = m_currentColor;
|
||
}
|
||
if (character & (1 << 1)) {
|
||
data[1 * bpl] = m_currentColor;
|
||
}
|
||
if (character & (1 << 2)) {
|
||
data[2 * bpl] = m_currentColor;
|
||
}
|
||
if (character & (1 << 3)) {
|
||
data[3 * bpl] = m_currentColor;
|
||
}
|
||
if (character & (1 << 4)) {
|
||
data[4 * bpl] = m_currentColor;
|
||
}
|
||
if (character & (1 << 5)) {
|
||
data[5 * bpl] = m_currentColor;
|
||
}
|
||
} else {
|
||
for (int i = 0; i < 6; i++, data += bpl) {
|
||
*data = (character & (1 << i)) * m_currentColor;
|
||
}
|
||
}
|
||
m_currentX++;
|
||
} else {
|
||
if (m_preserveBackground) {
|
||
// Konsole is built without _any_ optimizations by default, so a little
|
||
// manual unrolling to avoid calling shift for every loop iteration.
|
||
if (character & (1 << 0)) {
|
||
memset(&data[0 * bpl], m_currentColor, repeat);
|
||
}
|
||
if (character & (1 << 1)) {
|
||
memset(&data[1 * bpl], m_currentColor, repeat);
|
||
}
|
||
if (character & (1 << 2)) {
|
||
memset(&data[2 * bpl], m_currentColor, repeat);
|
||
}
|
||
if (character & (1 << 3)) {
|
||
memset(&data[3 * bpl], m_currentColor, repeat);
|
||
}
|
||
if (character & (1 << 4)) {
|
||
memset(&data[4 * bpl], m_currentColor, repeat);
|
||
}
|
||
if (character & (1 << 5)) {
|
||
memset(&data[5 * bpl], m_currentColor, repeat);
|
||
}
|
||
} else {
|
||
for (int i = 0; i < 6; i++, data += bpl) {
|
||
memset(data, (character & (1 << i)) * m_currentColor, repeat);
|
||
}
|
||
}
|
||
m_currentX += repeat;
|
||
}
|
||
if (m_currentX > m_actualSize.width()) {
|
||
m_actualSize.setWidth(m_currentX);
|
||
}
|
||
if (bottom > m_actualSize.height()) {
|
||
m_actualSize.setHeight(bottom);
|
||
}
|
||
}
|
||
|
||
bool Vt102Emulation::processSixel(uint cc)
|
||
{
|
||
switch (cc) {
|
||
case '$':
|
||
SixelCharacterAdd('\r');
|
||
resetTokenizer();
|
||
return true;
|
||
case '-':
|
||
SixelCharacterAdd('\r');
|
||
SixelCharacterAdd('\n');
|
||
resetTokenizer();
|
||
return true;
|
||
default:
|
||
break;
|
||
}
|
||
uint *s = tokenBuffer;
|
||
const int p = tokenBufferPos;
|
||
|
||
if (!m_SixelStarted && (sixel() || s[0] == '!' || s[0] == '#')) {
|
||
m_aspect = qMakePair(1, 1);
|
||
SixelModeEnable(30, 6);
|
||
}
|
||
if (sixel()) {
|
||
SixelCharacterAdd(uint8_t(cc));
|
||
resetTokenizer();
|
||
return true;
|
||
}
|
||
if (ccc(DIG)) {
|
||
addDigit(cc - '0');
|
||
return true;
|
||
}
|
||
if (cc == ';') {
|
||
addArgument();
|
||
return true;
|
||
}
|
||
|
||
if (s[0] == '"') {
|
||
if (p < 3) {
|
||
return true;
|
||
}
|
||
addArgument();
|
||
|
||
if (params.count == 4 || params.count == 2) {
|
||
// We just ignore the pixel aspect ratio, it's dumb
|
||
// const int pixelWidth = params.value[0];
|
||
// const int pixelHeight = params.value[1];
|
||
|
||
if (!m_SixelStarted) {
|
||
if (params.value[1] == 0 || params.value[0] == 0) {
|
||
m_aspect = qMakePair(1, 1);
|
||
} else {
|
||
m_aspect = qMakePair(params.value[0], params.value[1]);
|
||
}
|
||
int width;
|
||
int height;
|
||
if (params.count == 4) {
|
||
width = params.value[2];
|
||
height = params.value[3];
|
||
} else {
|
||
// Default size
|
||
width = 8;
|
||
height = 6;
|
||
}
|
||
SixelModeEnable(width, height);
|
||
}
|
||
resetTokenizer();
|
||
receiveChars(QVector<uint>{cc}); // re-send the actual character
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
// Repeat character
|
||
if (s[0] == '!') {
|
||
if (p < 2) {
|
||
return true;
|
||
}
|
||
if (ces(DIG)) {
|
||
addDigit(cc - '0');
|
||
return true;
|
||
}
|
||
|
||
SixelCharacterAdd(cc, params.value[0]);
|
||
resetTokenizer();
|
||
return true;
|
||
}
|
||
|
||
if (s[0] == '#') {
|
||
if (p < 2) {
|
||
return true;
|
||
}
|
||
addArgument();
|
||
if (params.count < 1) {
|
||
return false;
|
||
}
|
||
const int index = params.value[0];
|
||
if (params.count == 5) {
|
||
const int colorspace = params.value[1];
|
||
switch (colorspace) {
|
||
case 1:
|
||
// Confusingly it is in HLS order...
|
||
SixelColorChangeHSL(index, params.value[2], params.value[4], params.value[3]);
|
||
break;
|
||
case 2:
|
||
SixelColorChangeRGB(index, params.value[2], params.value[3], params.value[4]);
|
||
break;
|
||
default:
|
||
return false;
|
||
}
|
||
} else if (params.count == 1 && index >= 0) { // Negative index is an error. Too large index is ignored
|
||
if (index < MAX_SIXEL_COLORS) {
|
||
m_currentColor = index;
|
||
}
|
||
} else {
|
||
return false;
|
||
}
|
||
resetTokenizer();
|
||
receiveChars(QVector<uint>{cc}); // re-send the actual character
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
int Vt102Emulation::getFreeGraphicsImageId()
|
||
{
|
||
int i = 1;
|
||
while (1) {
|
||
if (!_graphicsImages.count(i)) {
|
||
return i;
|
||
}
|
||
i++;
|
||
}
|
||
}
|