/* This file is part of KDDockWidgets. SPDX-FileCopyrightText: 2019-2021 Klarälvdalens Datakonsult AB, a KDAB Group company Author: Sérgio Martins SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only Contact KDAB at for commercial licensing options. */ #ifndef KD_WIDGET_RESIZE_HANDLER_P_H #define KD_WIDGET_RESIZE_HANDLER_P_H #include "kddockwidgets/KDDockWidgets.h" #include "kddockwidgets/QWidgetAdapter.h" #include "kddockwidgets/Qt5Qt6Compat_p.h" #include "kddockwidgets/docks_export.h" #include "kddockwidgets/private/FloatingWindow_p.h" #include #include #include #include QT_BEGIN_NAMESPACE class QMouseEvent; QT_END_NAMESPACE namespace KDDockWidgets { class FloatingWindow; class DOCKS_EXPORT WidgetResizeHandler : public QObject { Q_OBJECT public: enum Feature { Feature_None = 0, Feature_NativeShadow = 1, Feature_NativeResize = 2, Feature_NativeDrag = 4, Feature_NativeMaximize = 8, Feature_All = Feature_NativeShadow | Feature_NativeResize | Feature_NativeDrag | Feature_NativeMaximize }; Q_DECLARE_FLAGS(Features, Feature); struct NativeFeatures { NativeFeatures() = default; NativeFeatures(QRect r) : htCaptionRect(r) { } NativeFeatures(Feature f) : features(f) { } NativeFeatures(Features f) : features(f) { } QRect htCaptionRect; // in global coordinates Features features = Feature_All; bool hasFeatures() const { return features != Feature_None; } bool hasShadow() const { return features & Feature_NativeShadow; } bool hasMaximize() const { return features & Feature_NativeMaximize; } bool hasResize() const { return features & Feature_NativeResize; } bool hasDrag() const { return (features & Feature_NativeDrag) && !htCaptionRect.isNull(); } }; /** * @brief CTOR * @param isTopLevelResizer If true, then this resize handler is for top-level widgets (aka windows) * if false, they are docked (like for example resizing docked MDI widgets, or the sidebar overlay) * @param target The target widget that will be resized. Also acts as parent QObject. */ explicit WidgetResizeHandler(bool isTopLevelResizer, QWidgetOrQuick *target); ~WidgetResizeHandler() override; /** * @brief Sets the sides the user is allowed to resize with mouse. * * By default the user can resize all 4 sides. * However, when a dock widget is overlayed (popuped), only one side can be resized. */ void setAllowedResizeSides(CursorPositions); /** * Sets the resize gap. By default 10. * * This is only used for non-top-level (child) widgets. * When resizing a child widget, it will be clipped by its parent, but we leave a little space so * we can resize it again. * * Meaning, if you're resizing 'bottom' of the child widget, it can never be bigger than parent.geometry().bottom() - gap. * The gap allows you to put your mouse there and resize again. */ void setResizeGap(int); bool isMDI() const; static int widgetResizeHandlerMargin(); static void setupWindow(QWindow *window); #ifdef Q_OS_WIN static bool handleWindowsNativeEvent(QWindow *w, MSG *msg, Qt5Qt6Compat::qintptr *result, const NativeFeatures & = {}); static bool handleWindowsNativeEvent(FloatingWindow *w, const QByteArray &eventType, void *message, Qt5Qt6Compat::qintptr *result); #endif static bool s_disableAllHandlers; protected: bool eventFilter(QObject *o, QEvent *e) override; private: void setTarget(QWidgetOrQuick *w); bool mouseMoveEvent(QMouseEvent *e); void updateCursor(CursorPosition m); void setMouseCursor(Qt::CursorShape cursor); void restoreMouseCursor(); CursorPosition cursorPosition(QPoint) const; QWidgetOrQuick *mTarget = nullptr; CursorPosition mCursorPos = CursorPosition_Undefined; QPoint mNewPosition; bool m_resizingInProgress = false; const bool m_isTopLevelWindowResizer; int m_resizeGap = 10; CursorPositions mAllowedResizeSides = CursorPosition_All; }; #if defined(Q_OS_WIN) && defined(KDDOCKWIDGETS_QTWIDGETS) /** * @brief Helper to rediriect WM_NCHITTEST from child widgets to the top-level widget * * To implement aero-snap the top-level window must respond to WM_NCHITTEST, we do that * in FloatingWindow::nativeEvent(). But if the child widgets have a native handle, then * the WM_NCHITTEST will go to them. They have to respond HTTRANSPARENT so the event * is redirected. * * This only affects QtWidgets, since QQuickItems never have native WId. */ class NCHITTESTEventFilter : public QAbstractNativeEventFilter { public: explicit NCHITTESTEventFilter(FloatingWindow *fw) : m_floatingWindow(fw) { } bool nativeEventFilter(const QByteArray &eventType, void *message, Qt5Qt6Compat::qintptr *result) override; QPointer m_floatingWindow; }; #endif // Q_OS_WIN class DOCKS_EXPORT CustomFrameHelper : public QObject , public QAbstractNativeEventFilter { Q_OBJECT public: typedef WidgetResizeHandler::NativeFeatures (*ShouldUseCustomFrame)(QWindow *); explicit CustomFrameHelper(ShouldUseCustomFrame shouldUseCustomFrameFunc, QObject *parent = nullptr); ~CustomFrameHelper() override; public Q_SLOTS: void applyCustomFrame(QWindow *); protected: bool nativeEventFilter(const QByteArray &eventType, void *message, Qt5Qt6Compat::qintptr *result) override; private: bool m_inDtor = false; ShouldUseCustomFrame m_shouldUseCustomFrameFunc = nullptr; }; } #endif