// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only #ifndef QWAYLANDDISPLAY_H #define QWAYLANDDISPLAY_H // // W A R N I N G // ------------- // // This file is not part of the Qt API. It exists purely as an // implementation detail. This header file may change from version to // version without notice, or even be removed. // // We mean it. // #include #include #include #include #include #include #include #include #include #include #include #include #if QT_CONFIG(xkbcommon) #include #endif struct wl_cursor_image; struct wp_viewport; QT_BEGIN_NAMESPACE #define WAYLAND_IM_KEY "wayland" class QAbstractEventDispatcher; class QSocketNotifier; class QPlatformScreen; class QPlatformPlaceholderScreen; namespace QtWayland { class zwp_text_input_manager_v1; class zwp_text_input_manager_v2; class zwp_text_input_manager_v3; class qt_text_input_method_manager_v1; class wp_cursor_shape_manager_v1; class wp_fractional_scale_manager_v1; class wp_viewporter; class xdg_toplevel_drag_manager_v1; } namespace QtWaylandClient { Q_WAYLANDCLIENT_EXPORT Q_DECLARE_LOGGING_CATEGORY(lcQpaWayland); class QWaylandInputDevice; class QWaylandBuffer; class QWaylandScreen; class QWaylandXdgOutputManagerV1; class QWaylandClientBufferIntegration; class QWaylandWindowManagerIntegration; class QWaylandDataDeviceManager; #if QT_CONFIG(wayland_client_primary_selection) class QWaylandPrimarySelectionDeviceManagerV1; #endif #if QT_CONFIG(tabletevent) class QWaylandTabletManagerV2; #endif class QWaylandPointerGestures; class QWaylandTouchExtension; class QWaylandQtKeyExtension; class QWaylandWindow; class QWaylandIntegration; class QWaylandHardwareIntegration; class QWaylandSurface; class QWaylandShellIntegration; class QWaylandCursor; class QWaylandCursorTheme; class EventThread; typedef void (*RegistryListener)(void *data, struct wl_registry *registry, uint32_t id, const QString &interface, uint32_t version); class Q_WAYLANDCLIENT_EXPORT QWaylandDisplay : public QObject, public QtWayland::wl_registry { Q_OBJECT public: QWaylandDisplay(QWaylandIntegration *waylandIntegration); ~QWaylandDisplay(void) override; bool initialize(); #if QT_CONFIG(xkbcommon) struct xkb_context *xkbContext() const { return mXkbContext.get(); } #endif QList screens() const { return mScreens; } QPlatformPlaceholderScreen *placeholderScreen() const { return mPlaceholderScreen; } void ensureScreen(); QWaylandScreen *screenForOutput(struct wl_output *output) const; void handleScreenInitialized(QWaylandScreen *screen); struct wl_surface *createSurface(void *handle); struct ::wl_region *createRegion(const QRegion &qregion); struct ::wl_subsurface *createSubSurface(QWaylandWindow *window, QWaylandWindow *parent); struct ::wp_viewport *createViewport(QWaylandWindow *window); QWaylandShellIntegration *shellIntegration() const; QWaylandClientBufferIntegration *clientBufferIntegration() const; QWaylandWindowManagerIntegration *windowManagerIntegration() const; #if QT_CONFIG(cursor) QWaylandCursor *waylandCursor(); QWaylandCursorTheme *loadCursorTheme(const QString &name, int pixelSize); #endif struct wl_display *wl_display() const { return mDisplay; } struct ::wl_registry *wl_registry() { return object(); } QtWayland::wl_compositor *compositor() { return mGlobals.compositor.get(); } QList inputDevices() const { return mInputDevices; } QWaylandInputDevice *defaultInputDevice() const; QWaylandInputDevice *currentInputDevice() const { return defaultInputDevice(); } #if QT_CONFIG(wayland_datadevice) QWaylandDataDeviceManager *dndSelectionHandler() const { return mGlobals.dndSelectionHandler.get(); } #endif #if QT_CONFIG(wayland_client_primary_selection) QWaylandPrimarySelectionDeviceManagerV1 *primarySelectionManager() const { return mGlobals.primarySelectionManager.get(); } #endif #if QT_CONFIG(tabletevent) QWaylandTabletManagerV2 *tabletManager() const { return mGlobals.tabletManager.get(); } #endif QWaylandPointerGestures *pointerGestures() const { return mGlobals.pointerGestures.get(); } QWaylandTouchExtension *touchExtension() const { return mGlobals.touchExtension.get(); } QtWayland::qt_text_input_method_manager_v1 *textInputMethodManager() const { return mGlobals.textInputMethodManager.get(); } QtWayland::zwp_text_input_manager_v1 *textInputManagerv1() const { return mGlobals.textInputManagerv1.get(); } QtWayland::zwp_text_input_manager_v2 *textInputManagerv2() const { return mGlobals.textInputManagerv2.get(); } QtWayland::zwp_text_input_manager_v3 *textInputManagerv3() const { return mGlobals.textInputManagerv3.get(); } QWaylandHardwareIntegration *hardwareIntegration() const { return mGlobals.hardwareIntegration.get(); } QWaylandXdgOutputManagerV1 *xdgOutputManager() const { return mGlobals.xdgOutputManager.get(); } QtWayland::wp_fractional_scale_manager_v1 *fractionalScaleManager() const { return mGlobals.fractionalScaleManager.get(); } QtWayland::wp_viewporter *viewporter() const { return mGlobals.viewporter.get(); } QtWayland::wp_cursor_shape_manager_v1 *cursorShapeManager() const { return mGlobals.cursorShapeManager.get(); } QtWayland::xdg_toplevel_drag_manager_v1 *xdgToplevelDragManager() const { return mGlobals.xdgToplevelDragManager.get(); } struct RegistryGlobal { uint32_t id; QString interface; uint32_t version; struct ::wl_registry *registry = nullptr; RegistryGlobal(uint32_t id_, const QString &interface_, uint32_t version_, struct ::wl_registry *registry_) : id(id_), interface(interface_), version(version_), registry(registry_) { } }; QList globals() const { return mRegistryGlobals; } bool hasRegistryGlobal(QStringView interfaceName) const; /* wl_registry_add_listener does not add but rather sets a listener, so this function is used * to enable many listeners at once. */ void addRegistryListener(RegistryListener listener, void *data); void removeListener(RegistryListener listener, void *data); QWaylandShm *shm() const { return mGlobals.shm.get(); } void forceRoundTrip(); bool supportsWindowDecoration() const; uint32_t lastInputSerial() const { return mLastInputSerial; } QWaylandInputDevice *lastInputDevice() const { return mLastInputDevice; } QWaylandWindow *lastInputWindow() const; void setLastInputDevice(QWaylandInputDevice *device, uint32_t serial, QWaylandWindow *window); bool isWindowActivated(const QWaylandWindow *window); void handleWindowActivated(QWaylandWindow *window); void handleWindowDeactivated(QWaylandWindow *window); void handleKeyboardFocusChanged(QWaylandInputDevice *inputDevice); void handleWindowDestroyed(QWaylandWindow *window); wl_event_queue *frameEventQueue() { return m_frameEventQueue; }; bool isKeyboardAvailable() const; bool isWaylandInputContextRequested() const; void initEventThread(); public Q_SLOTS: void blockingReadEvents(); void flushRequests(); Q_SIGNALS: void connected(); void globalAdded(const RegistryGlobal &global); void globalRemoved(const RegistryGlobal &global); private: void checkWaylandError(); void reconnect(); void setupConnection(); void handleWaylandSync(); void requestWaylandSync(); void checkTextInputProtocol(); struct Listener { Listener() = default; Listener(RegistryListener incomingListener, void* incomingData) : listener(incomingListener), data(incomingData) {} RegistryListener listener = nullptr; void *data = nullptr; }; struct wl_display *mDisplay = nullptr; std::unique_ptr m_eventThread; wl_event_queue *m_frameEventQueue = nullptr; QScopedPointer m_frameEventQueueThread; QList mWaitingScreens; QList mScreens; QPlatformPlaceholderScreen *mPlaceholderScreen = nullptr; QList mInputDevices; QList mRegistryListeners; QWaylandIntegration *mWaylandIntegration = nullptr; #if QT_CONFIG(cursor) struct WaylandCursorTheme { QString name; int pixelSize; std::unique_ptr theme; }; std::vector mCursorThemes; struct FindExistingCursorThemeResult { std::vector::const_iterator position; bool found; QWaylandCursorTheme *theme() const noexcept { return found ? position->theme.get() : nullptr; } }; FindExistingCursorThemeResult findExistingCursorTheme(const QString &name, int pixelSize) const noexcept; QScopedPointer mCursor; #endif struct GlobalHolder { std::unique_ptr compositor; std::unique_ptr shm; #if QT_CONFIG(wayland_datadevice) std::unique_ptr dndSelectionHandler; #endif std::unique_ptr subCompositor; std::unique_ptr touchExtension; std::unique_ptr qtKeyExtension; #if QT_CONFIG(tabletevent) std::unique_ptr tabletManager; #endif std::unique_ptr pointerGestures; #if QT_CONFIG(wayland_client_primary_selection) std::unique_ptr primarySelectionManager; #endif std::unique_ptr textInputMethodManager; std::unique_ptr textInputManagerv1; std::unique_ptr textInputManagerv2; std::unique_ptr textInputManagerv3; std::unique_ptr hardwareIntegration; std::unique_ptr xdgOutputManager; std::unique_ptr viewporter; std::unique_ptr fractionalScaleManager; std::unique_ptr cursorShapeManager; std::unique_ptr xdgToplevelDragManager; std::unique_ptr windowManagerIntegration; } mGlobals; int mFd = -1; int mWritableNotificationFd = -1; QList mRegistryGlobals; uint32_t mLastInputSerial = 0; QWaylandInputDevice *mLastInputDevice = nullptr; QPointer mLastInputWindow; QPointer mLastKeyboardFocus; QList mActiveWindows; struct wl_callback *mSyncCallback = nullptr; static const wl_callback_listener syncCallbackListener; bool mWaylandTryReconnect = false; bool mWaylandInputContextRequested = [] () { const auto requested = QPlatformInputContextFactory::requested(); return requested.isEmpty() || requested.contains(QLatin1String(WAYLAND_IM_KEY)); }(); QStringList mTextInputManagerList; int mTextInputManagerIndex = INT_MAX; void registry_global(uint32_t id, const QString &interface, uint32_t version) override; void registry_global_remove(uint32_t id) override; #if QT_CONFIG(xkbcommon) QXkbCommon::ScopedXKBContext mXkbContext; #endif friend class QWaylandIntegration; }; } QT_END_NAMESPACE #endif // QWAYLANDDISPLAY_H