2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2016-03-14 09:23:52 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2016 Martin Gräßlin <mgraesslin@kde.org>
|
2016-03-14 09:23:52 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2016-03-14 09:23:52 +00:00
|
|
|
#include "debug_console.h"
|
2016-11-10 14:56:36 +00:00
|
|
|
#include "composite.h"
|
2016-05-24 09:04:47 +00:00
|
|
|
#include "input_event.h"
|
2021-10-02 08:26:51 +00:00
|
|
|
#include "inputdevice.h"
|
2019-08-26 07:44:04 +00:00
|
|
|
#include "internal_client.h"
|
2021-08-31 06:45:20 +00:00
|
|
|
#include "keyboard_input.h"
|
2016-03-14 09:23:52 +00:00
|
|
|
#include "main.h"
|
2017-09-08 13:49:52 +00:00
|
|
|
#include "scene.h"
|
2021-08-31 06:45:20 +00:00
|
|
|
#include "subsurfacemonitor.h"
|
2016-03-14 09:23:52 +00:00
|
|
|
#include "unmanaged.h"
|
2021-08-31 06:45:20 +00:00
|
|
|
#include "wayland_server.h"
|
2020-07-22 18:35:41 +00:00
|
|
|
#include "waylandclient.h"
|
2016-03-14 09:23:52 +00:00
|
|
|
#include "workspace.h"
|
2021-08-31 06:45:20 +00:00
|
|
|
#include "x11client.h"
|
2016-08-10 06:25:20 +00:00
|
|
|
#include <kwinglplatform.h>
|
|
|
|
#include <kwinglutils.h>
|
2021-10-19 03:18:27 +00:00
|
|
|
#include <cerrno>
|
2016-03-14 09:23:52 +00:00
|
|
|
|
|
|
|
#include "ui_debug_console.h"
|
|
|
|
|
|
|
|
// KWayland
|
2021-08-31 06:45:20 +00:00
|
|
|
#include <KWaylandServer/abstract_data_source.h>
|
2020-04-29 15:18:41 +00:00
|
|
|
#include <KWaylandServer/clientconnection.h>
|
2021-08-31 06:45:20 +00:00
|
|
|
#include <KWaylandServer/datacontrolsource_v1_interface.h>
|
|
|
|
#include <KWaylandServer/datasource_interface.h>
|
|
|
|
#include <KWaylandServer/display.h>
|
|
|
|
#include <KWaylandServer/primaryselectionsource_v1_interface.h>
|
|
|
|
#include <KWaylandServer/seat_interface.h>
|
|
|
|
#include <KWaylandServer/shmclientbuffer.h>
|
2020-04-29 15:18:41 +00:00
|
|
|
#include <KWaylandServer/subcompositor_interface.h>
|
|
|
|
#include <KWaylandServer/surface_interface.h>
|
2016-03-14 09:23:52 +00:00
|
|
|
// frameworks
|
|
|
|
#include <KLocalizedString>
|
|
|
|
#include <NETWM>
|
|
|
|
// Qt
|
2021-08-31 06:45:20 +00:00
|
|
|
#include <QFutureWatcher>
|
2016-03-14 09:23:52 +00:00
|
|
|
#include <QMetaProperty>
|
|
|
|
#include <QMetaType>
|
2021-08-31 06:45:20 +00:00
|
|
|
#include <QMouseEvent>
|
|
|
|
#include <QScopeGuard>
|
|
|
|
#include <QtConcurrentRun>
|
|
|
|
|
|
|
|
#include <wayland-server-core.h>
|
2016-03-14 09:23:52 +00:00
|
|
|
|
2016-11-16 10:18:36 +00:00
|
|
|
// xkb
|
|
|
|
#include <xkbcommon/xkbcommon.h>
|
|
|
|
|
2021-08-31 06:45:20 +00:00
|
|
|
#include <fcntl.h>
|
2016-11-16 10:18:36 +00:00
|
|
|
#include <functional>
|
2021-08-31 06:45:20 +00:00
|
|
|
#include <sys/poll.h>
|
|
|
|
#include <unistd.h>
|
2016-11-16 10:18:36 +00:00
|
|
|
|
2016-03-14 09:23:52 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2016-03-30 14:23:58 +00:00
|
|
|
|
|
|
|
static QString tableHeaderRow(const QString &title)
|
|
|
|
{
|
|
|
|
return QStringLiteral("<tr><th colspan=\"2\">%1</th></tr>").arg(title);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
static
|
|
|
|
QString tableRow(const QString &title, const T &argument)
|
|
|
|
{
|
|
|
|
return QStringLiteral("<tr><td>%1</td><td>%2</td></tr>").arg(title).arg(argument);
|
|
|
|
}
|
|
|
|
|
|
|
|
static QString timestampRow(quint32 timestamp)
|
|
|
|
{
|
|
|
|
return tableRow(i18n("Timestamp"), timestamp);
|
|
|
|
}
|
|
|
|
|
2016-10-07 12:47:25 +00:00
|
|
|
static QString timestampRowUsec(quint64 timestamp)
|
|
|
|
{
|
|
|
|
return tableRow(i18n("Timestamp (µsec)"), timestamp);
|
|
|
|
}
|
|
|
|
|
2016-03-30 14:23:58 +00:00
|
|
|
static QString buttonToString(Qt::MouseButton button)
|
|
|
|
{
|
|
|
|
switch (button) {
|
|
|
|
case Qt::LeftButton:
|
|
|
|
return i18nc("A mouse button", "Left");
|
|
|
|
case Qt::RightButton:
|
|
|
|
return i18nc("A mouse button", "Right");
|
|
|
|
case Qt::MiddleButton:
|
|
|
|
return i18nc("A mouse button", "Middle");
|
|
|
|
case Qt::BackButton:
|
|
|
|
return i18nc("A mouse button", "Back");
|
|
|
|
case Qt::ForwardButton:
|
|
|
|
return i18nc("A mouse button", "Forward");
|
|
|
|
case Qt::TaskButton:
|
|
|
|
return i18nc("A mouse button", "Task");
|
|
|
|
case Qt::ExtraButton4:
|
|
|
|
return i18nc("A mouse button", "Extra Button 4");
|
|
|
|
case Qt::ExtraButton5:
|
|
|
|
return i18nc("A mouse button", "Extra Button 5");
|
|
|
|
case Qt::ExtraButton6:
|
|
|
|
return i18nc("A mouse button", "Extra Button 6");
|
|
|
|
case Qt::ExtraButton7:
|
|
|
|
return i18nc("A mouse button", "Extra Button 7");
|
|
|
|
case Qt::ExtraButton8:
|
|
|
|
return i18nc("A mouse button", "Extra Button 8");
|
|
|
|
case Qt::ExtraButton9:
|
|
|
|
return i18nc("A mouse button", "Extra Button 9");
|
|
|
|
case Qt::ExtraButton10:
|
|
|
|
return i18nc("A mouse button", "Extra Button 10");
|
|
|
|
case Qt::ExtraButton11:
|
|
|
|
return i18nc("A mouse button", "Extra Button 11");
|
|
|
|
case Qt::ExtraButton12:
|
|
|
|
return i18nc("A mouse button", "Extra Button 12");
|
|
|
|
case Qt::ExtraButton13:
|
|
|
|
return i18nc("A mouse button", "Extra Button 13");
|
|
|
|
case Qt::ExtraButton14:
|
|
|
|
return i18nc("A mouse button", "Extra Button 14");
|
|
|
|
case Qt::ExtraButton15:
|
|
|
|
return i18nc("A mouse button", "Extra Button 15");
|
|
|
|
case Qt::ExtraButton16:
|
|
|
|
return i18nc("A mouse button", "Extra Button 16");
|
|
|
|
case Qt::ExtraButton17:
|
|
|
|
return i18nc("A mouse button", "Extra Button 17");
|
|
|
|
case Qt::ExtraButton18:
|
|
|
|
return i18nc("A mouse button", "Extra Button 18");
|
|
|
|
case Qt::ExtraButton19:
|
|
|
|
return i18nc("A mouse button", "Extra Button 19");
|
|
|
|
case Qt::ExtraButton20:
|
|
|
|
return i18nc("A mouse button", "Extra Button 20");
|
|
|
|
case Qt::ExtraButton21:
|
|
|
|
return i18nc("A mouse button", "Extra Button 21");
|
|
|
|
case Qt::ExtraButton22:
|
|
|
|
return i18nc("A mouse button", "Extra Button 22");
|
|
|
|
case Qt::ExtraButton23:
|
|
|
|
return i18nc("A mouse button", "Extra Button 23");
|
|
|
|
case Qt::ExtraButton24:
|
|
|
|
return i18nc("A mouse button", "Extra Button 24");
|
|
|
|
default:
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-02 08:26:51 +00:00
|
|
|
static QString deviceRow(InputDevice *device)
|
2016-05-24 09:04:47 +00:00
|
|
|
{
|
|
|
|
if (!device) {
|
|
|
|
return tableRow(i18n("Input Device"), i18nc("The input device of the event is not known", "Unknown"));
|
|
|
|
}
|
2021-08-22 13:15:16 +00:00
|
|
|
return tableRow(i18n("Input Device"), QStringLiteral("%1 (%2)").arg(device->name(), device->sysName()));
|
2016-05-24 09:04:47 +00:00
|
|
|
}
|
|
|
|
|
2016-03-30 14:23:58 +00:00
|
|
|
static QString buttonsToString(Qt::MouseButtons buttons)
|
|
|
|
{
|
|
|
|
QString ret;
|
|
|
|
for (uint i = 1; i < Qt::ExtraButton24; i = i << 1) {
|
|
|
|
if (buttons & i) {
|
|
|
|
ret.append(buttonToString(Qt::MouseButton(uint(buttons) & i)));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
};
|
2016-05-30 13:32:05 +00:00
|
|
|
return ret.trimmed();
|
2016-03-30 14:23:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const QString s_hr = QStringLiteral("<hr/>");
|
|
|
|
static const QString s_tableStart = QStringLiteral("<table>");
|
|
|
|
static const QString s_tableEnd = QStringLiteral("</table>");
|
|
|
|
|
|
|
|
DebugConsoleFilter::DebugConsoleFilter(QTextEdit *textEdit)
|
2017-01-01 18:06:04 +00:00
|
|
|
: InputEventSpy()
|
2016-03-30 14:23:58 +00:00
|
|
|
, m_textEdit(textEdit)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
DebugConsoleFilter::~DebugConsoleFilter() = default;
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::pointerEvent(MouseEvent *event)
|
2016-03-30 14:23:58 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
const QString timestamp = timestampRow(event->timestamp());
|
|
|
|
|
|
|
|
text.append(s_tableStart);
|
|
|
|
switch (event->type()) {
|
2016-10-07 12:47:25 +00:00
|
|
|
case QEvent::MouseMove: {
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(tableHeaderRow(i18nc("A mouse pointer motion event", "Pointer Motion")));
|
2017-01-01 18:06:04 +00:00
|
|
|
text.append(deviceRow(event->device()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(timestamp);
|
2017-01-01 18:06:04 +00:00
|
|
|
if (event->timestampMicroseconds() != 0) {
|
|
|
|
text.append(timestampRowUsec(event->timestampMicroseconds()));
|
2016-10-07 12:47:25 +00:00
|
|
|
}
|
2017-01-01 18:06:04 +00:00
|
|
|
if (event->delta() != QSizeF()) {
|
2016-10-07 12:47:25 +00:00
|
|
|
text.append(tableRow(i18nc("The relative mouse movement", "Delta"),
|
2017-01-01 18:06:04 +00:00
|
|
|
QStringLiteral("%1/%2").arg(event->delta().width()).arg(event->delta().height())));
|
2016-10-07 12:47:25 +00:00
|
|
|
}
|
2017-01-01 18:06:04 +00:00
|
|
|
if (event->deltaUnaccelerated() != QSizeF()) {
|
2016-10-07 12:47:25 +00:00
|
|
|
text.append(tableRow(i18nc("The relative mouse movement", "Delta (not accelerated)"),
|
2017-01-01 18:06:04 +00:00
|
|
|
QStringLiteral("%1/%2").arg(event->deltaUnaccelerated().width()).arg(event->deltaUnaccelerated().height())));
|
2016-10-07 12:47:25 +00:00
|
|
|
}
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(tableRow(i18nc("The global mouse pointer position", "Global Position"), QStringLiteral("%1/%2").arg(event->pos().x()).arg(event->pos().y())));
|
|
|
|
break;
|
2016-10-07 12:47:25 +00:00
|
|
|
}
|
2016-03-30 14:23:58 +00:00
|
|
|
case QEvent::MouseButtonPress:
|
|
|
|
text.append(tableHeaderRow(i18nc("A mouse pointer button press event", "Pointer Button Press")));
|
2017-01-01 18:06:04 +00:00
|
|
|
text.append(deviceRow(event->device()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(timestamp);
|
|
|
|
text.append(tableRow(i18nc("A button in a mouse press/release event", "Button"), buttonToString(event->button())));
|
2017-01-01 18:06:04 +00:00
|
|
|
text.append(tableRow(i18nc("A button in a mouse press/release event", "Native Button code"), event->nativeButton()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(tableRow(i18nc("All currently pressed buttons in a mouse press/release event", "Pressed Buttons"), buttonsToString(event->buttons())));
|
|
|
|
break;
|
|
|
|
case QEvent::MouseButtonRelease:
|
|
|
|
text.append(tableHeaderRow(i18nc("A mouse pointer button release event", "Pointer Button Release")));
|
2017-01-01 18:06:04 +00:00
|
|
|
text.append(deviceRow(event->device()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(timestamp);
|
|
|
|
text.append(tableRow(i18nc("A button in a mouse press/release event", "Button"), buttonToString(event->button())));
|
2017-01-01 18:06:04 +00:00
|
|
|
text.append(tableRow(i18nc("A button in a mouse press/release event", "Native Button code"), event->nativeButton()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(tableRow(i18nc("All currently pressed buttons in a mouse press/release event", "Pressed Buttons"), buttonsToString(event->buttons())));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::wheelEvent(WheelEvent *event)
|
2016-03-30 14:23:58 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A mouse pointer axis (wheel) event", "Pointer Axis")));
|
2018-01-23 16:28:18 +00:00
|
|
|
text.append(deviceRow(event->device()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(timestampRow(event->timestamp()));
|
|
|
|
const Qt::Orientation orientation = event->angleDelta().x() == 0 ? Qt::Vertical : Qt::Horizontal;
|
|
|
|
text.append(tableRow(i18nc("The orientation of a pointer axis event", "Orientation"),
|
|
|
|
orientation == Qt::Horizontal ? i18nc("An orientation of a pointer axis event", "Horizontal")
|
|
|
|
: i18nc("An orientation of a pointer axis event", "Vertical")));
|
2018-09-16 21:01:29 +00:00
|
|
|
text.append(tableRow(i18nc("The angle delta of a pointer axis event", "Delta"),
|
|
|
|
orientation == Qt::Horizontal ? event->angleDelta().x() : event->angleDelta().y()));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::keyEvent(KeyEvent *event)
|
2016-03-30 14:23:58 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
|
|
|
|
switch (event->type()) {
|
|
|
|
case QEvent::KeyPress:
|
|
|
|
text.append(tableHeaderRow(i18nc("A key press event", "Key Press")));
|
|
|
|
break;
|
|
|
|
case QEvent::KeyRelease:
|
|
|
|
text.append(tableHeaderRow(i18nc("A key release event", "Key Release")));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2018-01-23 16:28:18 +00:00
|
|
|
text.append(deviceRow(event->device()));
|
2016-03-30 14:23:58 +00:00
|
|
|
auto modifiersToString = [event] {
|
|
|
|
QString ret;
|
|
|
|
if (event->modifiers().testFlag(Qt::ShiftModifier)) {
|
|
|
|
ret.append(i18nc("A keyboard modifier", "Shift"));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
if (event->modifiers().testFlag(Qt::ControlModifier)) {
|
|
|
|
ret.append(i18nc("A keyboard modifier", "Control"));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
if (event->modifiers().testFlag(Qt::AltModifier)) {
|
|
|
|
ret.append(i18nc("A keyboard modifier", "Alt"));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
if (event->modifiers().testFlag(Qt::MetaModifier)) {
|
|
|
|
ret.append(i18nc("A keyboard modifier", "Meta"));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
if (event->modifiers().testFlag(Qt::KeypadModifier)) {
|
|
|
|
ret.append(i18nc("A keyboard modifier", "Keypad"));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
if (event->modifiers().testFlag(Qt::GroupSwitchModifier)) {
|
|
|
|
ret.append(i18nc("A keyboard modifier", "Group-switch"));
|
|
|
|
ret.append(QStringLiteral(" "));
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
};
|
|
|
|
text.append(timestampRow(event->timestamp()));
|
|
|
|
text.append(tableRow(i18nc("Whether the event is an automatic key repeat", "Repeat"), event->isAutoRepeat()));
|
2019-12-11 17:35:28 +00:00
|
|
|
|
|
|
|
const auto keyMetaObject = Qt::qt_getEnumMetaObject(Qt::Key());
|
|
|
|
const auto enumerator = keyMetaObject->enumerator(keyMetaObject->indexOfEnumerator("Key"));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(tableRow(i18nc("The code as read from the input device", "Scan code"), event->nativeScanCode()));
|
2019-12-11 17:35:28 +00:00
|
|
|
text.append(tableRow(i18nc("Key according to Qt", "Qt::Key code"),
|
|
|
|
enumerator.valueToKey(event->key())));
|
2016-03-30 14:23:58 +00:00
|
|
|
text.append(tableRow(i18nc("The translated code to an Xkb symbol", "Xkb symbol"), event->nativeVirtualKey()));
|
|
|
|
text.append(tableRow(i18nc("The translated code interpreted as text", "Utf8"), event->text()));
|
|
|
|
text.append(tableRow(i18nc("The currently active modifiers", "Modifiers"), modifiersToString()));
|
|
|
|
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2019-08-11 19:57:45 +00:00
|
|
|
void DebugConsoleFilter::touchDown(qint32 id, const QPointF &pos, quint32 time)
|
2016-03-30 14:23:58 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A touch down event", "Touch down")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("The id of the touch point in the touch event", "Point identifier"), id));
|
|
|
|
text.append(tableRow(i18nc("The global position of the touch point", "Global position"),
|
|
|
|
QStringLiteral("%1/%2").arg(pos.x()).arg(pos.y())));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2019-08-11 19:57:45 +00:00
|
|
|
void DebugConsoleFilter::touchMotion(qint32 id, const QPointF &pos, quint32 time)
|
2016-03-30 14:23:58 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A touch motion event", "Touch Motion")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("The id of the touch point in the touch event", "Point identifier"), id));
|
|
|
|
text.append(tableRow(i18nc("The global position of the touch point", "Global position"),
|
|
|
|
QStringLiteral("%1/%2").arg(pos.x()).arg(pos.y())));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2019-08-11 19:57:45 +00:00
|
|
|
void DebugConsoleFilter::touchUp(qint32 id, quint32 time)
|
2016-03-30 14:23:58 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A touch up event", "Touch Up")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("The id of the touch point in the touch event", "Point identifier"), id));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::pinchGestureBegin(int fingerCount, quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A pinch gesture is started", "Pinch start")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("Number of fingers in this pinch gesture", "Finger count"), fingerCount));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::pinchGestureUpdate(qreal scale, qreal angleDelta, const QSizeF &delta, quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A pinch gesture is updated", "Pinch update")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("Current scale in pinch gesture", "Scale"), scale));
|
|
|
|
text.append(tableRow(i18nc("Current angle in pinch gesture", "Angle delta"), angleDelta));
|
|
|
|
text.append(tableRow(i18nc("Current delta in pinch gesture", "Delta x"), delta.width()));
|
|
|
|
text.append(tableRow(i18nc("Current delta in pinch gesture", "Delta y"), delta.height()));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::pinchGestureEnd(quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A pinch gesture ended", "Pinch end")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::pinchGestureCancelled(quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A pinch gesture got cancelled", "Pinch cancelled")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::swipeGestureBegin(int fingerCount, quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A swipe gesture is started", "Swipe start")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("Number of fingers in this swipe gesture", "Finger count"), fingerCount));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::swipeGestureUpdate(const QSizeF &delta, quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A swipe gesture is updated", "Swipe update")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(tableRow(i18nc("Current delta in swipe gesture", "Delta x"), delta.width()));
|
|
|
|
text.append(tableRow(i18nc("Current delta in swipe gesture", "Delta y"), delta.height()));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::swipeGestureEnd(quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A swipe gesture ended", "Swipe end")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-01-01 18:06:04 +00:00
|
|
|
void DebugConsoleFilter::swipeGestureCancelled(quint32 time)
|
2016-08-05 12:35:33 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A swipe gesture got cancelled", "Swipe cancelled")));
|
|
|
|
text.append(timestampRow(time));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2017-12-27 19:25:36 +00:00
|
|
|
void DebugConsoleFilter::switchEvent(SwitchEvent *event)
|
|
|
|
{
|
|
|
|
QString text = s_hr;
|
|
|
|
text.append(s_tableStart);
|
|
|
|
text.append(tableHeaderRow(i18nc("A hardware switch (e.g. notebook lid) got toggled", "Switch toggled")));
|
|
|
|
text.append(timestampRow(event->timestamp()));
|
|
|
|
if (event->timestampMicroseconds() != 0) {
|
|
|
|
text.append(timestampRowUsec(event->timestampMicroseconds()));
|
|
|
|
}
|
|
|
|
text.append(deviceRow(event->device()));
|
|
|
|
QString switchName;
|
|
|
|
if (event->device()->isLidSwitch()) {
|
|
|
|
switchName = i18nc("Name of a hardware switch", "Notebook lid");
|
|
|
|
} else if (event->device()->isTabletModeSwitch()) {
|
|
|
|
switchName = i18nc("Name of a hardware switch", "Tablet mode");
|
|
|
|
}
|
|
|
|
text.append(tableRow(i18nc("A hardware switch", "Switch"), switchName));
|
|
|
|
QString switchState;
|
|
|
|
switch (event->state()) {
|
|
|
|
case SwitchEvent::State::Off:
|
|
|
|
switchState = i18nc("The hardware switch got turned off", "Off");
|
|
|
|
break;
|
|
|
|
case SwitchEvent::State::On:
|
|
|
|
switchState = i18nc("The hardware switch got turned on", "On");
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
}
|
|
|
|
text.append(tableRow(i18nc("State of a hardware switch (on/off)", "State"), switchState));
|
|
|
|
text.append(s_tableEnd);
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2020-03-17 14:21:35 +00:00
|
|
|
void DebugConsoleFilter::tabletToolEvent(TabletEvent *event)
|
2019-12-01 17:51:15 +00:00
|
|
|
{
|
|
|
|
QString typeString;
|
|
|
|
{
|
|
|
|
QDebug d(&typeString);
|
|
|
|
d << event->type();
|
|
|
|
}
|
|
|
|
|
|
|
|
QString text = s_hr + s_tableStart + tableHeaderRow(i18n("Tablet Tool"))
|
|
|
|
+ tableRow(i18n("EventType"), typeString)
|
|
|
|
+ tableRow(i18n("Position"),
|
|
|
|
QStringLiteral("%1,%2").arg(event->pos().x()).arg(event->pos().y()))
|
|
|
|
+ tableRow(i18n("Tilt"),
|
|
|
|
QStringLiteral("%1,%2").arg(event->xTilt()).arg(event->yTilt()))
|
|
|
|
+ tableRow(i18n("Rotation"), QString::number(event->rotation()))
|
|
|
|
+ tableRow(i18n("Pressure"), QString::number(event->pressure()))
|
|
|
|
+ tableRow(i18n("Buttons"), QString::number(event->buttons()))
|
|
|
|
+ tableRow(i18n("Modifiers"), QString::number(event->modifiers()))
|
|
|
|
+ s_tableEnd;
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2020-11-20 01:28:05 +00:00
|
|
|
void DebugConsoleFilter::tabletToolButtonEvent(uint button, bool pressed, const TabletToolId &tabletToolId)
|
2019-12-01 17:51:15 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr + s_tableStart + tableHeaderRow(i18n("Tablet Tool Button"))
|
2020-11-20 01:28:05 +00:00
|
|
|
+ tableRow(i18n("Button"), button)
|
|
|
|
+ tableRow(i18n("Pressed"), pressed)
|
2020-12-22 16:34:10 +00:00
|
|
|
+ tableRow(i18n("Tablet"), qHash(tabletToolId.m_deviceGroupData))
|
2019-12-01 17:51:15 +00:00
|
|
|
+ s_tableEnd;
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2020-12-22 16:34:10 +00:00
|
|
|
void DebugConsoleFilter::tabletPadButtonEvent(uint button, bool pressed, const TabletPadId &tabletPadId)
|
2019-12-01 17:51:15 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr + s_tableStart
|
|
|
|
+ tableHeaderRow(i18n("Tablet Pad Button"))
|
2020-11-20 01:28:05 +00:00
|
|
|
+ tableRow(i18n("Button"), button)
|
|
|
|
+ tableRow(i18n("Pressed"), pressed)
|
2020-12-22 16:34:10 +00:00
|
|
|
+ tableRow(i18n("Tablet"), qHash(tabletPadId.data))
|
2019-12-01 17:51:15 +00:00
|
|
|
+ s_tableEnd;
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2020-12-22 16:34:10 +00:00
|
|
|
void DebugConsoleFilter::tabletPadStripEvent(int number, int position, bool isFinger, const TabletPadId &tabletPadId)
|
2019-12-01 17:51:15 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr + s_tableStart + tableHeaderRow(i18n("Tablet Pad Strip"))
|
|
|
|
+ tableRow(i18n("Number"), number)
|
|
|
|
+ tableRow(i18n("Position"), position)
|
|
|
|
+ tableRow(i18n("isFinger"), isFinger)
|
2020-12-22 16:34:10 +00:00
|
|
|
+ tableRow(i18n("Tablet"), qHash(tabletPadId.data))
|
2019-12-01 17:51:15 +00:00
|
|
|
+ s_tableEnd;
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2020-12-22 16:34:10 +00:00
|
|
|
void DebugConsoleFilter::tabletPadRingEvent(int number, int position, bool isFinger, const TabletPadId &tabletPadId)
|
2019-12-01 17:51:15 +00:00
|
|
|
{
|
|
|
|
QString text = s_hr + s_tableStart + tableHeaderRow(i18n("Tablet Pad Ring"))
|
|
|
|
+ tableRow(i18n("Number"), number)
|
|
|
|
+ tableRow(i18n("Position"), position)
|
|
|
|
+ tableRow(i18n("isFinger"), isFinger)
|
2020-12-22 16:34:10 +00:00
|
|
|
+ tableRow(i18n("Tablet"), qHash(tabletPadId.data))
|
2019-12-01 17:51:15 +00:00
|
|
|
+ s_tableEnd;
|
|
|
|
|
|
|
|
m_textEdit->insertHtml(text);
|
|
|
|
m_textEdit->ensureCursorVisible();
|
|
|
|
}
|
|
|
|
|
2021-08-31 06:45:20 +00:00
|
|
|
static QString sourceString(const KWaylandServer::AbstractDataSource *const source)
|
|
|
|
{
|
|
|
|
if (!source) {
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!source->client()) {
|
|
|
|
return QStringLiteral("XWayland source");
|
|
|
|
}
|
|
|
|
|
|
|
|
const QString executable = waylandServer()->display()->getConnection(source->client())->executablePath();
|
|
|
|
|
|
|
|
if (auto dataSource = qobject_cast<const KWaylandServer::DataSourceInterface *const>(source)) {
|
|
|
|
return QStringLiteral("wl_data_source@%1 of %2").arg(wl_resource_get_id(dataSource->resource())).arg(executable);
|
|
|
|
} else if (qobject_cast<const KWaylandServer::PrimarySelectionSourceV1Interface *const>(source)) {
|
|
|
|
return QStringLiteral("zwp_primary_selection_source_v1 of %2").arg(executable);
|
|
|
|
} else if (qobject_cast<const KWaylandServer::DataControlSourceV1Interface *const>(source)) {
|
|
|
|
return QStringLiteral("data control by %1").arg(executable);
|
|
|
|
}
|
|
|
|
return QStringLiteral("unknown source of").arg(executable);
|
|
|
|
}
|
|
|
|
|
2016-03-14 09:23:52 +00:00
|
|
|
DebugConsole::DebugConsole()
|
|
|
|
: QWidget()
|
|
|
|
, m_ui(new Ui::DebugConsole)
|
|
|
|
{
|
2016-08-29 13:34:02 +00:00
|
|
|
setAttribute(Qt::WA_ShowWithoutActivating);
|
2016-03-14 09:23:52 +00:00
|
|
|
m_ui->setupUi(this);
|
2016-03-31 07:19:35 +00:00
|
|
|
m_ui->windowsView->setItemDelegate(new DebugConsoleDelegate(this));
|
|
|
|
m_ui->windowsView->setModel(new DebugConsoleModel(this));
|
|
|
|
m_ui->surfacesView->setModel(new SurfaceTreeModel(this));
|
2021-08-31 06:45:20 +00:00
|
|
|
m_ui->clipboardContent->setModel(new DataSourceModel(this));
|
|
|
|
m_ui->primaryContent->setModel(new DataSourceModel(this));
|
2021-10-02 08:26:51 +00:00
|
|
|
m_ui->inputDevicesView->setModel(new InputDeviceModel(this));
|
|
|
|
m_ui->inputDevicesView->setItemDelegate(new DebugConsoleDelegate(this));
|
2016-03-14 09:23:52 +00:00
|
|
|
m_ui->quitButton->setIcon(QIcon::fromTheme(QStringLiteral("application-exit")));
|
2016-03-31 07:19:35 +00:00
|
|
|
m_ui->tabWidget->setTabIcon(0, QIcon::fromTheme(QStringLiteral("view-list-tree")));
|
|
|
|
m_ui->tabWidget->setTabIcon(1, QIcon::fromTheme(QStringLiteral("view-list-tree")));
|
2016-03-23 10:25:58 +00:00
|
|
|
|
|
|
|
if (kwinApp()->operationMode() == Application::OperationMode::OperationModeX11) {
|
2016-03-31 07:19:35 +00:00
|
|
|
m_ui->tabWidget->setTabEnabled(1, false);
|
|
|
|
m_ui->tabWidget->setTabEnabled(2, false);
|
2021-08-31 06:45:20 +00:00
|
|
|
m_ui->tabWidget->setTabEnabled(6, false);
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
2016-03-14 09:23:52 +00:00
|
|
|
|
|
|
|
connect(m_ui->quitButton, &QAbstractButton::clicked, this, &DebugConsole::deleteLater);
|
2016-03-31 07:19:35 +00:00
|
|
|
connect(m_ui->tabWidget, &QTabWidget::currentChanged, this,
|
|
|
|
[this] (int index) {
|
|
|
|
// delay creation of input event filter until the tab is selected
|
|
|
|
if (index == 2 && m_inputFilter.isNull()) {
|
2016-03-30 14:23:58 +00:00
|
|
|
m_inputFilter.reset(new DebugConsoleFilter(m_ui->inputTextEdit));
|
2017-01-01 18:06:04 +00:00
|
|
|
input()->installInputEventSpy(m_inputFilter.data());
|
2016-03-30 14:23:58 +00:00
|
|
|
}
|
2016-11-16 10:18:36 +00:00
|
|
|
if (index == 5) {
|
|
|
|
updateKeyboardTab();
|
|
|
|
connect(input(), &InputRedirection::keyStateChanged, this, &DebugConsole::updateKeyboardTab);
|
|
|
|
}
|
2021-08-31 06:45:20 +00:00
|
|
|
if (index == 6) {
|
|
|
|
static_cast<DataSourceModel *>(m_ui->clipboardContent->model())->setSource(waylandServer()->seat()->selection());
|
|
|
|
m_ui->clipboardSource->setText(sourceString(waylandServer()->seat()->selection()));
|
|
|
|
connect(waylandServer()->seat(), &KWaylandServer::SeatInterface::selectionChanged, this, [this](KWaylandServer::AbstractDataSource *source) {
|
|
|
|
static_cast<DataSourceModel *>(m_ui->clipboardContent->model())->setSource(source);
|
|
|
|
m_ui->clipboardSource->setText(sourceString(source));
|
|
|
|
});
|
|
|
|
static_cast<DataSourceModel *>(m_ui->primaryContent->model())->setSource(waylandServer()->seat()->primarySelection());
|
|
|
|
m_ui->primarySource->setText(sourceString(waylandServer()->seat()->primarySelection()));
|
|
|
|
connect(waylandServer()->seat(), &KWaylandServer::SeatInterface::primarySelectionChanged, this, [this](KWaylandServer::AbstractDataSource *source) {
|
|
|
|
static_cast<DataSourceModel *>(m_ui->primaryContent->model())->setSource(source);
|
|
|
|
m_ui->primarySource->setText(sourceString(source));
|
|
|
|
});
|
|
|
|
}
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
);
|
2016-03-14 09:23:52 +00:00
|
|
|
|
|
|
|
// for X11
|
|
|
|
setWindowFlags(Qt::X11BypassWindowManagerHint);
|
2016-08-10 06:25:20 +00:00
|
|
|
|
|
|
|
initGLTab();
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DebugConsole::~DebugConsole() = default;
|
|
|
|
|
2016-08-10 06:25:20 +00:00
|
|
|
void DebugConsole::initGLTab()
|
|
|
|
{
|
2016-11-16 07:36:51 +00:00
|
|
|
if (!effects || !effects->isOpenGLCompositing()) {
|
2016-08-10 06:25:20 +00:00
|
|
|
m_ui->noOpenGLLabel->setVisible(true);
|
|
|
|
m_ui->glInfoScrollArea->setVisible(false);
|
|
|
|
return;
|
|
|
|
}
|
2016-11-16 07:36:51 +00:00
|
|
|
GLPlatform *gl = GLPlatform::instance();
|
2016-08-10 06:25:20 +00:00
|
|
|
m_ui->noOpenGLLabel->setVisible(false);
|
|
|
|
m_ui->glInfoScrollArea->setVisible(true);
|
|
|
|
m_ui->glVendorStringLabel->setText(QString::fromLocal8Bit(gl->glVendorString()));
|
|
|
|
m_ui->glRendererStringLabel->setText(QString::fromLocal8Bit(gl->glRendererString()));
|
|
|
|
m_ui->glVersionStringLabel->setText(QString::fromLocal8Bit(gl->glVersionString()));
|
|
|
|
m_ui->glslVersionStringLabel->setText(QString::fromLocal8Bit(gl->glShadingLanguageVersionString()));
|
|
|
|
m_ui->glDriverLabel->setText(GLPlatform::driverToString(gl->driver()));
|
|
|
|
m_ui->glGPULabel->setText(GLPlatform::chipClassToString(gl->chipClass()));
|
|
|
|
m_ui->glVersionLabel->setText(GLPlatform::versionToString(gl->glVersion()));
|
|
|
|
m_ui->glslLabel->setText(GLPlatform::versionToString(gl->glslVersion()));
|
|
|
|
|
2017-09-08 13:49:52 +00:00
|
|
|
auto extensionsString = [] (const auto &extensions) {
|
2016-08-10 06:25:20 +00:00
|
|
|
QString text = QStringLiteral("<ul>");
|
|
|
|
for (auto extension : extensions) {
|
|
|
|
text.append(QStringLiteral("<li>%1</li>").arg(QString::fromLocal8Bit(extension)));
|
|
|
|
}
|
|
|
|
text.append(QStringLiteral("</ul>"));
|
|
|
|
return text;
|
|
|
|
};
|
|
|
|
|
2017-09-08 13:49:52 +00:00
|
|
|
m_ui->platformExtensionsLabel->setText(extensionsString(Compositor::self()->scene()->openGLPlatformInterfaceExtensions()));
|
2016-08-10 06:25:20 +00:00
|
|
|
m_ui->openGLExtensionsLabel->setText(extensionsString(openGLExtensions()));
|
|
|
|
}
|
|
|
|
|
2016-11-16 10:18:36 +00:00
|
|
|
template <typename T>
|
|
|
|
QString keymapComponentToString(xkb_keymap *map, const T &count, std::function<const char*(xkb_keymap*,T)> f)
|
|
|
|
{
|
|
|
|
QString text = QStringLiteral("<ul>");
|
|
|
|
for (T i = 0; i < count; i++) {
|
|
|
|
text.append(QStringLiteral("<li>%1</li>").arg(QString::fromLocal8Bit(f(map, i))));
|
|
|
|
}
|
|
|
|
text.append(QStringLiteral("</ul>"));
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
QString stateActiveComponents(xkb_state *state, const T &count, std::function<int(xkb_state*,T)> f, std::function<const char*(xkb_keymap*,T)> name)
|
|
|
|
{
|
|
|
|
QString text = QStringLiteral("<ul>");
|
|
|
|
xkb_keymap *map = xkb_state_get_keymap(state);
|
|
|
|
for (T i = 0; i < count; i++) {
|
|
|
|
if (f(state, i) == 1) {
|
|
|
|
text.append(QStringLiteral("<li>%1</li>").arg(QString::fromLocal8Bit(name(map, i))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
text.append(QStringLiteral("</ul>"));
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsole::updateKeyboardTab()
|
|
|
|
{
|
|
|
|
auto xkb = input()->keyboard()->xkb();
|
|
|
|
xkb_keymap *map = xkb->keymap();
|
|
|
|
xkb_state *state = xkb->state();
|
|
|
|
m_ui->layoutsLabel->setText(keymapComponentToString<xkb_layout_index_t>(map, xkb_keymap_num_layouts(map), &xkb_keymap_layout_get_name));
|
|
|
|
m_ui->currentLayoutLabel->setText(xkb_keymap_layout_get_name(map, xkb->currentLayout()));
|
|
|
|
m_ui->modifiersLabel->setText(keymapComponentToString<xkb_mod_index_t>(map, xkb_keymap_num_mods(map), &xkb_keymap_mod_get_name));
|
|
|
|
m_ui->ledsLabel->setText(keymapComponentToString<xkb_led_index_t>(map, xkb_keymap_num_leds(map), &xkb_keymap_led_get_name));
|
|
|
|
m_ui->activeLedsLabel->setText(stateActiveComponents<xkb_led_index_t>(state, xkb_keymap_num_leds(map), &xkb_state_led_index_is_active, &xkb_keymap_led_get_name));
|
|
|
|
|
|
|
|
using namespace std::placeholders;
|
|
|
|
auto modActive = std::bind(xkb_state_mod_index_is_active, _1, _2, XKB_STATE_MODS_EFFECTIVE);
|
|
|
|
m_ui->activeModifiersLabel->setText(stateActiveComponents<xkb_mod_index_t>(state, xkb_keymap_num_mods(map), modActive, &xkb_keymap_mod_get_name));
|
|
|
|
}
|
|
|
|
|
2016-10-04 12:06:01 +00:00
|
|
|
void DebugConsole::showEvent(QShowEvent *event)
|
|
|
|
{
|
|
|
|
QWidget::showEvent(event);
|
|
|
|
|
|
|
|
// delay the connection to the show event as in ctor the windowHandle returns null
|
|
|
|
connect(windowHandle(), &QWindow::visibleChanged, this,
|
|
|
|
[this] (bool visible) {
|
|
|
|
if (visible) {
|
|
|
|
// ignore
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
deleteLater();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-03-14 09:23:52 +00:00
|
|
|
DebugConsoleDelegate::DebugConsoleDelegate(QObject *parent)
|
|
|
|
: QStyledItemDelegate(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
DebugConsoleDelegate::~DebugConsoleDelegate() = default;
|
|
|
|
|
|
|
|
QString DebugConsoleDelegate::displayText(const QVariant &value, const QLocale &locale) const
|
|
|
|
{
|
2020-03-15 19:59:29 +00:00
|
|
|
switch (value.userType()) {
|
2016-03-14 09:23:52 +00:00
|
|
|
case QMetaType::QPoint: {
|
|
|
|
const QPoint p = value.toPoint();
|
|
|
|
return QStringLiteral("%1,%2").arg(p.x()).arg(p.y());
|
|
|
|
}
|
|
|
|
case QMetaType::QPointF: {
|
|
|
|
const QPointF p = value.toPointF();
|
|
|
|
return QStringLiteral("%1,%2").arg(p.x()).arg(p.y());
|
|
|
|
}
|
|
|
|
case QMetaType::QSize: {
|
|
|
|
const QSize s = value.toSize();
|
|
|
|
return QStringLiteral("%1x%2").arg(s.width()).arg(s.height());
|
|
|
|
}
|
|
|
|
case QMetaType::QSizeF: {
|
|
|
|
const QSizeF s = value.toSizeF();
|
|
|
|
return QStringLiteral("%1x%2").arg(s.width()).arg(s.height());
|
|
|
|
}
|
|
|
|
case QMetaType::QRect: {
|
|
|
|
const QRect r = value.toRect();
|
|
|
|
return QStringLiteral("%1,%2 %3x%4").arg(r.x()).arg(r.y()).arg(r.width()).arg(r.height());
|
|
|
|
}
|
|
|
|
default:
|
2020-04-29 15:18:41 +00:00
|
|
|
if (value.userType() == qMetaTypeId<KWaylandServer::SurfaceInterface*>()) {
|
|
|
|
if (auto s = value.value<KWaylandServer::SurfaceInterface*>()) {
|
|
|
|
return QStringLiteral("KWaylandServer::SurfaceInterface(0x%1)").arg(qulonglong(s), 0, 16);
|
2016-03-14 09:23:52 +00:00
|
|
|
} else {
|
|
|
|
return QStringLiteral("nullptr");
|
|
|
|
}
|
|
|
|
}
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
if (value.userType() == qMetaTypeId<Qt::MouseButtons>()) {
|
|
|
|
const auto buttons = value.value<Qt::MouseButtons>();
|
|
|
|
if (buttons == Qt::NoButton) {
|
|
|
|
return i18n("No Mouse Buttons");
|
|
|
|
}
|
|
|
|
QStringList list;
|
|
|
|
if (buttons.testFlag(Qt::LeftButton)) {
|
|
|
|
list << i18nc("Mouse Button", "left");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::RightButton)) {
|
|
|
|
list << i18nc("Mouse Button", "right");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::MiddleButton)) {
|
|
|
|
list << i18nc("Mouse Button", "middle");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::BackButton)) {
|
|
|
|
list << i18nc("Mouse Button", "back");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ForwardButton)) {
|
|
|
|
list << i18nc("Mouse Button", "forward");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton1)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 1");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton2)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 2");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton3)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 3");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton4)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 4");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton5)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 5");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton6)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 6");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton7)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 7");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton8)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 8");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton9)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 9");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton10)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 10");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton11)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 11");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton12)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 12");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton13)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 13");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton14)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 14");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton15)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 15");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton16)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 16");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton17)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 17");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton18)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 18");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton19)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 19");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton20)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 20");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton21)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 21");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton22)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 22");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton23)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 23");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::ExtraButton24)) {
|
|
|
|
list << i18nc("Mouse Button", "extra 24");
|
|
|
|
}
|
|
|
|
if (buttons.testFlag(Qt::TaskButton)) {
|
|
|
|
list << i18nc("Mouse Button", "task");
|
|
|
|
}
|
|
|
|
return list.join(QStringLiteral(", "));
|
|
|
|
}
|
2016-03-14 09:23:52 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return QStyledItemDelegate::displayText(value, locale);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const int s_x11ClientId = 1;
|
|
|
|
static const int s_x11UnmanagedId = 2;
|
|
|
|
static const int s_waylandClientId = 3;
|
2019-08-26 07:44:04 +00:00
|
|
|
static const int s_workspaceInternalId = 4;
|
2016-03-14 09:23:52 +00:00
|
|
|
static const quint32 s_propertyBitMask = 0xFFFF0000;
|
|
|
|
static const quint32 s_clientBitMask = 0x0000FFFF;
|
|
|
|
static const quint32 s_idDistance = 10000;
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void DebugConsoleModel::add(int parentRow, QVector<T*> &clients, T *client)
|
|
|
|
{
|
|
|
|
beginInsertRows(index(parentRow, 0, QModelIndex()), clients.count(), clients.count());
|
|
|
|
clients.append(client);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void DebugConsoleModel::remove(int parentRow, QVector<T*> &clients, T *client)
|
|
|
|
{
|
|
|
|
const int remove = clients.indexOf(client);
|
|
|
|
if (remove == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
beginRemoveRows(index(parentRow, 0, QModelIndex()), remove, remove);
|
|
|
|
clients.removeAt(remove);
|
|
|
|
endRemoveRows();
|
|
|
|
}
|
|
|
|
|
|
|
|
DebugConsoleModel::DebugConsoleModel(QObject *parent)
|
|
|
|
: QAbstractItemModel(parent)
|
|
|
|
{
|
2020-10-01 11:28:14 +00:00
|
|
|
const auto clients = workspace()->allClientList();
|
|
|
|
for (auto c : clients) {
|
2020-07-22 18:35:41 +00:00
|
|
|
handleClientAdded(c);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
2020-07-22 18:35:41 +00:00
|
|
|
connect(workspace(), &Workspace::clientAdded, this, &DebugConsoleModel::handleClientAdded);
|
|
|
|
connect(workspace(), &Workspace::clientRemoved, this, &DebugConsoleModel::handleClientRemoved);
|
2016-03-14 09:23:52 +00:00
|
|
|
|
|
|
|
const auto unmangeds = workspace()->unmanagedList();
|
|
|
|
for (auto u : unmangeds) {
|
|
|
|
m_unmanageds.append(u);
|
|
|
|
}
|
|
|
|
connect(workspace(), &Workspace::unmanagedAdded, this,
|
|
|
|
[this] (Unmanaged *u) {
|
|
|
|
add(s_x11UnmanagedId -1, m_unmanageds, u);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(workspace(), &Workspace::unmanagedRemoved, this,
|
|
|
|
[this] (Unmanaged *u) {
|
|
|
|
remove(s_x11UnmanagedId -1, m_unmanageds, u);
|
|
|
|
}
|
|
|
|
);
|
2019-08-26 07:44:04 +00:00
|
|
|
for (InternalClient *client : workspace()->internalClients()) {
|
|
|
|
m_internalClients.append(client);
|
|
|
|
}
|
|
|
|
connect(workspace(), &Workspace::internalClientAdded, this,
|
|
|
|
[this](InternalClient *client) {
|
|
|
|
add(s_workspaceInternalId -1, m_internalClients, client);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(workspace(), &Workspace::internalClientRemoved, this,
|
|
|
|
[this](InternalClient *client) {
|
|
|
|
remove(s_workspaceInternalId -1, m_internalClients, client);
|
|
|
|
}
|
|
|
|
);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
|
2020-07-22 18:35:41 +00:00
|
|
|
void DebugConsoleModel::handleClientAdded(AbstractClient *client)
|
|
|
|
{
|
|
|
|
X11Client *x11Client = qobject_cast<X11Client *>(client);
|
|
|
|
if (x11Client) {
|
|
|
|
add(s_x11ClientId - 1, m_x11Clients, x11Client);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
WaylandClient *waylandClient = qobject_cast<WaylandClient *>(client);
|
|
|
|
if (waylandClient) {
|
|
|
|
add(s_waylandClientId - 1, m_waylandClients, waylandClient);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsoleModel::handleClientRemoved(AbstractClient *client)
|
|
|
|
{
|
|
|
|
X11Client *x11Client = qobject_cast<X11Client *>(client);
|
|
|
|
if (x11Client) {
|
|
|
|
remove(s_x11ClientId - 1, m_x11Clients, x11Client);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
WaylandClient *waylandClient = qobject_cast<WaylandClient *>(client);
|
|
|
|
if (waylandClient) {
|
|
|
|
remove(s_waylandClientId - 1, m_waylandClients, waylandClient);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-14 09:23:52 +00:00
|
|
|
DebugConsoleModel::~DebugConsoleModel() = default;
|
|
|
|
|
|
|
|
int DebugConsoleModel::columnCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(parent)
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
int DebugConsoleModel::topLevelRowCount() const
|
|
|
|
{
|
|
|
|
return kwinApp()->shouldUseWaylandForCompositing() ? 4 : 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
int DebugConsoleModel::propertyCount(const QModelIndex &parent, T *(DebugConsoleModel::*filter)(const QModelIndex&) const) const
|
|
|
|
{
|
|
|
|
if (T *t = (this->*filter)(parent)) {
|
|
|
|
return t->metaObject()->propertyCount();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int DebugConsoleModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (!parent.isValid()) {
|
|
|
|
return topLevelRowCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (parent.internalId()) {
|
|
|
|
case s_x11ClientId:
|
|
|
|
return m_x11Clients.count();
|
|
|
|
case s_x11UnmanagedId:
|
|
|
|
return m_unmanageds.count();
|
|
|
|
case s_waylandClientId:
|
2020-03-04 07:55:26 +00:00
|
|
|
return m_waylandClients.count();
|
2019-08-26 07:44:04 +00:00
|
|
|
case s_workspaceInternalId:
|
2016-03-14 09:23:52 +00:00
|
|
|
return m_internalClients.count();
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parent.internalId() & s_propertyBitMask) {
|
|
|
|
// properties do not have children
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parent.internalId() < s_idDistance * (s_x11ClientId + 1)) {
|
|
|
|
return propertyCount(parent, &DebugConsoleModel::x11Client);
|
|
|
|
} else if (parent.internalId() < s_idDistance * (s_x11UnmanagedId + 1)) {
|
|
|
|
return propertyCount(parent, &DebugConsoleModel::unmanaged);
|
|
|
|
} else if (parent.internalId() < s_idDistance * (s_waylandClientId + 1)) {
|
2020-03-04 07:55:26 +00:00
|
|
|
return propertyCount(parent, &DebugConsoleModel::waylandClient);
|
2019-08-26 07:44:04 +00:00
|
|
|
} else if (parent.internalId() < s_idDistance * (s_workspaceInternalId + 1)) {
|
2016-03-14 09:23:52 +00:00
|
|
|
return propertyCount(parent, &DebugConsoleModel::internalClient);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
QModelIndex DebugConsoleModel::indexForClient(int row, int column, const QVector<T*> &clients, int id) const
|
|
|
|
{
|
|
|
|
if (column != 0) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
if (row >= clients.count()) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
return createIndex(row, column, s_idDistance * id + row);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
QModelIndex DebugConsoleModel::indexForProperty(int row, int column, const QModelIndex &parent, T *(DebugConsoleModel::*filter)(const QModelIndex&) const) const
|
|
|
|
{
|
|
|
|
if (T *t = (this->*filter)(parent)) {
|
|
|
|
if (row >= t->metaObject()->propertyCount()) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
return createIndex(row, column, quint32(row + 1) << 16 | parent.internalId());
|
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex DebugConsoleModel::index(int row, int column, const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (!parent.isValid()) {
|
|
|
|
// index for a top level item
|
|
|
|
if (column != 0 || row >= topLevelRowCount()) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
return createIndex(row, column, row + 1);
|
|
|
|
}
|
|
|
|
if (column >= 2) {
|
|
|
|
// max of 2 columns
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
// index for a client (second level)
|
|
|
|
switch (parent.internalId()) {
|
|
|
|
case s_x11ClientId:
|
|
|
|
return indexForClient(row, column, m_x11Clients, s_x11ClientId);
|
|
|
|
case s_x11UnmanagedId:
|
|
|
|
return indexForClient(row, column, m_unmanageds, s_x11UnmanagedId);
|
|
|
|
case s_waylandClientId:
|
2020-03-04 07:55:26 +00:00
|
|
|
return indexForClient(row, column, m_waylandClients, s_waylandClientId);
|
2019-08-26 07:44:04 +00:00
|
|
|
case s_workspaceInternalId:
|
|
|
|
return indexForClient(row, column, m_internalClients, s_workspaceInternalId);
|
2016-03-14 09:23:52 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// index for a property (third level)
|
|
|
|
if (parent.internalId() < s_idDistance * (s_x11ClientId + 1)) {
|
|
|
|
return indexForProperty(row, column, parent, &DebugConsoleModel::x11Client);
|
|
|
|
} else if (parent.internalId() < s_idDistance * (s_x11UnmanagedId + 1)) {
|
|
|
|
return indexForProperty(row, column, parent, &DebugConsoleModel::unmanaged);
|
|
|
|
} else if (parent.internalId() < s_idDistance * (s_waylandClientId + 1)) {
|
2020-03-04 07:55:26 +00:00
|
|
|
return indexForProperty(row, column, parent, &DebugConsoleModel::waylandClient);
|
2019-08-26 07:44:04 +00:00
|
|
|
} else if (parent.internalId() < s_idDistance * (s_workspaceInternalId + 1)) {
|
2016-03-14 09:23:52 +00:00
|
|
|
return indexForProperty(row, column, parent, &DebugConsoleModel::internalClient);
|
|
|
|
}
|
|
|
|
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex DebugConsoleModel::parent(const QModelIndex &child) const
|
|
|
|
{
|
2019-08-26 07:44:04 +00:00
|
|
|
if (child.internalId() <= s_workspaceInternalId) {
|
2016-03-14 09:23:52 +00:00
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
if (child.internalId() & s_propertyBitMask) {
|
|
|
|
// a property
|
|
|
|
const quint32 parentId = child.internalId() & s_clientBitMask;
|
|
|
|
if (parentId < s_idDistance * (s_x11ClientId + 1)) {
|
|
|
|
return createIndex(parentId - (s_idDistance * s_x11ClientId), 0, parentId);
|
|
|
|
} else if (parentId < s_idDistance * (s_x11UnmanagedId + 1)) {
|
|
|
|
return createIndex(parentId - (s_idDistance * s_x11UnmanagedId), 0, parentId);
|
|
|
|
} else if (parentId < s_idDistance * (s_waylandClientId + 1)) {
|
|
|
|
return createIndex(parentId - (s_idDistance * s_waylandClientId), 0, parentId);
|
2019-08-26 07:44:04 +00:00
|
|
|
} else if (parentId < s_idDistance * (s_workspaceInternalId + 1)) {
|
|
|
|
return createIndex(parentId - (s_idDistance * s_workspaceInternalId), 0, parentId);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
if (child.internalId() < s_idDistance * (s_x11ClientId + 1)) {
|
|
|
|
return createIndex(s_x11ClientId -1, 0, s_x11ClientId);
|
|
|
|
} else if (child.internalId() < s_idDistance * (s_x11UnmanagedId + 1)) {
|
|
|
|
return createIndex(s_x11UnmanagedId -1, 0, s_x11UnmanagedId);
|
|
|
|
} else if (child.internalId() < s_idDistance * (s_waylandClientId + 1)) {
|
|
|
|
return createIndex(s_waylandClientId -1, 0, s_waylandClientId);
|
2019-08-26 07:44:04 +00:00
|
|
|
} else if (child.internalId() < s_idDistance * (s_workspaceInternalId + 1)) {
|
|
|
|
return createIndex(s_workspaceInternalId -1, 0, s_workspaceInternalId);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DebugConsoleModel::propertyData(QObject *object, const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(role)
|
|
|
|
const auto property = object->metaObject()->property(index.row());
|
|
|
|
if (index.column() == 0) {
|
|
|
|
return property.name();
|
|
|
|
} else {
|
|
|
|
const QVariant value = property.read(object);
|
|
|
|
if (qstrcmp(property.name(), "windowType") == 0) {
|
|
|
|
switch (value.toInt()) {
|
|
|
|
case NET::Normal:
|
|
|
|
return QStringLiteral("NET::Normal");
|
|
|
|
case NET::Desktop:
|
|
|
|
return QStringLiteral("NET::Desktop");
|
|
|
|
case NET::Dock:
|
|
|
|
return QStringLiteral("NET::Dock");
|
|
|
|
case NET::Toolbar:
|
|
|
|
return QStringLiteral("NET::Toolbar");
|
|
|
|
case NET::Menu:
|
|
|
|
return QStringLiteral("NET::Menu");
|
|
|
|
case NET::Dialog:
|
|
|
|
return QStringLiteral("NET::Dialog");
|
|
|
|
case NET::Override:
|
|
|
|
return QStringLiteral("NET::Override");
|
|
|
|
case NET::TopMenu:
|
|
|
|
return QStringLiteral("NET::TopMenu");
|
|
|
|
case NET::Utility:
|
|
|
|
return QStringLiteral("NET::Utility");
|
|
|
|
case NET::Splash:
|
|
|
|
return QStringLiteral("NET::Splash");
|
|
|
|
case NET::DropdownMenu:
|
|
|
|
return QStringLiteral("NET::DropdownMenu");
|
|
|
|
case NET::PopupMenu:
|
|
|
|
return QStringLiteral("NET::PopupMenu");
|
|
|
|
case NET::Tooltip:
|
|
|
|
return QStringLiteral("NET::Tooltip");
|
|
|
|
case NET::Notification:
|
|
|
|
return QStringLiteral("NET::Notification");
|
|
|
|
case NET::ComboBox:
|
|
|
|
return QStringLiteral("NET::ComboBox");
|
|
|
|
case NET::DNDIcon:
|
|
|
|
return QStringLiteral("NET::DNDIcon");
|
|
|
|
case NET::OnScreenDisplay:
|
|
|
|
return QStringLiteral("NET::OnScreenDisplay");
|
2019-05-02 08:29:38 +00:00
|
|
|
case NET::CriticalNotification:
|
|
|
|
return QStringLiteral("NET::CriticalNotification");
|
2016-03-14 09:23:52 +00:00
|
|
|
case NET::Unknown:
|
|
|
|
default:
|
|
|
|
return QStringLiteral("NET::Unknown");
|
|
|
|
}
|
2021-05-14 07:46:10 +00:00
|
|
|
} else if (qstrcmp(property.name(), "layer") == 0) {
|
|
|
|
return QMetaEnum::fromType<Layer>().valueToKey(value.value<Layer>());
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class T>
|
2021-04-07 16:53:09 +00:00
|
|
|
QVariant DebugConsoleModel::clientData(const QModelIndex &index, int role, const QVector<T*> clients, const std::function<QString(T*)> &toString) const
|
2016-03-14 09:23:52 +00:00
|
|
|
{
|
|
|
|
if (index.row() >= clients.count()) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
auto c = clients.at(index.row());
|
|
|
|
if (role == Qt::DisplayRole) {
|
2021-04-07 16:53:09 +00:00
|
|
|
return toString(c);
|
2016-03-14 09:23:52 +00:00
|
|
|
} else if (role == Qt::DecorationRole) {
|
|
|
|
return c->icon();
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DebugConsoleModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if (!index.isValid()) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
if (!index.parent().isValid()) {
|
|
|
|
// one of the top levels
|
|
|
|
if (index.column() != 0 || role != Qt::DisplayRole) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
switch (index.internalId()) {
|
|
|
|
case s_x11ClientId:
|
|
|
|
return i18n("X11 Client Windows");
|
|
|
|
case s_x11UnmanagedId:
|
|
|
|
return i18n("X11 Unmanaged Windows");
|
|
|
|
case s_waylandClientId:
|
|
|
|
return i18n("Wayland Windows");
|
2019-08-26 07:44:04 +00:00
|
|
|
case s_workspaceInternalId:
|
2016-03-14 09:23:52 +00:00
|
|
|
return i18n("Internal Windows");
|
|
|
|
default:
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (index.internalId() & s_propertyBitMask) {
|
|
|
|
if (index.column() >= 2 || role != Qt::DisplayRole) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
2020-03-04 07:55:26 +00:00
|
|
|
if (AbstractClient *c = waylandClient(index)) {
|
2016-03-14 09:23:52 +00:00
|
|
|
return propertyData(c, index, role);
|
2019-08-26 07:44:04 +00:00
|
|
|
} else if (InternalClient *c = internalClient(index)) {
|
2016-03-14 09:23:52 +00:00
|
|
|
return propertyData(c, index, role);
|
2019-09-24 08:48:08 +00:00
|
|
|
} else if (X11Client *c = x11Client(index)) {
|
2016-03-14 09:23:52 +00:00
|
|
|
return propertyData(c, index, role);
|
|
|
|
} else if (Unmanaged *u = unmanaged(index)) {
|
|
|
|
return propertyData(u, index, role);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (index.column() != 0) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
2021-04-07 16:53:09 +00:00
|
|
|
|
2021-04-23 10:23:57 +00:00
|
|
|
auto generic = [] (AbstractClient *c) -> QString {
|
|
|
|
return c->caption() + QLatin1Char(' ') + QString::fromUtf8(c->metaObject()->className());
|
|
|
|
};
|
2016-03-14 09:23:52 +00:00
|
|
|
switch (index.parent().internalId()) {
|
|
|
|
case s_x11ClientId:
|
2021-04-23 10:23:57 +00:00
|
|
|
return clientData<X11Client>(index, role, m_x11Clients, [](X11Client *c) -> QString {
|
|
|
|
return QStringLiteral("0x%1: %2").arg(c->window(), 0, 16).arg(c->caption());
|
|
|
|
});
|
2016-03-14 09:23:52 +00:00
|
|
|
case s_x11UnmanagedId: {
|
|
|
|
if (index.row() >= m_unmanageds.count()) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
auto u = m_unmanageds.at(index.row());
|
|
|
|
if (role == Qt::DisplayRole) {
|
2020-12-03 22:09:15 +00:00
|
|
|
return QStringLiteral("0x%1").arg(u->window(), 0, 16);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case s_waylandClientId:
|
2021-04-07 16:53:09 +00:00
|
|
|
return clientData<WaylandClient>(index, role, m_waylandClients, generic);
|
2019-08-26 07:44:04 +00:00
|
|
|
case s_workspaceInternalId:
|
2021-04-07 16:53:09 +00:00
|
|
|
return clientData<InternalClient>(index, role, m_internalClients, generic);
|
2016-03-14 09:23:52 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
static T *clientForIndex(const QModelIndex &index, const QVector<T*> &clients, int id)
|
|
|
|
{
|
|
|
|
const qint32 row = (index.internalId() & s_clientBitMask) - (s_idDistance * id);
|
|
|
|
if (row < 0 || row >= clients.count()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
return clients.at(row);
|
|
|
|
}
|
|
|
|
|
2020-07-22 18:35:41 +00:00
|
|
|
WaylandClient *DebugConsoleModel::waylandClient(const QModelIndex &index) const
|
2016-03-14 09:23:52 +00:00
|
|
|
{
|
2020-03-04 07:55:26 +00:00
|
|
|
return clientForIndex(index, m_waylandClients, s_waylandClientId);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
|
2019-08-26 07:44:04 +00:00
|
|
|
InternalClient *DebugConsoleModel::internalClient(const QModelIndex &index) const
|
2016-03-14 09:23:52 +00:00
|
|
|
{
|
2019-08-26 07:44:04 +00:00
|
|
|
return clientForIndex(index, m_internalClients, s_workspaceInternalId);
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 08:48:08 +00:00
|
|
|
X11Client *DebugConsoleModel::x11Client(const QModelIndex &index) const
|
2016-03-14 09:23:52 +00:00
|
|
|
{
|
|
|
|
return clientForIndex(index, m_x11Clients, s_x11ClientId);
|
|
|
|
}
|
|
|
|
|
|
|
|
Unmanaged *DebugConsoleModel::unmanaged(const QModelIndex &index) const
|
|
|
|
{
|
|
|
|
return clientForIndex(index, m_unmanageds, s_x11UnmanagedId);
|
|
|
|
}
|
|
|
|
|
2016-03-23 10:25:58 +00:00
|
|
|
/////////////////////////////////////// SurfaceTreeModel
|
|
|
|
SurfaceTreeModel::SurfaceTreeModel(QObject *parent)
|
|
|
|
: QAbstractItemModel(parent)
|
|
|
|
{
|
|
|
|
// TODO: it would be nice to not have to reset the model on each change
|
|
|
|
auto reset = [this] {
|
|
|
|
beginResetModel();
|
|
|
|
endResetModel();
|
|
|
|
};
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2016-03-23 10:25:58 +00:00
|
|
|
|
2020-09-15 16:36:10 +00:00
|
|
|
auto watchSubsurfaces = [this, reset](AbstractClient *c) {
|
2016-05-03 07:48:57 +00:00
|
|
|
if (!c->surface()) {
|
2020-09-15 16:36:10 +00:00
|
|
|
return;
|
2016-05-03 07:48:57 +00:00
|
|
|
}
|
2020-09-15 16:36:10 +00:00
|
|
|
auto monitor = new SubSurfaceMonitor(c->surface(), this);
|
|
|
|
connect(monitor, &SubSurfaceMonitor::subSurfaceAdded, this, reset);
|
|
|
|
connect(monitor, &SubSurfaceMonitor::subSurfaceRemoved, this, reset);
|
|
|
|
connect (c, &QObject::destroyed, monitor, &QObject::deleteLater);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (auto c : workspace()->allClientList()) {
|
|
|
|
watchSubsurfaces(c);
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
connect(workspace(), &Workspace::clientAdded, this,
|
2020-12-11 17:26:12 +00:00
|
|
|
[reset, watchSubsurfaces] (AbstractClient *c) {
|
2020-09-15 16:36:10 +00:00
|
|
|
watchSubsurfaces(c);
|
2016-03-23 10:25:58 +00:00
|
|
|
reset();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
connect(workspace(), &Workspace::clientRemoved, this, reset);
|
2020-09-15 16:36:10 +00:00
|
|
|
connect(workspace(), &Workspace::unmanagedAdded, this, reset);
|
2016-03-23 10:25:58 +00:00
|
|
|
connect(workspace(), &Workspace::unmanagedRemoved, this, reset);
|
|
|
|
}
|
|
|
|
|
|
|
|
SurfaceTreeModel::~SurfaceTreeModel() = default;
|
|
|
|
|
|
|
|
int SurfaceTreeModel::columnCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(parent)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SurfaceTreeModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (parent.isValid()) {
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2016-03-23 10:25:58 +00:00
|
|
|
if (SurfaceInterface *surface = static_cast<SurfaceInterface*>(parent.internalPointer())) {
|
2021-06-19 12:59:07 +00:00
|
|
|
return surface->below().count() + surface->above().count();
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
// toplevel are all windows
|
|
|
|
return workspace()->allClientList().count() +
|
2019-08-26 07:44:04 +00:00
|
|
|
workspace()->unmanagedList().count();
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex SurfaceTreeModel::index(int row, int column, const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (column != 0) {
|
|
|
|
// invalid column
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (parent.isValid()) {
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2016-03-23 10:25:58 +00:00
|
|
|
if (SurfaceInterface *surface = static_cast<SurfaceInterface*>(parent.internalPointer())) {
|
2021-06-19 12:59:07 +00:00
|
|
|
int reference = 0;
|
|
|
|
const auto &below = surface->below();
|
|
|
|
if (row < reference + below.count()) {
|
|
|
|
return createIndex(row, column, below.at(row - reference)->surface());
|
|
|
|
}
|
|
|
|
reference += below.count();
|
|
|
|
|
|
|
|
const auto &above = surface->above();
|
|
|
|
if (row < reference + above.count()) {
|
|
|
|
return createIndex(row, column, above.at(row - reference)->surface());
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
// a window
|
|
|
|
const auto &allClients = workspace()->allClientList();
|
|
|
|
if (row < allClients.count()) {
|
|
|
|
// references a client
|
|
|
|
return createIndex(row, column, allClients.at(row)->surface());
|
|
|
|
}
|
|
|
|
int reference = allClients.count();
|
|
|
|
const auto &unmanaged = workspace()->unmanagedList();
|
|
|
|
if (row < reference + unmanaged.count()) {
|
|
|
|
return createIndex(row, column, unmanaged.at(row-reference)->surface());
|
|
|
|
}
|
|
|
|
reference += unmanaged.count();
|
|
|
|
// not found
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex SurfaceTreeModel::parent(const QModelIndex &child) const
|
|
|
|
{
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2016-03-23 10:25:58 +00:00
|
|
|
if (SurfaceInterface *surface = static_cast<SurfaceInterface*>(child.internalPointer())) {
|
|
|
|
const auto &subsurface = surface->subSurface();
|
2020-10-31 15:09:34 +00:00
|
|
|
if (!subsurface) {
|
2016-03-23 10:25:58 +00:00
|
|
|
// doesn't reference a subsurface, this is a top-level window
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
2020-10-31 15:09:34 +00:00
|
|
|
SurfaceInterface *parent = subsurface->parentSurface();
|
2016-03-23 10:25:58 +00:00
|
|
|
if (!parent) {
|
|
|
|
// something is wrong
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
// is the parent a subsurface itself?
|
|
|
|
if (parent->subSurface()) {
|
|
|
|
auto grandParent = parent->subSurface()->parentSurface();
|
2020-10-31 15:09:34 +00:00
|
|
|
if (!grandParent) {
|
2016-03-23 10:25:58 +00:00
|
|
|
// something is wrong
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
2021-06-19 12:59:07 +00:00
|
|
|
int row = 0;
|
|
|
|
const auto &below = grandParent->below();
|
|
|
|
for (int i = 0; i < below.count(); i++) {
|
|
|
|
if (below.at(i) == parent->subSurface()) {
|
|
|
|
return createIndex(row + i, 0, parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
row += below.count();
|
|
|
|
const auto &above = grandParent->above();
|
|
|
|
for (int i = 0; i < above.count(); i++) {
|
|
|
|
if (above.at(i) == parent->subSurface()) {
|
|
|
|
return createIndex(row + i, 0, parent);
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
// not a subsurface, thus it's a true window
|
|
|
|
int row = 0;
|
|
|
|
const auto &allClients = workspace()->allClientList();
|
|
|
|
for (; row < allClients.count(); row++) {
|
|
|
|
if (allClients.at(row)->surface() == parent) {
|
|
|
|
return createIndex(row, 0, parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
row = allClients.count();
|
|
|
|
const auto &unmanaged = workspace()->unmanagedList();
|
|
|
|
for (int i = 0; i < unmanaged.count(); i++) {
|
|
|
|
if (unmanaged.at(i)->surface() == parent) {
|
|
|
|
return createIndex(row + i, 0, parent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
row += unmanaged.count();
|
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant SurfaceTreeModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if (!index.isValid()) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2016-03-23 10:25:58 +00:00
|
|
|
if (SurfaceInterface *surface = static_cast<SurfaceInterface*>(index.internalPointer())) {
|
|
|
|
if (role == Qt::DisplayRole || role == Qt::ToolTipRole) {
|
|
|
|
return QStringLiteral("%1 (%2) - %3").arg(surface->client()->executablePath())
|
|
|
|
.arg(surface->client()->processId())
|
|
|
|
.arg(surface->id());
|
|
|
|
} else if (role == Qt::DecorationRole) {
|
2021-07-20 19:37:03 +00:00
|
|
|
if (auto buffer = qobject_cast<KWaylandServer::ShmClientBuffer *>(surface->buffer())) {
|
|
|
|
return buffer->data().scaled(QSize(64, 64), Qt::KeepAspectRatio);
|
2016-03-23 10:25:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
InputDeviceModel::InputDeviceModel(QObject *parent)
|
|
|
|
: QAbstractItemModel(parent)
|
2021-10-02 08:26:51 +00:00
|
|
|
, m_devices(input()->devices())
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
{
|
2016-05-06 12:48:25 +00:00
|
|
|
for (auto it = m_devices.constBegin(); it != m_devices.constEnd(); ++it) {
|
|
|
|
setupDeviceConnections(*it);
|
|
|
|
}
|
2021-10-02 08:26:51 +00:00
|
|
|
|
|
|
|
connect(input(), &InputRedirection::deviceAdded, this,
|
|
|
|
[this] (InputDevice *d) {
|
2016-05-06 12:48:25 +00:00
|
|
|
beginInsertRows(QModelIndex(), m_devices.count(), m_devices.count());
|
|
|
|
m_devices << d;
|
|
|
|
setupDeviceConnections(d);
|
|
|
|
endInsertRows();
|
|
|
|
}
|
|
|
|
);
|
2021-10-02 08:26:51 +00:00
|
|
|
connect(input(), &InputRedirection::deviceRemoved, this,
|
|
|
|
[this] (InputDevice *d) {
|
2016-05-06 12:48:25 +00:00
|
|
|
const int index = m_devices.indexOf(d);
|
|
|
|
if (index == -1) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
beginRemoveRows(QModelIndex(), index, index);
|
|
|
|
m_devices.removeAt(index);
|
|
|
|
endRemoveRows();
|
|
|
|
}
|
|
|
|
);
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
InputDeviceModel::~InputDeviceModel() = default;
|
|
|
|
|
|
|
|
|
|
|
|
int InputDeviceModel::columnCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
Q_UNUSED(parent)
|
|
|
|
return 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant InputDeviceModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if (!index.isValid()) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
if (!index.parent().isValid() && index.column() == 0) {
|
2021-10-02 08:26:51 +00:00
|
|
|
if (index.row() >= m_devices.count()) {
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
if (role == Qt::DisplayRole) {
|
2021-10-02 08:26:51 +00:00
|
|
|
return m_devices.at(index.row())->name();
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (index.parent().isValid()) {
|
|
|
|
if (role == Qt::DisplayRole) {
|
2021-10-02 08:26:51 +00:00
|
|
|
const auto device = m_devices.at(index.parent().row());
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
const auto property = device->metaObject()->property(index.row());
|
|
|
|
if (index.column() == 0) {
|
|
|
|
return property.name();
|
|
|
|
} else if (index.column() == 1) {
|
|
|
|
return device->property(property.name());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex InputDeviceModel::index(int row, int column, const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (column >= 2) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
if (parent.isValid()) {
|
|
|
|
if (parent.internalId() & s_propertyBitMask) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
2021-10-02 08:26:51 +00:00
|
|
|
if (row >= m_devices.at(parent.row())->metaObject()->propertyCount()) {
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
return createIndex(row, column, quint32(row + 1) << 16 | parent.internalId());
|
|
|
|
}
|
2021-10-02 08:26:51 +00:00
|
|
|
if (row >= m_devices.count()) {
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
return createIndex(row, column, row + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
int InputDeviceModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (!parent.isValid()) {
|
2021-10-02 08:26:51 +00:00
|
|
|
return m_devices.count();
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
}
|
|
|
|
if (parent.internalId() & s_propertyBitMask) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2021-10-02 08:26:51 +00:00
|
|
|
return m_devices.at(parent.row())->metaObject()->propertyCount();
|
[libinput] Add a wrapper class Device for a libinput_device
Summary:
The Device class wraps all the information we can get from libinput
about the device, like whether it's a keyboard, pointer, touch, etc.
In addition some more information is queried to figure out how "useful"
a device is. For a keyboard all alphanumeric keys are checked whether
they exist, for a pointer all (normal) buttons are queried.
All the information is exposed as Q_PROPERTY and used by the
DebugConsole. The DebugConsole gained a new tab "Input Devices" which
renders all devices and their properties in a tree view. When plugging
in/out a device, the model gets reset, so it's always up to date.
The new Device class can be used in future to configure the device,
e.g. disable touch pad, set mouse acceleration, etc.
Reviewers: #plasma
Subscribers: plasma-devel
Projects: #plasma
Differential Revision: https://phabricator.kde.org/D1538
2016-05-04 11:42:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex InputDeviceModel::parent(const QModelIndex &child) const
|
|
|
|
{
|
|
|
|
if (child.internalId() & s_propertyBitMask) {
|
|
|
|
const quintptr parentId = child.internalId() & s_clientBitMask;
|
|
|
|
return createIndex(parentId - 1, 0, parentId);
|
|
|
|
}
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
2021-10-22 16:50:18 +00:00
|
|
|
void InputDeviceModel::slotPropertyChanged()
|
2016-05-06 12:48:25 +00:00
|
|
|
{
|
2021-10-02 08:26:51 +00:00
|
|
|
const auto device = static_cast<InputDevice *>(sender());
|
2021-10-22 16:50:18 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < device->metaObject()->propertyCount(); ++i) {
|
|
|
|
const QMetaProperty metaProperty = device->metaObject()->property(i);
|
|
|
|
if (metaProperty.notifySignalIndex() == senderSignalIndex()) {
|
2016-05-06 12:48:25 +00:00
|
|
|
const QModelIndex parent = index(m_devices.indexOf(device), 0, QModelIndex());
|
2021-10-22 16:50:18 +00:00
|
|
|
const QModelIndex child = index(i, 1, parent);
|
2021-06-08 07:02:14 +00:00
|
|
|
Q_EMIT dataChanged(child, child, QVector<int>{Qt::DisplayRole});
|
2016-05-06 12:48:25 +00:00
|
|
|
}
|
2021-10-22 16:50:18 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-02 08:26:51 +00:00
|
|
|
void InputDeviceModel::setupDeviceConnections(InputDevice *device)
|
2021-10-22 16:50:18 +00:00
|
|
|
{
|
|
|
|
QMetaMethod handler = metaObject()->method(metaObject()->indexOfMethod("slotPropertyChanged()"));
|
|
|
|
for (int i = 0; i < device->metaObject()->propertyCount(); ++i) {
|
|
|
|
const QMetaProperty metaProperty = device->metaObject()->property(i);
|
|
|
|
if (metaProperty.hasNotifySignal()) {
|
|
|
|
connect(device, metaProperty.notifySignal(), this, handler);
|
2016-05-06 12:48:25 +00:00
|
|
|
}
|
2021-10-22 16:50:18 +00:00
|
|
|
}
|
2016-05-06 12:48:25 +00:00
|
|
|
}
|
|
|
|
|
2021-08-31 06:45:20 +00:00
|
|
|
QModelIndex DataSourceModel::index(int row, int column, const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (!m_source || parent.isValid() || column >= 2 || row >= m_source->mimeTypes().size()) {
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
return createIndex(row, column, nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
QModelIndex DataSourceModel::parent(const QModelIndex &child) const
|
|
|
|
{
|
|
|
|
return QModelIndex();
|
|
|
|
}
|
|
|
|
|
|
|
|
int DataSourceModel::rowCount(const QModelIndex &parent) const
|
|
|
|
{
|
|
|
|
if (!parent.isValid()) {
|
|
|
|
return m_source ? m_source->mimeTypes().count() : 0;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DataSourceModel::headerData(int section, Qt::Orientation orientation, int role) const
|
|
|
|
{
|
|
|
|
if (role != Qt::DisplayRole || orientation != Qt::Horizontal || section >= 2) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
return section == 0 ? QStringLiteral("Mime type") : QStringLiteral("Content");
|
|
|
|
}
|
|
|
|
|
|
|
|
QVariant DataSourceModel::data(const QModelIndex &index, int role) const
|
|
|
|
{
|
|
|
|
if (!checkIndex(index, CheckIndexOption::ParentIsInvalid | CheckIndexOption::IndexIsValid)) {
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
const QString mimeType = m_source->mimeTypes().at(index.row());
|
|
|
|
;
|
|
|
|
if (index.column() == 0 && role == Qt::DisplayRole) {
|
|
|
|
return mimeType;
|
Fix crash on the debug console
Do not access fields in DataSourceModel::m_data that are not available
0 0x00007f3445a4bd22 in raise () at /usr/lib/libc.so.6
1 0x00007f3445a3590e in abort () at /usr/lib/libc.so.6
2 0x00007f3446362bb1 in qt_message_fatal (message=<synthetic pointer>..., context=...) at /home/apol/devel/frameworks/qt5/qtbase/src/corelib/global/qlogging.cpp:1914
3 QMessageLogger::fatal(char const*, ...) const (this=this@entry=0x7fff288a4ff8, msg=msg@entry=0x7f3446663028 "ASSERT failure in %s: \"%s\", file %s, line %d") at /home/apol/devel/frameworks/qt5/qtbase/src/corelib/global/qlogging.cpp:893
4 0x00007f344636203e in qt_assert_x(char const*, char const*, char const*, int) (where=<optimized out>, what=<optimized out>, file=<optimized out>, line=<optimized out>) at /home/apol/devel/frameworks/qt5/qtbase/src/corelib/global/qglobal.cpp:3366
5 0x00007f344973e118 in QVector<QByteArray>::at(int) const (this=0x55f42c20d338, i=7) at /home/apol/devel/kde5/include/QtCore/qvector.h:449
6 0x00007f34497305df in KWin::DataSourceModel::data(QModelIndex const&, int) const (this=0x55f42c20d320, index=..., role=6) at /home/apol/devel/frameworks/kwin/src/debug_console.cpp:1657
7 0x00007f3447674ed1 in QModelIndex::data(int) const (arole=6, this=0x7fff288a55c0) at ../../include/QtCore/../../../../../devel/frameworks/qt5/qtbase/src/corelib/itemmodels/qabstractitemmodel.h:460
2021-09-03 17:45:50 +00:00
|
|
|
} else if (index.column() == 1 && index.row() < m_data.count()) {
|
2021-08-31 06:45:20 +00:00
|
|
|
const QByteArray &data = m_data.at(index.row());
|
|
|
|
if (mimeType.contains(QLatin1String("image"))) {
|
|
|
|
if (role == Qt::DecorationRole) {
|
|
|
|
return QImage::fromData(data);
|
|
|
|
}
|
|
|
|
} else if (role == Qt::DisplayRole) {
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QVariant();
|
|
|
|
}
|
|
|
|
|
|
|
|
static QByteArray readData(int fd)
|
|
|
|
{
|
|
|
|
pollfd pfd;
|
|
|
|
pfd.fd = fd;
|
|
|
|
pfd.events = POLLIN;
|
|
|
|
auto closeFd = qScopeGuard([fd] {
|
|
|
|
close(fd);
|
|
|
|
});
|
|
|
|
QByteArray data;
|
|
|
|
while (true) {
|
|
|
|
const int ready = poll(&pfd, 1, 1000);
|
|
|
|
if (ready < 0) {
|
|
|
|
if (errno != EINTR) {
|
|
|
|
return QByteArrayLiteral("poll() failed: ") + strerror(errno);
|
|
|
|
}
|
|
|
|
} else if (ready == 0) {
|
|
|
|
return QByteArrayLiteral("timeout reading from pipe");
|
|
|
|
} else {
|
|
|
|
char buf[4096];
|
|
|
|
int n = read(fd, buf, sizeof buf);
|
|
|
|
|
|
|
|
if (n < 0) {
|
|
|
|
return QByteArrayLiteral("read failed: ") + strerror(errno);
|
|
|
|
} else if (n == 0) {
|
|
|
|
return data;
|
|
|
|
} else if (n > 0) {
|
|
|
|
data.append(buf, n);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DataSourceModel::setSource(KWaylandServer::AbstractDataSource *source)
|
|
|
|
{
|
|
|
|
beginResetModel();
|
|
|
|
m_source = source;
|
|
|
|
m_data.clear();
|
|
|
|
if (source) {
|
|
|
|
m_data.resize(m_source->mimeTypes().size());
|
|
|
|
for (auto type = m_source->mimeTypes().cbegin(); type != m_source->mimeTypes().cend(); ++type) {
|
|
|
|
int pipeFds[2];
|
|
|
|
if (pipe2(pipeFds, O_CLOEXEC) != 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
source->requestData(*type, pipeFds[1]);
|
|
|
|
QFuture<QByteArray> data = QtConcurrent::run(readData, pipeFds[0]);
|
|
|
|
auto watcher = new QFutureWatcher<QByteArray>(this);
|
|
|
|
watcher->setFuture(data);
|
|
|
|
const int index = type - m_source->mimeTypes().cbegin();
|
|
|
|
connect(watcher, &QFutureWatcher<QByteArray>::finished, this, [this, watcher, index, source = QPointer(source)] {
|
|
|
|
watcher->deleteLater();
|
|
|
|
if (source && source == m_source) {
|
|
|
|
m_data[index] = watcher->result();
|
|
|
|
Q_EMIT dataChanged(this->index(index, 1), this->index(index, 1), {Qt::DecorationRole | Qt::DisplayRole});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
endResetModel();
|
|
|
|
}
|
2016-03-14 09:23:52 +00:00
|
|
|
}
|