3493e97655
Summary: The clipboard sync is done by a dedicated helper binary launched by KWin. This helper binary is forced to xcb platform to piggy-back on Qt's implementation of the X11 clipboard. In addition it implements the Wayland clipboard - which is much simpler. Reading the Wayland clipboard is based on the implementation in QtWayland. KWin internally knows the DataDeviceInterface belonging to the helper application. Whenever an xwayland client is focussed, this DataDevice is allowed to set the selection and KWin manually updates the current selection in the SeatInterface. By that the sync from X11 to Wayland is implemented. When afterwards a Wayland client is selected, it's sent the current selection which references the X clipboard and a data transfer can be initiated in the normal Wayland way. For the other direction KWin sends the current selection to the helper's DataDevice whenever an xwayland window is focused. The helper application reads the Wayland clipboard and sets it on the X11 clipboard. Thus the Wayland clipboard is synced to X11. The approach used here will also be useful for implementing a clipboard manager (aka klipper). Currently the implementation is not yet fully completed. We need to make sure that the helper application gets restarted in case of a crash. Test Plan: See added test case Reviewers: #plasma_on_wayland, #kwin Subscribers: plasma-devel, kwin Tags: #plasma_on_wayland, #kwin Differential Revision: https://phabricator.kde.org/D1973
220 lines
6.8 KiB
C++
220 lines
6.8 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2015 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*********************************************************************/
|
|
#ifndef KWIN_WAYLAND_SERVER_H
|
|
#define KWIN_WAYLAND_SERVER_H
|
|
|
|
#include <kwinglobals.h>
|
|
|
|
#include <QObject>
|
|
#include <QPointer>
|
|
|
|
class QThread;
|
|
class QProcess;
|
|
class QWindow;
|
|
|
|
namespace KWayland
|
|
{
|
|
namespace Client
|
|
{
|
|
class ConnectionThread;
|
|
class Registry;
|
|
class ShmPool;
|
|
class Surface;
|
|
}
|
|
namespace Server
|
|
{
|
|
class ClientConnection;
|
|
class CompositorInterface;
|
|
class Display;
|
|
class DataDeviceInterface;
|
|
class ShellInterface;
|
|
class SeatInterface;
|
|
class ServerSideDecorationManagerInterface;
|
|
class SurfaceInterface;
|
|
class OutputInterface;
|
|
class PlasmaShellInterface;
|
|
class PlasmaWindowManagementInterface;
|
|
class QtSurfaceExtensionInterface;
|
|
class OutputManagementInterface;
|
|
class OutputConfigurationInterface;
|
|
}
|
|
}
|
|
|
|
namespace KWin
|
|
{
|
|
class ShellClient;
|
|
|
|
class AbstractClient;
|
|
class Toplevel;
|
|
|
|
class KWIN_EXPORT WaylandServer : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
enum class InitalizationFlag {
|
|
NoOptions = 0x0,
|
|
LockScreen = 0x1,
|
|
NoLockScreenIntegration = 0x2
|
|
};
|
|
|
|
Q_DECLARE_FLAGS(InitalizationFlags, InitalizationFlag)
|
|
|
|
virtual ~WaylandServer();
|
|
void init(const QByteArray &socketName = QByteArray(), InitalizationFlags flags = InitalizationFlag::NoOptions);
|
|
void terminateClientConnections();
|
|
|
|
KWayland::Server::Display *display() {
|
|
return m_display;
|
|
}
|
|
KWayland::Server::CompositorInterface *compositor() {
|
|
return m_compositor;
|
|
}
|
|
KWayland::Server::SeatInterface *seat() {
|
|
return m_seat;
|
|
}
|
|
KWayland::Server::ShellInterface *shell() {
|
|
return m_shell;
|
|
}
|
|
KWayland::Server::PlasmaWindowManagementInterface *windowManagement() {
|
|
return m_windowManagement;
|
|
}
|
|
KWayland::Server::ServerSideDecorationManagerInterface *decorationManager() const {
|
|
return m_decorationManager;
|
|
}
|
|
QList<ShellClient*> clients() const {
|
|
return m_clients;
|
|
}
|
|
QList<ShellClient*> internalClients() const {
|
|
return m_internalClients;
|
|
}
|
|
void removeClient(ShellClient *c);
|
|
ShellClient *findClient(quint32 id) const;
|
|
ShellClient *findClient(KWayland::Server::SurfaceInterface *surface) const;
|
|
ShellClient *findClient(QWindow *w) const;
|
|
|
|
/**
|
|
* @returns file descriptor for Xwayland to connect to.
|
|
**/
|
|
int createXWaylandConnection();
|
|
void destroyXWaylandConnection();
|
|
|
|
/**
|
|
* @returns file descriptor to the input method server's socket.
|
|
**/
|
|
int createInputMethodConnection();
|
|
void destroyInputMethodConnection();
|
|
|
|
int createXclipboardSyncConnection();
|
|
|
|
/**
|
|
* @returns true if screen is locked.
|
|
**/
|
|
bool isScreenLocked() const;
|
|
/**
|
|
* @returns whether integration with KScreenLocker is available.
|
|
**/
|
|
bool hasScreenLockerIntegration() const;
|
|
|
|
void createInternalConnection();
|
|
void initWorkspace();
|
|
|
|
KWayland::Server::ClientConnection *xWaylandConnection() const {
|
|
return m_xwayland.client;
|
|
}
|
|
KWayland::Server::ClientConnection *inputMethodConnection() const {
|
|
return m_inputMethodServerConnection;
|
|
}
|
|
KWayland::Server::ClientConnection *internalConnection() const {
|
|
return m_internalConnection.server;
|
|
}
|
|
KWayland::Server::ClientConnection *screenLockerClientConnection() const {
|
|
return m_screenLockerClientConnection;
|
|
}
|
|
QPointer<KWayland::Server::DataDeviceInterface> xclipboardSyncDataDevice() const {
|
|
return m_xclipbaordSync.ddi;
|
|
}
|
|
KWayland::Client::ShmPool *internalShmPool() {
|
|
return m_internalConnection.shm;
|
|
}
|
|
KWayland::Client::ConnectionThread *internalClientConection() {
|
|
return m_internalConnection.client;
|
|
}
|
|
KWayland::Client::Registry *internalClientRegistry() {
|
|
return m_internalConnection.registry;
|
|
}
|
|
void dispatch();
|
|
quint32 createWindowId(KWayland::Server::SurfaceInterface *surface);
|
|
|
|
Q_SIGNALS:
|
|
void shellClientAdded(KWin::ShellClient*);
|
|
void shellClientRemoved(KWin::ShellClient*);
|
|
void terminatingInternalClientConnection();
|
|
|
|
private:
|
|
void setupX11ClipboardSync();
|
|
void shellClientShown(Toplevel *t);
|
|
void initOutputs();
|
|
quint16 createClientId(KWayland::Server::ClientConnection *c);
|
|
void destroyInternalConnection();
|
|
void configurationChangeRequested(KWayland::Server::OutputConfigurationInterface *config);
|
|
KWayland::Server::Display *m_display = nullptr;
|
|
KWayland::Server::CompositorInterface *m_compositor = nullptr;
|
|
KWayland::Server::SeatInterface *m_seat = nullptr;
|
|
KWayland::Server::ShellInterface *m_shell = nullptr;
|
|
KWayland::Server::PlasmaShellInterface *m_plasmaShell = nullptr;
|
|
KWayland::Server::PlasmaWindowManagementInterface *m_windowManagement = nullptr;
|
|
KWayland::Server::QtSurfaceExtensionInterface *m_qtExtendedSurface = nullptr;
|
|
KWayland::Server::ServerSideDecorationManagerInterface *m_decorationManager = nullptr;
|
|
KWayland::Server::OutputManagementInterface *m_outputManagement = nullptr;
|
|
struct {
|
|
KWayland::Server::ClientConnection *client = nullptr;
|
|
QMetaObject::Connection destroyConnection;
|
|
} m_xwayland;
|
|
KWayland::Server::ClientConnection *m_inputMethodServerConnection = nullptr;
|
|
KWayland::Server::ClientConnection *m_screenLockerClientConnection = nullptr;
|
|
struct {
|
|
KWayland::Server::ClientConnection *server = nullptr;
|
|
KWayland::Client::ConnectionThread *client = nullptr;
|
|
QThread *clientThread = nullptr;
|
|
KWayland::Client::Registry *registry = nullptr;
|
|
KWayland::Client::ShmPool *shm = nullptr;
|
|
|
|
} m_internalConnection;
|
|
struct {
|
|
QProcess *process = nullptr;
|
|
KWayland::Server::ClientConnection *client = nullptr;
|
|
QPointer<KWayland::Server::DataDeviceInterface> ddi;
|
|
} m_xclipbaordSync;
|
|
QList<ShellClient*> m_clients;
|
|
QList<ShellClient*> m_internalClients;
|
|
QHash<KWayland::Server::ClientConnection*, quint16> m_clientIds;
|
|
InitalizationFlags m_initFlags;
|
|
KWIN_SINGLETON(WaylandServer)
|
|
};
|
|
|
|
inline
|
|
WaylandServer *waylandServer() {
|
|
return WaylandServer::self();
|
|
}
|
|
|
|
} // namespace KWin
|
|
|
|
#endif
|
|
|