/* 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 #include #include #include QT_BEGIN_NAMESPACE class QMouseEvent; QT_END_NAMESPACE namespace KDDockWidgets { class FloatingWindow; class DOCKS_EXPORT WidgetResizeHandler : public QObject { Q_OBJECT public: /** * @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(); #ifdef Q_OS_WIN 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 } #endif