forked from qt-creator/qt-creator
Terminal: Add mouse support
Change-Id: Ibeb8e13b5f8f75f16ec86f64536235587c844ffc Reviewed-by: Cristian Adam <cristian.adam@qt.io>
This commit is contained in:
80
src/libs/3rdparty/libptyqt/conptyprocess.cpp
vendored
80
src/libs/3rdparty/libptyqt/conptyprocess.cpp
vendored
@@ -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)
|
||||||
|
77
src/libs/3rdparty/libptyqt/conptyprocess.h
vendored
77
src/libs/3rdparty/libptyqt/conptyprocess.h
vendored
@@ -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};
|
||||||
|
|
||||||
|
1
src/libs/3rdparty/libptyqt/iptyprocess.h
vendored
1
src/libs/3rdparty/libptyqt/iptyprocess.h
vendored
@@ -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; }
|
||||||
|
13
src/libs/3rdparty/libptyqt/ptyqt.cpp
vendored
13
src/libs/3rdparty/libptyqt/ptyqt.cpp
vendored
@@ -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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
2
src/libs/3rdparty/libptyqt/ptyqt.h
vendored
2
src/libs/3rdparty/libptyqt/ptyqt.h
vendored
@@ -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
|
||||||
|
2
src/libs/3rdparty/libptyqt/unixptyprocess.h
vendored
2
src/libs/3rdparty/libptyqt/unixptyprocess.h
vendored
@@ -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:
|
||||||
|
2
src/libs/3rdparty/libptyqt/winptyprocess.h
vendored
2
src/libs/3rdparty/libptyqt/winptyprocess.h
vendored
@@ -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:
|
||||||
|
12
src/libs/3rdparty/libvterm/src/state.c
vendored
12
src/libs/3rdparty/libvterm/src/state.c
vendored
@@ -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
|
||||||
|
@@ -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
|
||||||
|
@@ -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."));
|
||||||
|
@@ -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;
|
||||||
|
|
||||||
|
@@ -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 {
|
||||||
|
@@ -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();
|
||||||
|
@@ -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});
|
||||||
|
@@ -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);
|
||||||
|
@@ -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});
|
||||||
|
@@ -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);
|
||||||
|
Reference in New Issue
Block a user