Make it possible to raise windows on top of the lockscreen
Requires clients to have the X-KDE-Wayland-Interfaces=kde_lockscreenallowed_v1 set in their desktop file, then they will be able to use the kde_lockscreenallowed_v1 protocol to raise any surface above the lockscreen. The protocol has only 1 method, raise_surface to do exactly that. Makes it possible to implement https://invent.kde.org/teams/plasma-mobile/issues/-/issues/98
This commit is contained in:
parent
123549f8f3
commit
4f20e9216f
12 changed files with 245 additions and 4 deletions
|
@ -544,7 +544,7 @@ private:
|
||||||
{
|
{
|
||||||
if (KWaylandServer::SurfaceInterface *s = (waylandServer()->seat()->*method)()) {
|
if (KWaylandServer::SurfaceInterface *s = (waylandServer()->seat()->*method)()) {
|
||||||
if (Window *t = waylandServer()->findWindow(s)) {
|
if (Window *t = waylandServer()->findWindow(s)) {
|
||||||
return t->isLockScreen() || t->isInputMethod();
|
return t->isLockScreen() || t->isInputMethod() || t->isLockScreenOverlay();
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -3261,7 +3261,7 @@ Window *InputRedirection::findManagedToplevel(const QPointF &pos)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
if (isScreenLocked) {
|
if (isScreenLocked) {
|
||||||
if (!window->isLockScreen() && !window->isInputMethod()) {
|
if (!window->isLockScreen() && !window->isInputMethod() && !window->isLockScreenOverlay()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -175,6 +175,10 @@ ecm_add_qtwayland_server_protocol_kde(WaylandProtocols_xml
|
||||||
PROTOCOL ${WaylandProtocols_DATADIR}/staging/drm-lease/drm-lease-v1.xml
|
PROTOCOL ${WaylandProtocols_DATADIR}/staging/drm-lease/drm-lease-v1.xml
|
||||||
BASENAME drm-lease-v1
|
BASENAME drm-lease-v1
|
||||||
)
|
)
|
||||||
|
ecm_add_qtwayland_server_protocol_kde(WaylandProtocols_xml
|
||||||
|
PROTOCOL ${PLASMA_WAYLAND_PROTOCOLS_DIR}/kde-lockscreen-overlay-v1.xml
|
||||||
|
BASENAME kde-lockscreen-overlay-v1
|
||||||
|
)
|
||||||
|
|
||||||
target_sources(kwin PRIVATE
|
target_sources(kwin PRIVATE
|
||||||
abstract_data_source.cpp
|
abstract_data_source.cpp
|
||||||
|
@ -208,6 +212,7 @@ target_sources(kwin PRIVATE
|
||||||
keystate_interface.cpp
|
keystate_interface.cpp
|
||||||
layershell_v1_interface.cpp
|
layershell_v1_interface.cpp
|
||||||
linuxdmabufv1clientbuffer.cpp
|
linuxdmabufv1clientbuffer.cpp
|
||||||
|
lockscreen_overlay_v1_interface.cpp
|
||||||
output_interface.cpp
|
output_interface.cpp
|
||||||
outputdevice_v2_interface.cpp
|
outputdevice_v2_interface.cpp
|
||||||
outputmanagement_v2_interface.cpp
|
outputmanagement_v2_interface.cpp
|
||||||
|
|
54
src/wayland/lockscreen_overlay_v1_interface.cpp
Normal file
54
src/wayland/lockscreen_overlay_v1_interface.cpp
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
SPDX-FileCopyrightText: 2022 Aleix Pol Gonzalez <aleixpol@kde.org>
|
||||||
|
|
||||||
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "lockscreen_overlay_v1_interface.h"
|
||||||
|
#include "display.h"
|
||||||
|
#include "seat_interface.h"
|
||||||
|
#include "surface_interface.h"
|
||||||
|
|
||||||
|
#include "qwayland-server-kde-lockscreen-overlay-v1.h"
|
||||||
|
|
||||||
|
namespace KWaylandServer
|
||||||
|
{
|
||||||
|
static constexpr int s_version = 1;
|
||||||
|
|
||||||
|
class LockscreenOverlayV1InterfacePrivate : public QtWaylandServer::kde_lockscreen_overlay_v1
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
LockscreenOverlayV1InterfacePrivate(Display *display, LockscreenOverlayV1Interface *q)
|
||||||
|
: QtWaylandServer::kde_lockscreen_overlay_v1(*display, s_version)
|
||||||
|
, q(q)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
void kde_lockscreen_overlay_v1_allow(Resource *resource, struct ::wl_resource *surface) override
|
||||||
|
{
|
||||||
|
auto surfaceIface = SurfaceInterface::get(surface);
|
||||||
|
if (surfaceIface->isMapped()) {
|
||||||
|
wl_resource_post_error(resource->handle, error_invalid_surface_state, "surface is already mapped");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Q_EMIT q->allowRequested(surfaceIface);
|
||||||
|
}
|
||||||
|
void kde_lockscreen_overlay_v1_destroy(Resource *resource) override
|
||||||
|
{
|
||||||
|
wl_resource_destroy(resource->handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
LockscreenOverlayV1Interface *const q;
|
||||||
|
};
|
||||||
|
|
||||||
|
LockscreenOverlayV1Interface::~LockscreenOverlayV1Interface() = default;
|
||||||
|
|
||||||
|
LockscreenOverlayV1Interface::LockscreenOverlayV1Interface(Display *display, QObject *parent)
|
||||||
|
: QObject(parent)
|
||||||
|
, d(new LockscreenOverlayV1InterfacePrivate(display, this))
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
43
src/wayland/lockscreen_overlay_v1_interface.h
Normal file
43
src/wayland/lockscreen_overlay_v1_interface.h
Normal file
|
@ -0,0 +1,43 @@
|
||||||
|
/*
|
||||||
|
SPDX-FileCopyrightText: 2022 Aleix Pol Gonzalez <aleixpol@kde.org>
|
||||||
|
|
||||||
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "kwin_export.h"
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QVector>
|
||||||
|
#include <functional>
|
||||||
|
#include <optional>
|
||||||
|
|
||||||
|
struct wl_resource;
|
||||||
|
|
||||||
|
namespace KWaylandServer
|
||||||
|
{
|
||||||
|
class Display;
|
||||||
|
class SurfaceInterface;
|
||||||
|
|
||||||
|
class LockscreenOverlayV1InterfacePrivate;
|
||||||
|
|
||||||
|
class KWIN_EXPORT LockscreenOverlayV1Interface : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
Q_DISABLE_COPY(LockscreenOverlayV1Interface)
|
||||||
|
public:
|
||||||
|
explicit LockscreenOverlayV1Interface(Display *display, QObject *parent = nullptr);
|
||||||
|
~LockscreenOverlayV1Interface() override;
|
||||||
|
|
||||||
|
Q_SIGNALS:
|
||||||
|
/// Notifies about the @p surface being activated
|
||||||
|
void allowRequested(SurfaceInterface *surface);
|
||||||
|
|
||||||
|
private:
|
||||||
|
friend class LockscreenOverlayV1InterfacePrivate;
|
||||||
|
LockscreenOverlayV1Interface(LockscreenOverlayV1Interface *parent);
|
||||||
|
QScopedPointer<LockscreenOverlayV1InterfacePrivate> d;
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
|
@ -37,6 +37,7 @@
|
||||||
#include "wayland/keyboard_shortcuts_inhibit_v1_interface.h"
|
#include "wayland/keyboard_shortcuts_inhibit_v1_interface.h"
|
||||||
#include "wayland/keystate_interface.h"
|
#include "wayland/keystate_interface.h"
|
||||||
#include "wayland/linuxdmabufv1clientbuffer.h"
|
#include "wayland/linuxdmabufv1clientbuffer.h"
|
||||||
|
#include "wayland/lockscreen_overlay_v1_interface.h"
|
||||||
#include "wayland/output_interface.h"
|
#include "wayland/output_interface.h"
|
||||||
#include "wayland/outputdevice_v2_interface.h"
|
#include "wayland/outputdevice_v2_interface.h"
|
||||||
#include "wayland/outputmanagement_v2_interface.h"
|
#include "wayland/outputmanagement_v2_interface.h"
|
||||||
|
@ -136,6 +137,7 @@ public:
|
||||||
QByteArrayLiteral("org_kde_kwin_keystate"),
|
QByteArrayLiteral("org_kde_kwin_keystate"),
|
||||||
QByteArrayLiteral("zkde_screencast_unstable_v1"),
|
QByteArrayLiteral("zkde_screencast_unstable_v1"),
|
||||||
QByteArrayLiteral("org_kde_plasma_activation_feedback"),
|
QByteArrayLiteral("org_kde_plasma_activation_feedback"),
|
||||||
|
QByteArrayLiteral("kde_lockscreen_overlay_v1"),
|
||||||
};
|
};
|
||||||
|
|
||||||
const QSet<QByteArray> inputmethodInterfaces = {"zwp_input_panel_v1", "zwp_input_method_v1"};
|
const QSet<QByteArray> inputmethodInterfaces = {"zwp_input_panel_v1", "zwp_input_method_v1"};
|
||||||
|
@ -471,6 +473,15 @@ bool WaylandServer::init(InitializationFlags flags)
|
||||||
connect(static_cast<Application *>(qApp), &Application::workspaceCreated, this, init);
|
connect(static_cast<Application *>(qApp), &Application::workspaceCreated, this, init);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto aboveLockscreen = new KWaylandServer::LockscreenOverlayV1Interface(m_display, this);
|
||||||
|
connect(aboveLockscreen, &KWaylandServer::LockscreenOverlayV1Interface::allowRequested, this, [](SurfaceInterface *surface) {
|
||||||
|
auto w = waylandServer()->findWindow(surface);
|
||||||
|
if (!w) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
w->setLockScreenOverlay(true);
|
||||||
|
});
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -911,6 +911,9 @@ Layer Window::belongsToLayer() const
|
||||||
if (isInputMethod()) {
|
if (isInputMethod()) {
|
||||||
return UnmanagedLayer;
|
return UnmanagedLayer;
|
||||||
}
|
}
|
||||||
|
if (isLockScreenOverlay() && waylandServer() && waylandServer()->isScreenLocked()) {
|
||||||
|
return UnmanagedLayer;
|
||||||
|
}
|
||||||
if (isDesktop()) {
|
if (isDesktop()) {
|
||||||
return workspace()->showingDesktop() ? AboveLayer : DesktopLayer;
|
return workspace()->showingDesktop() ? AboveLayer : DesktopLayer;
|
||||||
}
|
}
|
||||||
|
@ -4497,6 +4500,20 @@ uint32_t Window::interactiveMoveResizeCount() const
|
||||||
return m_interactiveMoveResize.counter;
|
return m_interactiveMoveResize.counter;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Window::setLockScreenOverlay(bool allowed)
|
||||||
|
{
|
||||||
|
if (m_lockScreenOverlay == allowed) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
m_lockScreenOverlay = allowed;
|
||||||
|
Q_EMIT lockScreenOverlayChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Window::isLockScreenOverlay() const
|
||||||
|
{
|
||||||
|
return m_lockScreenOverlay;
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace KWin
|
} // namespace KWin
|
||||||
|
|
||||||
#include "moc_window.cpp"
|
#include "moc_window.cpp"
|
||||||
|
|
|
@ -712,6 +712,9 @@ public:
|
||||||
bool isOnCurrentActivity() const;
|
bool isOnCurrentActivity() const;
|
||||||
bool isOnAllDesktops() const;
|
bool isOnAllDesktops() const;
|
||||||
bool isOnAllActivities() const;
|
bool isOnAllActivities() const;
|
||||||
|
bool isLockScreenOverlay() const;
|
||||||
|
|
||||||
|
void setLockScreenOverlay(bool allowed);
|
||||||
|
|
||||||
virtual QByteArray windowRole() const;
|
virtual QByteArray windowRole() const;
|
||||||
QByteArray sessionId() const;
|
QByteArray sessionId() const;
|
||||||
|
@ -1543,6 +1546,7 @@ Q_SIGNALS:
|
||||||
void unresponsiveChanged(bool);
|
void unresponsiveChanged(bool);
|
||||||
void decorationChanged();
|
void decorationChanged();
|
||||||
void hiddenChanged();
|
void hiddenChanged();
|
||||||
|
void lockScreenOverlayChanged();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void setWindowHandles(xcb_window_t client);
|
void setWindowHandles(xcb_window_t client);
|
||||||
|
@ -2011,6 +2015,7 @@ private:
|
||||||
|
|
||||||
WindowRules m_rules;
|
WindowRules m_rules;
|
||||||
quint32 m_lastUsageSerial = 0;
|
quint32 m_lastUsageSerial = 0;
|
||||||
|
bool m_lockScreenOverlay = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -35,6 +35,7 @@ WindowItem::WindowItem(Window *window, Item *parent)
|
||||||
if (waylandServer()) {
|
if (waylandServer()) {
|
||||||
connect(waylandServer(), &WaylandServer::lockStateChanged, this, &WindowItem::updateVisibility);
|
connect(waylandServer(), &WaylandServer::lockStateChanged, this, &WindowItem::updateVisibility);
|
||||||
}
|
}
|
||||||
|
connect(window, &Window::lockScreenOverlayChanged, this, &WindowItem::updateVisibility);
|
||||||
connect(window, &Window::minimizedChanged, this, &WindowItem::updateVisibility);
|
connect(window, &Window::minimizedChanged, this, &WindowItem::updateVisibility);
|
||||||
connect(window, &Window::hiddenChanged, this, &WindowItem::updateVisibility);
|
connect(window, &Window::hiddenChanged, this, &WindowItem::updateVisibility);
|
||||||
connect(window, &Window::activitiesChanged, this, &WindowItem::updateVisibility);
|
connect(window, &Window::activitiesChanged, this, &WindowItem::updateVisibility);
|
||||||
|
@ -127,7 +128,7 @@ void WindowItem::handleWindowClosed(Window *original, Deleted *deleted)
|
||||||
bool WindowItem::computeVisibility() const
|
bool WindowItem::computeVisibility() const
|
||||||
{
|
{
|
||||||
if (waylandServer() && waylandServer()->isScreenLocked()) {
|
if (waylandServer() && waylandServer()->isScreenLocked()) {
|
||||||
return m_window->isLockScreen() || m_window->isInputMethod();
|
return m_window->isLockScreen() || m_window->isInputMethod() || m_window->isLockScreenOverlay();
|
||||||
}
|
}
|
||||||
if (m_window->isDeleted()) {
|
if (m_window->isDeleted()) {
|
||||||
if (m_forceVisibleByDeleteCount == 0) {
|
if (m_forceVisibleByDeleteCount == 0) {
|
||||||
|
|
|
@ -29,7 +29,7 @@ static bool isPrivilegedInWindowManagement(const ClientConnection *client)
|
||||||
{
|
{
|
||||||
Q_ASSERT(client);
|
Q_ASSERT(client);
|
||||||
auto requestedInterfaces = client->property("requestedInterfaces").toStringList();
|
auto requestedInterfaces = client->property("requestedInterfaces").toStringList();
|
||||||
return requestedInterfaces.contains(QLatin1String("org_kde_plasma_window_management"));
|
return requestedInterfaces.contains(QLatin1String("org_kde_plasma_window_management")) || requestedInterfaces.contains(QLatin1String("kde_lockscreen_overlay_v1"));
|
||||||
}
|
}
|
||||||
|
|
||||||
static const QString windowDesktopFileName(Window *window)
|
static const QString windowDesktopFileName(Window *window)
|
||||||
|
|
|
@ -48,6 +48,13 @@ if (QT_MAJOR_VERSION EQUAL "5")
|
||||||
PROTOCOL ${WaylandProtocols_DATADIR}/staging/xdg-activation/xdg-activation-v1.xml
|
PROTOCOL ${WaylandProtocols_DATADIR}/staging/xdg-activation/xdg-activation-v1.xml
|
||||||
BASENAME xdg-activation-v1
|
BASENAME xdg-activation-v1
|
||||||
)
|
)
|
||||||
|
|
||||||
|
add_executable(lockscreenoverlaytest lockscreenoverlaytest.cpp)
|
||||||
|
target_link_libraries(lockscreenoverlaytest Qt::Widgets Qt::WaylandClient Qt::WaylandClientPrivate Wayland::Client KF5::WindowSystem)
|
||||||
|
ecm_add_qtwayland_client_protocol(lockscreenoverlaytest
|
||||||
|
PROTOCOL ${PLASMA_WAYLAND_PROTOCOLS_DIR}/kde-lockscreen-overlay-v1.xml
|
||||||
|
BASENAME kde-lockscreen-overlay-v1
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (TARGET Qt6::Gui)
|
if (TARGET Qt6::Gui)
|
||||||
|
|
89
tests/lockscreenoverlaytest.cpp
Normal file
89
tests/lockscreenoverlaytest.cpp
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
/*
|
||||||
|
SPDX-FileCopyrightText: 2022 Aleix Pol Gonzalez <aleixpol@kde.org>
|
||||||
|
|
||||||
|
SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "qwayland-kde-lockscreen-overlay-v1.h"
|
||||||
|
#include <KWindowSystem>
|
||||||
|
#include <QWaylandClientExtensionTemplate>
|
||||||
|
#include <QtWidgets>
|
||||||
|
#include <qpa/qplatformnativeinterface.h>
|
||||||
|
|
||||||
|
class WaylandAboveLockscreen : public QWaylandClientExtensionTemplate<WaylandAboveLockscreen>, public QtWayland::kde_lockscreen_overlay_v1
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
WaylandAboveLockscreen()
|
||||||
|
: QWaylandClientExtensionTemplate<WaylandAboveLockscreen>(1)
|
||||||
|
{
|
||||||
|
QMetaObject::invokeMethod(this, "addRegistryListener");
|
||||||
|
}
|
||||||
|
|
||||||
|
void allowWindow(QWindow *window)
|
||||||
|
{
|
||||||
|
QPlatformNativeInterface *native = qGuiApp->platformNativeInterface();
|
||||||
|
wl_surface *surface = reinterpret_cast<wl_surface *>(native->nativeResourceForWindow(QByteArrayLiteral("surface"), window));
|
||||||
|
allow(surface);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class WidgetAllower : public QObject
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
WidgetAllower(QWidget *widget)
|
||||||
|
: QObject(widget)
|
||||||
|
, m_widget(widget)
|
||||||
|
{
|
||||||
|
widget->installEventFilter(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool eventFilter(QObject * /*watched*/, QEvent *event) override
|
||||||
|
{
|
||||||
|
if (auto w = m_widget->windowHandle()) {
|
||||||
|
WaylandAboveLockscreen aboveLockscreen;
|
||||||
|
Q_ASSERT(aboveLockscreen.isInitialized());
|
||||||
|
aboveLockscreen.allowWindow(w);
|
||||||
|
deleteLater();
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
QWidget *const m_widget;
|
||||||
|
};
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
|
||||||
|
QApplication app(argc, argv);
|
||||||
|
QWidget window1(nullptr, Qt::Window);
|
||||||
|
window1.setWindowTitle("Window 1");
|
||||||
|
window1.setLayout(new QVBoxLayout);
|
||||||
|
QPushButton p("Lock && Raise the Window 2");
|
||||||
|
window1.layout()->addWidget(&p);
|
||||||
|
window1.show();
|
||||||
|
|
||||||
|
QWidget window2(nullptr, Qt::Window);
|
||||||
|
window2.setWindowTitle("Window 2");
|
||||||
|
window2.setLayout(new QVBoxLayout);
|
||||||
|
QPushButton p2("Close");
|
||||||
|
window2.layout()->addWidget(&p2);
|
||||||
|
|
||||||
|
new WidgetAllower(&window2);
|
||||||
|
auto raiseWindow2 = [&] {
|
||||||
|
KWindowSystem::requestXdgActivationToken(window2.windowHandle(), 0, "lockscreenoverlaytest.desktop");
|
||||||
|
};
|
||||||
|
QObject::connect(KWindowSystem::self(), &KWindowSystem::xdgActivationTokenArrived, &window2, [&window2](int, const QString &token) {
|
||||||
|
KWindowSystem::setCurrentXdgActivationToken(token);
|
||||||
|
KWindowSystem::activateWindow(window2.windowHandle());
|
||||||
|
});
|
||||||
|
|
||||||
|
QObject::connect(&p, &QPushButton::clicked, &app, [&] {
|
||||||
|
QProcess::execute("loginctl", {"lock-session"});
|
||||||
|
window2.showFullScreen();
|
||||||
|
QTimer::singleShot(3000, &app, raiseWindow2);
|
||||||
|
});
|
||||||
|
|
||||||
|
QObject::connect(&p2, &QPushButton::clicked, &window2, &QWidget::close);
|
||||||
|
|
||||||
|
return app.exec();
|
||||||
|
}
|
9
tests/lockscreenoverlaytest.desktop
Normal file
9
tests/lockscreenoverlaytest.desktop
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
# copy to $(qtpaths --install-prefix)/share/applications/
|
||||||
|
# remember to change the Exec line to your builddir path
|
||||||
|
|
||||||
|
[Desktop Entry]
|
||||||
|
Exec=absolute/path/to/the/executable/bin/lockscreenoverlaytest
|
||||||
|
Type=Application
|
||||||
|
X-KDE-Wayland-Interfaces=kde_lockscreen_overlay_v1
|
||||||
|
NoDisplay=true
|
||||||
|
Name=KWin LockScreen Overay Test
|
Loading…
Reference in a new issue