/* KWin - the KDE window manager This file is part of the KDE project. SPDX-FileCopyrightText: 1999, 2000 Matthias Ettrich SPDX-FileCopyrightText: 2003 Lubos Lunak SPDX-License-Identifier: GPL-2.0-or-later */ #pragma once // kwin #include "decorationitem.h" #include "utils/xcbutils.h" #include "window.h" // Qt #include #include #include #include #include // X #include // TODO: Cleanup the order of things in this .h file class QTimer; class KStartupInfoData; class KStartupInfoId; namespace KWin { /** * @brief Defines Predicates on how to search for a Client. * * Used by Workspace::findClient. */ enum class Predicate { WindowMatch, WrapperIdMatch, FrameIdMatch, InputIdMatch, }; /** * @todo Remove when the X11 platform support is dropped. This decoration renderer * will be used if compositing is off. */ class X11DecorationRenderer : public DecorationRenderer { Q_OBJECT public: explicit X11DecorationRenderer(Decoration::DecoratedClientImpl *client); ~X11DecorationRenderer() override; protected: void render(const QRegion ®ion) override; private: void update(); QTimer *m_scheduleTimer; xcb_gcontext_t m_gc; }; class KWIN_EXPORT X11Window : public Window { Q_OBJECT /** * By how much the window wishes to grow/shrink at least. Usually QSize(1,1). * MAY BE DISOBEYED BY THE WM! It's only for information, do NOT rely on it at all. * The value is evaluated each time the getter is called. * Because of that no changed signal is provided. */ Q_PROPERTY(QSizeF basicUnit READ basicUnit) /** * A client can block compositing. That is while the Client is alive and the state is set, * Compositing is suspended and is resumed when there are no Clients blocking compositing any * more. * * This is actually set by a window property, unfortunately not used by the target application * group. For convenience it's exported as a property to the scripts. * * Use with care! */ Q_PROPERTY(bool blocksCompositing READ isBlockingCompositing WRITE setBlockingCompositing NOTIFY blockingCompositingChanged) /** * Whether the Client uses client side window decorations. * Only GTK+ are detected. */ Q_PROPERTY(bool clientSideDecorated READ isClientSideDecorated NOTIFY clientSideDecoratedChanged) public: explicit X11Window(); ~X11Window() override; ///< Use destroyWindow() or releaseWindow() xcb_window_t wrapperId() const; xcb_window_t inputId() const { return m_decoInputExtent; } xcb_window_t frameId() const override; QRectF inputGeometry() const override; QPointF framePosToClientPos(const QPointF &point) const override; QPointF clientPosToFramePos(const QPointF &point) const override; QSizeF frameSizeToClientSize(const QSizeF &size) const override; QSizeF clientSizeToFrameSize(const QSizeF &size) const override; QRectF frameRectToBufferRect(const QRectF &rect) const; QSizeF implicitSize() const; QMatrix4x4 inputTransformation() const override; bool isTransient() const override; bool groupTransient() const override; bool wasOriginallyGroupTransient() const; QList mainWindows() const override; // Call once before loop , is not indirect bool hasTransient(const Window *c, bool indirect) const override; void checkTransient(xcb_window_t w); Window *findModal(bool allow_itself = false) override; const Group *group() const override; Group *group() override; void checkGroup(Group *gr = nullptr, bool force = false); void changeClientLeaderGroup(Group *gr); bool supportsWindowRules() const override; void updateWindowRules(Rules::Types selection) override; void applyWindowRules() override; void updateFullscreenMonitors(NETFullscreenMonitors topology); bool hasNETSupport() const; QSizeF minSize() const override; QSizeF maxSize() const override; QSizeF basicUnit() const; QPointF inputPos() const { return input_offset; } // Inside of geometry() bool windowEvent(xcb_generic_event_t *e); NET::WindowType windowType(bool direct = false, int supported_types = 0) const override; bool manage(xcb_window_t w, bool isMapped); void releaseWindow(bool on_shutdown = false); void destroyWindow() override; QStringList activities() const override; void doSetOnActivities(const QStringList &newActivitiesList) override; void updateActivities(bool includeTransients) override; /// Is not minimized and not hidden. I.e. normally visible on some virtual desktop. bool isShown() const override; bool isHiddenInternal() const override; // For compositing bool isShadeable() const override; bool isMaximizable() const override; MaximizeMode maximizeMode() const override; bool isMinimizable() const override; QRectF iconGeometry() const override; bool isFullScreenable() const override; void setFullScreen(bool set, bool user = true) override; bool isFullScreen() const override; bool userCanSetFullScreen() const override; int fullScreenMode() const { return m_fullscreenMode; // only for session saving } bool userNoBorder() const; bool noBorder() const override; void setNoBorder(bool set) override; bool userCanSetNoBorder() const override; void checkNoBorder() override; int sessionStackingOrder() const; // Auxiliary functions, depend on the windowType bool wantsInput() const override; bool isResizable() const override; bool isMovable() const override; bool isMovableAcrossScreens() const override; bool isCloseable() const override; ///< May be closed by the user (May have a close button) bool takeFocus() override; void invalidateDecoration() override; void updateShape(); /// resizeWithChecks() resizes according to gravity, and checks workarea position QRectF resizeWithChecks(const QRectF &geometry, const QSizeF &size) override; QRectF resizeWithChecks(const QRectF &geometry, qreal w, qreal h, xcb_gravity_t gravity); QRectF resizeWithChecks(const QRectF &geometry, const QSizeF &s, xcb_gravity_t gravity); QSizeF constrainClientSize(const QSizeF &size, SizeMode mode = SizeModeAny) const override; bool providesContextHelp() const override; xcb_colormap_t colormap() const; /// Updates visibility depending on being shaded, virtual desktop, etc. void updateVisibility(); /// Hides a client - Basically like minimize, but without effects, it's simply hidden void hideClient() override; void showClient() override; bool hiddenPreview() const; ///< Window is mapped in order to get a window pixmap bool setupCompositing() override; void finishCompositing(ReleaseReason releaseReason = ReleaseReason::Release) override; void setBlockingCompositing(bool block); inline bool isBlockingCompositing() { return blocks_compositing; } QString captionNormal() const override { return cap_normal; } QString captionSuffix() const override { return cap_suffix; } using Window::keyPressEvent; void keyPressEvent(uint key_code, xcb_timestamp_t time); // FRAME ?? void updateMouseGrab() override; xcb_window_t moveResizeGrabWindow() const; QPointF gravityAdjustment(xcb_gravity_t gravity) const; const QPointF calculateGravitation(bool invert) const; void NETMoveResize(qreal x_root, qreal y_root, NET::Direction direction); void NETMoveResizeWindow(int flags, qreal x, qreal y, qreal width, qreal height); void GTKShowWindowMenu(qreal x_root, qreal y_root); void restackWindow(xcb_window_t above, int detail, NET::RequestSource source, xcb_timestamp_t timestamp, bool send_event = false); void gotPing(xcb_timestamp_t timestamp); void updateUserTime(xcb_timestamp_t time = XCB_TIME_CURRENT_TIME); xcb_timestamp_t userTime() const override; bool hasUserTimeSupport() const; /// Does 'delete c;' static void deleteClient(X11Window *c); static bool belongToSameApplication(const X11Window *c1, const X11Window *c2, SameApplicationChecks checks = SameApplicationChecks()); static bool sameAppWindowRoleMatch(const X11Window *c1, const X11Window *c2, bool active_hack); void killWindow() override; void showContextHelp() override; void checkActiveModal(); StrutRect strutRect(StrutArea area) const override; bool hasStrut() const override; bool isClientSideDecorated() const; Xcb::Property fetchFirstInTabBox() const; void readFirstInTabBox(Xcb::Property &property); void updateFirstInTabBox(); Xcb::StringProperty fetchPreferredColorScheme() const; QString readPreferredColorScheme(Xcb::StringProperty &property) const; QString preferredColorScheme() const override; // sets whether the client should be faked as being on all activities (and be shown during session save) void setSessionActivityOverride(bool needed); bool isClient() const override; void cancelFocusOutTimer(); /** * Restores the Client after it had been hidden due to show on screen edge functionality. * In addition the property gets deleted so that the Client knows that it is visible again. */ void showOnScreenEdge() override; Xcb::StringProperty fetchApplicationMenuServiceName() const; void readApplicationMenuServiceName(Xcb::StringProperty &property); void checkApplicationMenuServiceName(); Xcb::StringProperty fetchApplicationMenuObjectPath() const; void readApplicationMenuObjectPath(Xcb::StringProperty &property); void checkApplicationMenuObjectPath(); struct SyncRequest { xcb_sync_counter_t counter; xcb_sync_int64_t value; xcb_sync_alarm_t alarm; xcb_timestamp_t lastTimestamp; QTimer *timeout, *failsafeTimeout; bool isPending; bool interactiveResize; }; const SyncRequest &syncRequest() const { return m_syncRequest; } virtual bool wantsSyncCounter() const; void handleSync(); void handleSyncTimeout(); bool allowWindowActivation(xcb_timestamp_t time = -1U, bool focus_in = false, bool ignore_desktop = false); static void cleanupX11(); public Q_SLOTS: void closeWindow() override; void updateCaption() override; private: // Handlers for X11 events bool mapRequestEvent(xcb_map_request_event_t *e); void unmapNotifyEvent(xcb_unmap_notify_event_t *e); void destroyNotifyEvent(xcb_destroy_notify_event_t *e); void configureRequestEvent(xcb_configure_request_event_t *e); void propertyNotifyEvent(xcb_property_notify_event_t *e) override; void clientMessageEvent(xcb_client_message_event_t *e) override; void enterNotifyEvent(xcb_enter_notify_event_t *e); void leaveNotifyEvent(xcb_leave_notify_event_t *e); void focusInEvent(xcb_focus_in_event_t *e); void focusOutEvent(xcb_focus_out_event_t *e); void damageNotifyEvent(); bool buttonPressEvent(xcb_window_t w, int button, int state, int x, int y, int x_root, int y_root, xcb_timestamp_t time = XCB_CURRENT_TIME); bool buttonReleaseEvent(xcb_window_t w, int button, int state, int x, int y, int x_root, int y_root); bool motionNotifyEvent(xcb_window_t w, int state, int x, int y, int x_root, int y_root); protected: bool belongsToSameApplication(const Window *other, SameApplicationChecks checks) const override; void doSetActive() override; void doSetKeepAbove() override; void doSetKeepBelow() override; void doSetShade(ShadeMode previousShadeMode) override; void doSetDesktop() override; void doMinimize() override; void doSetSkipPager() override; void doSetSkipTaskbar() override; void doSetSkipSwitcher() override; void doSetDemandsAttention() override; bool belongsToDesktop() const override; bool doStartInteractiveMoveResize() override; bool isWaitingForInteractiveMoveResizeSync() const override; void doInteractiveResizeSync(const QRectF &rect) override; QSizeF resizeIncrements() const override; bool acceptsFocus() const override; void moveResizeInternal(const QRectF &rect, MoveResizeMode mode) override; WindowItem *createItem() override; // Signals for the scripting interface // Signals make an excellent way for communication // in between objects as compared to simple function // calls Q_SIGNALS: void clientManaging(KWin::X11Window *); void clientFullScreenSet(KWin::X11Window *, bool, bool); /** * Emitted whenever the Client want to show it menu */ void showRequest(); /** * Emitted whenever the Client's menu is closed */ void menuHidden(); /** * Emitted whenever the Client's menu is available */ void appMenuAvailable(); /** * Emitted whenever the Client's menu is unavailable */ void appMenuUnavailable(); /** * Emitted whenever the Client's block compositing state changes. */ void blockingCompositingChanged(KWin::X11Window *client); void clientSideDecoratedChanged(); private: void exportMappingState(int s); // ICCCM 4.1.3.1, 4.1.4, NETWM 2.5.1 bool isManaged() const; ///< Returns false if this client is not yet managed void updateAllowedActions(bool force = false); QRect fullscreenMonitorsArea(NETFullscreenMonitors topology) const; void changeMaximize(bool horizontal, bool vertical, bool adjust) override; void getWmNormalHints(); void getMotifHints(); void getIcons(); void fetchName(); void fetchIconicName(); QString readName() const; void setCaption(const QString &s, bool force = false); bool hasTransientInternal(const X11Window *c, bool indirect, QList &set) const; void setShortcutInternal() override; void configureRequest(int value_mask, qreal rx, qreal ry, qreal rw, qreal rh, int gravity, bool from_tool); NETExtendedStrut strut() const; int checkShadeGeometry(int w, int h); void getSyncCounter(); void sendSyncRequest(); void leaveInteractiveMoveResize() override; void performInteractiveResize(); void establishCommandWindowGrab(uint8_t button); void establishCommandAllGrab(uint8_t button); void resizeDecoration(); void pingWindow(); void killProcess(bool ask, xcb_timestamp_t timestamp = XCB_TIME_CURRENT_TIME); void updateUrgency(); static void sendClientMessage(xcb_window_t w, xcb_atom_t a, xcb_atom_t protocol, uint32_t data1 = 0, uint32_t data2 = 0, uint32_t data3 = 0); void embedClient(xcb_window_t w, xcb_visualid_t visualid, xcb_colormap_t colormap, uint8_t depth); void detectNoBorder(); void updateFrameExtents(); void setClientFrameExtents(const NETStrut &strut); void internalShow(); void internalHide(); void internalKeep(); void map(); void unmap(); void updateHiddenPreview(); void updateInputShape(); void updateServerGeometry(); void discardWindowPixmap(); void updateWindowPixmap(); xcb_timestamp_t readUserTimeMapTimestamp(const KStartupInfoId *asn_id, const KStartupInfoData *asn_data, bool session) const; xcb_timestamp_t readUserCreationTime() const; void startupIdChanged(); void updateInputWindow(); Xcb::Property fetchShowOnScreenEdge() const; void readShowOnScreenEdge(Xcb::Property &property); /** * Reads the property and creates/destroys the screen edge if required * and shows/hides the client. */ void updateShowOnScreenEdge(); void maybeCreateX11DecorationRenderer(); void maybeDestroyX11DecorationRenderer(); void updateDecoration(bool check_workspace_pos, bool force = false); void createDecoration(const QRectF &oldgeom); void destroyDecoration(); Xcb::Window m_client; Xcb::Window m_wrapper; Xcb::Window m_frame; int m_activityUpdatesBlocked; bool m_blockedActivityUpdatesRequireTransients; Xcb::Window m_moveResizeGrabWindow; bool move_resize_has_keyboard_grab; bool m_managed; Xcb::GeometryHints m_geometryHints; void sendSyntheticConfigureNotify(); enum MappingState { Withdrawn, ///< Not handled, as per ICCCM WithdrawnState Mapped, ///< The frame is mapped Unmapped, ///< The frame is not mapped Kept ///< The frame should be unmapped, but is kept (For compositing) }; MappingState mapping_state; Xcb::TransientFor fetchTransient() const; void readTransientProperty(Xcb::TransientFor &transientFor); void readTransient(); xcb_window_t verifyTransientFor(xcb_window_t transient_for, bool set); void addTransient(Window *cl) override; void removeTransient(Window *cl) override; void removeFromMainClients(); void cleanGrouping(); void checkGroupTransients(); void setTransient(xcb_window_t new_transient_for_id); xcb_window_t m_transientForId; xcb_window_t m_originalTransientForId; X11Window *shade_below; Xcb::MotifHints m_motif; uint hidden : 1; ///< Forcibly hidden by calling hide() uint noborder : 1; uint app_noborder : 1; ///< App requested no border via window type, shape extension, etc. uint ignore_focus_stealing : 1; ///< Don't apply focus stealing prevention to this client bool blocks_compositing; enum FullScreenMode { FullScreenNone, FullScreenNormal } m_fullscreenMode; MaximizeMode max_mode; xcb_colormap_t m_colormap; QString cap_normal, cap_iconic, cap_suffix; Group *in_group; QTimer *ping_timer; qint64 m_killHelperPID; xcb_timestamp_t m_pingTimestamp; xcb_timestamp_t m_userTime; NET::Actions allowed_actions; bool shade_geometry_change; SyncRequest m_syncRequest; static bool check_active_modal; ///< \see X11Window::checkActiveModal() int sm_stacking_order; friend struct ResetupRulesProcedure; friend bool performTransiencyCheck(); Xcb::StringProperty fetchActivities() const; void readActivities(Xcb::StringProperty &property); void checkActivities(); bool activitiesDefined; // whether the x property was actually set bool sessionActivityOverride; bool needsXWindowMove; Xcb::Window m_decoInputExtent; QPointF input_offset; QTimer *m_focusOutTimer; QMetaObject::Connection m_edgeRemoveConnection; QMetaObject::Connection m_edgeGeometryTrackingConnection; QMarginsF m_clientFrameExtents; Output *m_lastOutput = nullptr; QRectF m_lastBufferGeometry; QRectF m_lastFrameGeometry; QRectF m_lastClientGeometry; std::unique_ptr m_decorationRenderer; }; inline xcb_window_t X11Window::wrapperId() const { return m_wrapper; } inline bool X11Window::isClientSideDecorated() const { return !m_clientFrameExtents.isNull(); } inline bool X11Window::groupTransient() const { return m_transientForId == kwinApp()->x11RootWindow(); } // Needed because verifyTransientFor() may set transient_for_id to root window, // if the original value has a problem (window doesn't exist, etc.) inline bool X11Window::wasOriginallyGroupTransient() const { return m_originalTransientForId == kwinApp()->x11RootWindow(); } inline bool X11Window::isTransient() const { return m_transientForId != XCB_WINDOW_NONE; } inline const Group *X11Window::group() const { return in_group; } inline Group *X11Window::group() { return in_group; } inline bool X11Window::isShown() const { return !isMinimized() && !hidden; } inline bool X11Window::isHiddenInternal() const { return hidden; } inline MaximizeMode X11Window::maximizeMode() const { return max_mode; } inline bool X11Window::isFullScreen() const { return m_fullscreenMode != FullScreenNone; } inline bool X11Window::hasNETSupport() const { return info->hasNETSupport(); } inline xcb_colormap_t X11Window::colormap() const { return m_colormap; } inline int X11Window::sessionStackingOrder() const { return sm_stacking_order; } inline bool X11Window::isManaged() const { return m_managed; } inline QRectF X11Window::resizeWithChecks(const QRectF &geometry, const QSizeF &s) { return resizeWithChecks(geometry, s.width(), s.height(), XCB_GRAVITY_BIT_FORGET); } inline QRectF X11Window::resizeWithChecks(const QRectF &geometry, const QSizeF &s, xcb_gravity_t gravity) { return resizeWithChecks(geometry, s.width(), s.height(), gravity); } inline bool X11Window::hasUserTimeSupport() const { return info->userTime() != -1U; } inline xcb_window_t X11Window::moveResizeGrabWindow() const { return m_moveResizeGrabWindow; } inline bool X11Window::hiddenPreview() const { return mapping_state == Kept; } } // namespace Q_DECLARE_METATYPE(KWin::X11Window *) Q_DECLARE_METATYPE(QList)