Terminal: Add mouse support

Change-Id: Ibeb8e13b5f8f75f16ec86f64536235587c844ffc
Reviewed-by: Cristian Adam <cristian.adam@qt.io>
This commit is contained in:
Marcus Tillmanns
2023-07-25 09:16:48 +02:00
parent 75d9dc2d31
commit 79e839b29f
17 changed files with 211 additions and 112 deletions

View File

@@ -12,6 +12,76 @@
#define READ_INTERVAL_MSEC 500 #define READ_INTERVAL_MSEC 500
//ConPTY is available only on Windows 10 released after 1903 (19H1) Windows release
class WindowsContext
{
private:
WindowsContext() {}
public:
typedef HRESULT (*CreatePseudoConsolePtr)(
COORD size, // ConPty Dimensions
HANDLE hInput, // ConPty Input
HANDLE hOutput, // ConPty Output
DWORD dwFlags, // ConPty Flags
HPCON* phPC); // ConPty Reference
typedef HRESULT (*ResizePseudoConsolePtr)(HPCON hPC, COORD size);
typedef VOID (*ClosePseudoConsolePtr)(HPCON hPC);
static WindowsContext &instance()
{
static WindowsContext ctx;
return ctx;
}
bool init()
{
//already initialized
if (createPseudoConsole)
return true;
//try to load symbols from library
//if it fails -> we can't use ConPty API
HANDLE kernel32Handle = LoadLibraryExW(L"kernel32.dll", 0, 0);
if (kernel32Handle != nullptr)
{
createPseudoConsole = (CreatePseudoConsolePtr)GetProcAddress((HMODULE)kernel32Handle, "CreatePseudoConsole");
resizePseudoConsole = (ResizePseudoConsolePtr)GetProcAddress((HMODULE)kernel32Handle, "ResizePseudoConsole");
closePseudoConsole = (ClosePseudoConsolePtr)GetProcAddress((HMODULE)kernel32Handle, "ClosePseudoConsole");
if (createPseudoConsole == NULL || resizePseudoConsole == NULL || closePseudoConsole == NULL)
{
m_lastError = QString("WindowsContext/ConPty error: %1").arg("Invalid on load API functions");
return false;
}
}
else
{
m_lastError = QString("WindowsContext/ConPty error: %1").arg("Unable to load kernel32");
return false;
}
return true;
}
QString lastError()
{
return m_lastError;
}
public:
//vars
CreatePseudoConsolePtr createPseudoConsole{nullptr};
ResizePseudoConsolePtr resizePseudoConsole{nullptr};
ClosePseudoConsolePtr closePseudoConsole{nullptr};
private:
QString m_lastError;
};
HRESULT ConPtyProcess::createPseudoConsoleAndPipes(HPCON* phPC, HANDLE* phPipeIn, HANDLE* phPipeOut, qint16 cols, qint16 rows) HRESULT ConPtyProcess::createPseudoConsoleAndPipes(HPCON* phPC, HANDLE* phPipeIn, HANDLE* phPipeOut, qint16 cols, qint16 rows)
{ {
HRESULT hr{ E_UNEXPECTED }; HRESULT hr{ E_UNEXPECTED };
@@ -23,7 +93,7 @@ HRESULT ConPtyProcess::createPseudoConsoleAndPipes(HPCON* phPC, HANDLE* phPipeIn
CreatePipe(phPipeIn, &hPipePTYOut, NULL, 0)) CreatePipe(phPipeIn, &hPipePTYOut, NULL, 0))
{ {
// Create the Pseudo Console of the required size, attached to the PTY-end of the pipes // Create the Pseudo Console of the required size, attached to the PTY-end of the pipes
hr = m_winContext.createPseudoConsole({cols, rows}, hPipePTYIn, hPipePTYOut, 0, phPC); hr = WindowsContext::instance().createPseudoConsole({cols, rows}, hPipePTYIn, hPipePTYOut, 0, phPC);
// Note: We can close the handles to the PTY-end of the pipes here // Note: We can close the handles to the PTY-end of the pipes here
// because the handles are dup'ed into the ConHost and will be released // because the handles are dup'ed into the ConHost and will be released
@@ -108,7 +178,7 @@ bool ConPtyProcess::startProcess(const QString &executable,
qint16 rows) qint16 rows)
{ {
if (!isAvailable()) { if (!isAvailable()) {
m_lastError = m_winContext.lastError(); m_lastError = WindowsContext::instance().lastError();
return false; return false;
} }
@@ -228,7 +298,7 @@ bool ConPtyProcess::resize(qint16 cols, qint16 rows)
return false; return false;
} }
bool res = SUCCEEDED(m_winContext.resizePseudoConsole(m_ptyHandler, {cols, rows})); bool res = SUCCEEDED(WindowsContext::instance().resizePseudoConsole(m_ptyHandler, {cols, rows}));
if (res) if (res)
{ {
@@ -248,7 +318,7 @@ bool ConPtyProcess::kill()
m_aboutToDestruct = true; m_aboutToDestruct = true;
// Close ConPTY - this will terminate client process if running // Close ConPTY - this will terminate client process if running
m_winContext.closePseudoConsole(m_ptyHandler); WindowsContext::instance().closePseudoConsole(m_ptyHandler);
// Clean-up the pipes // Clean-up the pipes
if (INVALID_HANDLE_VALUE != m_hPipeOut) if (INVALID_HANDLE_VALUE != m_hPipeOut)
@@ -334,7 +404,7 @@ bool ConPtyProcess::isAvailable()
qint32 buildNumber = QSysInfo::kernelVersion().split(".").last().toInt(); qint32 buildNumber = QSysInfo::kernelVersion().split(".").last().toInt();
if (buildNumber < CONPTY_MINIMAL_WINDOWS_VERSION) if (buildNumber < CONPTY_MINIMAL_WINDOWS_VERSION)
return false; return false;
return m_winContext.init(); return WindowsContext::instance().init();
} }
void ConPtyProcess::moveToThread(QThread *targetThread) void ConPtyProcess::moveToThread(QThread *targetThread)

View File

@@ -25,80 +25,6 @@ typedef VOID* HPCON;
class QWinEventNotifier; class QWinEventNotifier;
template <typename T>
std::vector<T> vectorFromString(const std::basic_string<T> &str)
{
return std::vector<T>(str.begin(), str.end());
}
//ConPTY available only on Windows 10 releazed after 1903 (19H1) Windows release
class WindowsContext
{
public:
typedef HRESULT (*CreatePseudoConsolePtr)(
COORD size, // ConPty Dimensions
HANDLE hInput, // ConPty Input
HANDLE hOutput, // ConPty Output
DWORD dwFlags, // ConPty Flags
HPCON* phPC); // ConPty Reference
typedef HRESULT (*ResizePseudoConsolePtr)(HPCON hPC, COORD size);
typedef VOID (*ClosePseudoConsolePtr)(HPCON hPC);
WindowsContext()
: createPseudoConsole(nullptr)
, resizePseudoConsole(nullptr)
, closePseudoConsole(nullptr)
{
}
bool init()
{
//already initialized
if (createPseudoConsole)
return true;
//try to load symbols from library
//if it fails -> we can't use ConPty API
HANDLE kernel32Handle = LoadLibraryExW(L"kernel32.dll", 0, 0);
if (kernel32Handle != nullptr)
{
createPseudoConsole = (CreatePseudoConsolePtr)GetProcAddress((HMODULE)kernel32Handle, "CreatePseudoConsole");
resizePseudoConsole = (ResizePseudoConsolePtr)GetProcAddress((HMODULE)kernel32Handle, "ResizePseudoConsole");
closePseudoConsole = (ClosePseudoConsolePtr)GetProcAddress((HMODULE)kernel32Handle, "ClosePseudoConsole");
if (createPseudoConsole == NULL || resizePseudoConsole == NULL || closePseudoConsole == NULL)
{
m_lastError = QString("WindowsContext/ConPty error: %1").arg("Invalid on load API functions");
return false;
}
}
else
{
m_lastError = QString("WindowsContext/ConPty error: %1").arg("Unable to load kernel32");
return false;
}
return true;
}
QString lastError()
{
return m_lastError;
}
public:
//vars
CreatePseudoConsolePtr createPseudoConsole;
ResizePseudoConsolePtr resizePseudoConsole;
ClosePseudoConsolePtr closePseudoConsole;
private:
QString m_lastError;
};
class PtyBuffer : public QIODevice class PtyBuffer : public QIODevice
{ {
friend class ConPtyProcess; friend class ConPtyProcess;
@@ -141,7 +67,7 @@ public:
virtual QIODevice *notifier(); virtual QIODevice *notifier();
virtual QByteArray readAll(); virtual QByteArray readAll();
virtual qint64 write(const QByteArray &byteArray); virtual qint64 write(const QByteArray &byteArray);
bool isAvailable(); static bool isAvailable();
void moveToThread(QThread *targetThread); void moveToThread(QThread *targetThread);
private: private:
@@ -149,7 +75,6 @@ private:
HRESULT initializeStartupInfoAttachedToPseudoConsole(STARTUPINFOEX* pStartupInfo, HPCON hPC); HRESULT initializeStartupInfoAttachedToPseudoConsole(STARTUPINFOEX* pStartupInfo, HPCON hPC);
private: private:
WindowsContext m_winContext;
HPCON m_ptyHandler{INVALID_HANDLE_VALUE}; HPCON m_ptyHandler{INVALID_HANDLE_VALUE};
HANDLE m_hPipeIn{INVALID_HANDLE_VALUE}, m_hPipeOut{INVALID_HANDLE_VALUE}; HANDLE m_hPipeIn{INVALID_HANDLE_VALUE}, m_hPipeOut{INVALID_HANDLE_VALUE};

View File

@@ -32,7 +32,6 @@ public:
virtual QIODevice *notifier() = 0; virtual QIODevice *notifier() = 0;
virtual QByteArray readAll() = 0; virtual QByteArray readAll() = 0;
virtual qint64 write(const QByteArray &byteArray) = 0; virtual qint64 write(const QByteArray &byteArray) = 0;
virtual bool isAvailable() = 0;
virtual void moveToThread(QThread *targetThread) = 0; virtual void moveToThread(QThread *targetThread) = 0;
qint64 pid() { return m_pid; } qint64 pid() { return m_pid; }
QPair<qint16, qint16> size() { return m_size; } QPair<qint16, qint16> size() { return m_size; }

View File

@@ -10,6 +10,15 @@
#include "unixptyprocess.h" #include "unixptyprocess.h"
#endif #endif
bool PtyQt::isUsingConPTY()
{
#ifdef Q_OS_WIN
if (ConPtyProcess::isAvailable() && qgetenv("QTC_USE_WINPTY").isEmpty())
return true;
#endif
return false;
}
IPtyProcess *PtyQt::createPtyProcess(IPtyProcess::PtyType ptyType) IPtyProcess *PtyQt::createPtyProcess(IPtyProcess::PtyType ptyType)
{ {
@@ -34,7 +43,7 @@ IPtyProcess *PtyQt::createPtyProcess(IPtyProcess::PtyType ptyType)
} }
#ifdef Q_OS_WIN #ifdef Q_OS_WIN
if (ConPtyProcess().isAvailable() && qgetenv("QTC_USE_WINPTY").isEmpty()) if (isUsingConPTY())
return new ConPtyProcess(); return new ConPtyProcess();
else else
return new WinPtyProcess(); return new WinPtyProcess();
@@ -43,3 +52,5 @@ IPtyProcess *PtyQt::createPtyProcess(IPtyProcess::PtyType ptyType)
return new UnixPtyProcess(); return new UnixPtyProcess();
#endif #endif
} }

View File

@@ -7,6 +7,8 @@ class PtyQt
{ {
public: public:
static IPtyProcess *createPtyProcess(IPtyProcess::PtyType ptyType); static IPtyProcess *createPtyProcess(IPtyProcess::PtyType ptyType);
static bool isUsingConPTY();
}; };
#endif // PTYQT_H #endif // PTYQT_H

View File

@@ -54,7 +54,7 @@ public:
virtual QIODevice *notifier(); virtual QIODevice *notifier();
virtual QByteArray readAll(); virtual QByteArray readAll();
virtual qint64 write(const QByteArray &byteArray); virtual qint64 write(const QByteArray &byteArray);
virtual bool isAvailable(); static bool isAvailable();
void moveToThread(QThread *targetThread); void moveToThread(QThread *targetThread);
private: private:

View File

@@ -26,7 +26,7 @@ public:
QIODevice *notifier(); QIODevice *notifier();
QByteArray readAll(); QByteArray readAll();
qint64 write(const QByteArray &byteArray); qint64 write(const QByteArray &byteArray);
bool isAvailable(); static bool isAvailable();
void moveToThread(QThread *targetThread); void moveToThread(QThread *targetThread);
private: private:

View File

@@ -1311,8 +1311,10 @@ static int on_csi(const char *leader, const long args[], int argcount, const cha
break; break;
case LEADER('?', 0x68): // DEC private mode set case LEADER('?', 0x68): // DEC private mode set
if(!CSI_ARG_IS_MISSING(args[0])) for (int i = 0; i < argcount; i++) {
set_dec_mode(state, CSI_ARG(args[0]), 1); if (!CSI_ARG_IS_MISSING(args[i]))
set_dec_mode(state, CSI_ARG(args[i]), 1);
}
break; break;
case 0x6a: // HPB - ECMA-48 8.3.58 case 0x6a: // HPB - ECMA-48 8.3.58
@@ -1333,8 +1335,10 @@ static int on_csi(const char *leader, const long args[], int argcount, const cha
break; break;
case LEADER('?', 0x6c): // DEC private mode reset case LEADER('?', 0x6c): // DEC private mode reset
if(!CSI_ARG_IS_MISSING(args[0])) for (int i = 0; i < argcount; i++) {
set_dec_mode(state, CSI_ARG(args[0]), 0); if (!CSI_ARG_IS_MISSING(args[i]))
set_dec_mode(state, CSI_ARG(args[i]), 0);
}
break; break;
case 0x6d: // SGR - ECMA-48 8.3.117 case 0x6d: // SGR - ECMA-48 8.3.117

View File

@@ -1,7 +1,7 @@
add_qtc_plugin(Terminal add_qtc_plugin(Terminal
PLUGIN_DEPENDS Core ProjectExplorer PLUGIN_DEPENDS Core ProjectExplorer
DEPENDS libvterm DEPENDS libvterm ptyqt
SOURCES SOURCES
celliterator.cpp celliterator.h celliterator.cpp celliterator.h
glyphcache.cpp glyphcache.h glyphcache.cpp glyphcache.h

View File

@@ -54,9 +54,6 @@ TerminalPane::TerminalPane(QObject *parent)
initActions(); initActions();
m_lockKeyboardButton = new QToolButton();
m_lockKeyboardButton->setDefaultAction(&lockKeyboard);
m_newTerminalButton = new QToolButton(); m_newTerminalButton = new QToolButton();
m_newTerminalButton->setDefaultAction(&newTerminal); m_newTerminalButton->setDefaultAction(&newTerminal);
@@ -94,11 +91,34 @@ TerminalPane::TerminalPane(QObject *parent)
connect(m_escSettingButton, &QToolButton::toggled, this, [this, updateEscButton] { connect(m_escSettingButton, &QToolButton::toggled, this, [this, updateEscButton] {
settings().sendEscapeToTerminal.setValue(m_escSettingButton->isChecked()); settings().sendEscapeToTerminal.setValue(m_escSettingButton->isChecked());
settings().writeSettings();
updateEscButton(); updateEscButton();
}); });
connect(&settings(), &TerminalSettings::applied, this, updateEscButton); connect(&settings(), &TerminalSettings::applied, this, updateEscButton);
const auto updateLockButton = [this] {
m_lockKeyboardButton->setChecked(settings().lockKeyboard());
if (settings().lockKeyboard()) {
m_lockKeyboardButton->setIcon(LOCK_KEYBOARD_ICON.icon());
m_lockKeyboardButton->setToolTip(
Tr::tr("Qt Creator shortcuts are blocked when focus is inside the terminal."));
} else {
m_lockKeyboardButton->setIcon(UNLOCK_KEYBOARD_ICON.icon());
m_lockKeyboardButton->setToolTip(Tr::tr("Qt Creator shortcuts take precedence."));
}
};
m_lockKeyboardButton = new QToolButton();
m_lockKeyboardButton->setCheckable(true);
updateLockButton();
connect(m_lockKeyboardButton, &QToolButton::toggled, this, [this, updateLockButton] {
settings().lockKeyboard.setValue(m_lockKeyboardButton->isChecked());
updateLockButton();
});
connect(&settings(), &TerminalSettings::applied, this, updateLockButton);
} }
TerminalPane::~TerminalPane() {} TerminalPane::~TerminalPane() {}
@@ -249,23 +269,6 @@ void TerminalPane::initActions()
{ {
createShellMenu(); createShellMenu();
lockKeyboard.setCheckable(true);
lockKeyboard.setChecked(settings().lockKeyboard());
auto updateLockKeyboard = [this](bool locked) {
settings().lockKeyboard.setValue(locked);
if (locked) {
lockKeyboard.setIcon(LOCK_KEYBOARD_ICON.icon());
lockKeyboard.setToolTip(Tr::tr("Sends keyboard shortcuts to Terminal."));
} else {
lockKeyboard.setIcon(UNLOCK_KEYBOARD_ICON.icon());
lockKeyboard.setToolTip(Tr::tr("Sends keyboard shortcuts to Qt Creator."));
}
};
updateLockKeyboard(settings().lockKeyboard());
connect(&lockKeyboard, &QAction::toggled, this, updateLockKeyboard);
newTerminal.setText(Tr::tr("New Terminal")); newTerminal.setText(Tr::tr("New Terminal"));
newTerminal.setIcon(NEW_TERMINAL_ICON.icon()); newTerminal.setIcon(NEW_TERMINAL_ICON.icon());
newTerminal.setToolTip(Tr::tr("Create a new Terminal.")); newTerminal.setToolTip(Tr::tr("Create a new Terminal."));

View File

@@ -68,7 +68,6 @@ private:
QAction nextTerminal; QAction nextTerminal;
QAction prevTerminal; QAction prevTerminal;
QAction closeTerminal; QAction closeTerminal;
QAction lockKeyboard;
QMenu m_shellMenu; QMenu m_shellMenu;

View File

@@ -8,6 +8,8 @@
#include <coreplugin/icore.h> #include <coreplugin/icore.h>
#include <coreplugin/dialogs/ioptionspage.h> #include <coreplugin/dialogs/ioptionspage.h>
#include <libptyqt/ptyqt.h>
#include <utils/dropsupport.h> #include <utils/dropsupport.h>
#include <utils/environment.h> #include <utils/environment.h>
#include <utils/expected.h> #include <utils/expected.h>
@@ -410,12 +412,23 @@ TerminalSettings::TerminalSettings()
"instead of closing the terminal.")); "instead of closing the terminal."));
sendEscapeToTerminal.setDefaultValue(false); sendEscapeToTerminal.setDefaultValue(false);
lockKeyboard.setSettingsKey("LockKeyboard");
lockKeyboard.setLabelText(Tr::tr("Block shortcuts in terminal"));
lockKeyboard.setToolTip(
Tr::tr("Keeps Qt Creator short cuts from interfering with the terminal."));
lockKeyboard.setDefaultValue(true);
audibleBell.setSettingsKey("AudibleBell"); audibleBell.setSettingsKey("AudibleBell");
audibleBell.setLabelText(Tr::tr("Audible bell")); audibleBell.setLabelText(Tr::tr("Audible bell"));
audibleBell.setToolTip(Tr::tr("Makes the terminal beep when a bell " audibleBell.setToolTip(Tr::tr("Makes the terminal beep when a bell "
"character is received.")); "character is received."));
audibleBell.setDefaultValue(true); audibleBell.setDefaultValue(true);
enableMouseTracking.setSettingsKey("EnableMouseTracking");
enableMouseTracking.setLabelText(Tr::tr("Enable mouse tracking"));
enableMouseTracking.setToolTip(Tr::tr("Enables mouse tracking in the terminal."));
enableMouseTracking.setDefaultValue(true);
setupColor(this, setupColor(this,
foregroundColor, foregroundColor,
"Foreground", "Foreground",
@@ -528,8 +541,10 @@ TerminalSettings::TerminalSettings()
Column { Column {
enableTerminal, st, enableTerminal, st,
sendEscapeToTerminal, st, sendEscapeToTerminal, st,
lockKeyboard, st,
audibleBell, st, audibleBell, st,
allowBlinkingCursor, st, allowBlinkingCursor, st,
enableMouseTracking, st,
}, },
}, },
Group { Group {

View File

@@ -31,6 +31,8 @@ public:
Utils::BoolAspect sendEscapeToTerminal{this}; Utils::BoolAspect sendEscapeToTerminal{this};
Utils::BoolAspect audibleBell{this}; Utils::BoolAspect audibleBell{this};
Utils::BoolAspect lockKeyboard{this}; Utils::BoolAspect lockKeyboard{this};
Utils::BoolAspect enableMouseTracking{this};
}; };
TerminalSettings &settings(); TerminalSettings &settings();

View File

@@ -494,6 +494,39 @@ ShellIntegration *TerminalSurface::shellIntegration() const
return d->m_shellIntegration; return d->m_shellIntegration;
} }
void TerminalSurface::mouseMove(QPoint pos, Qt::KeyboardModifiers modifiers)
{
vterm_mouse_move(d->m_vterm.get(), pos.y(), pos.x(), Internal::qtModifierToVTerm(modifiers));
}
void TerminalSurface::mouseButton(Qt::MouseButton button,
bool pressed,
Qt::KeyboardModifiers modifiers)
{
int btnIdx = 0;
switch (button) {
case Qt::LeftButton:
btnIdx = 1;
break;
case Qt::RightButton:
btnIdx = 3;
break;
case Qt::MiddleButton:
btnIdx = 2;
break;
case Qt::ExtraButton1:
btnIdx = 4;
break;
case Qt::ExtraButton2:
btnIdx = 5;
break;
default:
return;
}
vterm_mouse_button(d->m_vterm.get(), btnIdx, pressed, Internal::qtModifierToVTerm(modifiers));
}
CellIterator TerminalSurface::begin() const CellIterator TerminalSurface::begin() const
{ {
auto res = CellIterator(this, {0, 0}); auto res = CellIterator(this, {0, 0});

View File

@@ -95,6 +95,8 @@ public:
ShellIntegration *shellIntegration() const; ShellIntegration *shellIntegration() const;
void mouseMove(QPoint pos, Qt::KeyboardModifiers modifiers);
void mouseButton(Qt::MouseButton button, bool pressed, Qt::KeyboardModifiers modifiers);
signals: signals:
void writeToPty(const QByteArray &data); void writeToPty(const QByteArray &data);
void invalidated(QRect grid); void invalidated(QRect grid);

View File

@@ -1280,6 +1280,14 @@ void TerminalWidget::updateViewportRect(const QRect &rect)
void TerminalWidget::wheelEvent(QWheelEvent *event) void TerminalWidget::wheelEvent(QWheelEvent *event)
{ {
verticalScrollBar()->event(event); verticalScrollBar()->event(event);
if (!settings().enableMouseTracking())
return;
if (event->angleDelta().ry() > 0)
m_surface->mouseButton(Qt::ExtraButton1, true, event->modifiers());
else if (event->angleDelta().ry() < 0)
m_surface->mouseButton(Qt::ExtraButton2, true, event->modifiers());
} }
void TerminalWidget::focusInEvent(QFocusEvent *) void TerminalWidget::focusInEvent(QFocusEvent *)
@@ -1306,8 +1314,18 @@ void TerminalWidget::inputMethodEvent(QInputMethodEvent *event)
m_surface->sendKey(event->commitString()); m_surface->sendKey(event->commitString());
} }
QPoint TerminalWidget::toGridPos(QMouseEvent *event) const
{
return globalToGrid(event->pos().toPointF() + QPointF(0, -topMargin() + 0.5));
}
void TerminalWidget::mousePressEvent(QMouseEvent *event) void TerminalWidget::mousePressEvent(QMouseEvent *event)
{ {
if (settings().enableMouseTracking()) {
m_surface->mouseMove(toGridPos(event), event->modifiers());
m_surface->mouseButton(event->button(), true, event->modifiers());
}
m_scrollDirection = 0; m_scrollDirection = 0;
m_activeMouseSelect.start = viewportToGlobal(event->pos()); m_activeMouseSelect.start = viewportToGlobal(event->pos());
@@ -1383,8 +1401,12 @@ void TerminalWidget::mousePressEvent(QMouseEvent *event)
} }
} }
} }
void TerminalWidget::mouseMoveEvent(QMouseEvent *event) void TerminalWidget::mouseMoveEvent(QMouseEvent *event)
{ {
if (settings().enableMouseTracking())
m_surface->mouseMove(toGridPos(event), event->modifiers());
if (m_selection && event->buttons() & Qt::LeftButton) { if (m_selection && event->buttons() & Qt::LeftButton) {
Selection newSelection = *m_selection; Selection newSelection = *m_selection;
int scrollVelocity = 0; int scrollVelocity = 0;
@@ -1487,6 +1509,11 @@ bool TerminalWidget::checkLinkAt(const QPoint &pos)
void TerminalWidget::mouseReleaseEvent(QMouseEvent *event) void TerminalWidget::mouseReleaseEvent(QMouseEvent *event)
{ {
if (settings().enableMouseTracking()) {
m_surface->mouseMove(toGridPos(event), event->modifiers());
m_surface->mouseButton(event->button(), false, event->modifiers());
}
m_scrollTimer.stop(); m_scrollTimer.stop();
if (m_selection && event->button() == Qt::LeftButton) { if (m_selection && event->button() == Qt::LeftButton) {
@@ -1528,6 +1555,12 @@ TerminalWidget::TextAndOffsets TerminalWidget::textAt(const QPoint &pos) const
void TerminalWidget::mouseDoubleClickEvent(QMouseEvent *event) void TerminalWidget::mouseDoubleClickEvent(QMouseEvent *event)
{ {
if (settings().enableMouseTracking()) {
m_surface->mouseMove(toGridPos(event), event->modifiers());
m_surface->mouseButton(event->button(), true, event->modifiers());
m_surface->mouseButton(event->button(), false, event->modifiers());
}
const auto hit = textAt(event->pos()); const auto hit = textAt(event->pos());
setSelection(Selection{hit.start, hit.end, true}); setSelection(Selection{hit.start, hit.end, true});

View File

@@ -156,6 +156,7 @@ protected:
QPoint globalToGrid(QPointF p) const; QPoint globalToGrid(QPointF p) const;
QPointF gridToGlobal(QPoint p, bool bottom = false, bool right = false) const; QPointF gridToGlobal(QPoint p, bool bottom = false, bool right = false) const;
QRect gridToViewport(QRect rect) const; QRect gridToViewport(QRect rect) const;
QPoint toGridPos(QMouseEvent *event) const;
void updateViewport(); void updateViewport();
void updateViewportRect(const QRect &rect); void updateViewportRect(const QRect &rect);