From 3871bfc5d3aff45e498fa2944c27e6eb5d146c8e Mon Sep 17 00:00:00 2001
From: SineStriker <trueful@163.com>
Date: 周三, 20 12月 2023 19:56:32 +0800
Subject: [PATCH] Add mac hot-switch implementations

---
 examples/mainwindow/mainwindow.cpp |  222 +++++++++++++++++++++++++++++++++++++++++++++++--------
 1 files changed, 188 insertions(+), 34 deletions(-)

diff --git a/examples/mainwindow/mainwindow.cpp b/examples/mainwindow/mainwindow.cpp
index 65c4b12..e289c3e 100644
--- a/examples/mainwindow/mainwindow.cpp
+++ b/examples/mainwindow/mainwindow.cpp
@@ -1,17 +1,23 @@
 #include "mainwindow.h"
 
 #include <QtCore/QDebug>
+#include <QtCore/QFile>
 #include <QtCore/QTime>
+#include <QtCore/QTimer>
+#include <QtWidgets/QApplication>
+#include <QtWidgets/QStyle>
 #include <QtWidgets/QPushButton>
 
 #include <QWKWidgets/widgetwindowagent.h>
 
 #include <widgetframe/windowbar.h>
+#include <widgetframe/windowbutton.h>
 
-class ClockWidget : public QPushButton {
+class ClockWidget : public QLabel {
 public:
-    explicit ClockWidget(QWidget *parent = nullptr) : QPushButton(parent) {
+    explicit ClockWidget(QWidget *parent = nullptr) : QLabel(parent) {
         startTimer(100);
+        setAlignment(Qt::AlignCenter);
     }
 
     ~ClockWidget() override = default;
@@ -24,73 +30,221 @@
 
 MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) {
     installWindowAgent();
+
+    auto clockWidget = new ClockWidget();
+    clockWidget->setObjectName(QStringLiteral("clock-widget"));
+    clockWidget->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
+    setCentralWidget(clockWidget);
+
+    loadStyleSheet(Dark);
+
+    setWindowTitle(tr("Example MainWindow"));
+    resize(800, 600);
 }
 
-MainWindow::~MainWindow() {
+static inline void emulateLeaveEvent(QWidget *widget) {
+    Q_ASSERT(widget);
+    if (!widget) {
+        return;
+    }
+    QTimer::singleShot(0, widget, [widget]() {
+#if (QT_VERSION >= QT_VERSION_CHECK(5, 14, 0))
+        const QScreen *screen = widget->screen();
+#else
+        const QScreen *screen = widget->windowHandle()->screen();
+#endif
+        const QPoint globalPos = QCursor::pos(screen);
+        if (!QRect(widget->mapToGlobal(QPoint{0, 0}), widget->size()).contains(globalPos)) {
+            QCoreApplication::postEvent(widget, new QEvent(QEvent::Leave));
+            if (widget->testAttribute(Qt::WA_Hover)) {
+                const QPoint localPos = widget->mapFromGlobal(globalPos);
+                const QPoint scenePos = widget->window()->mapFromGlobal(globalPos);
+                static constexpr const auto oldPos = QPoint{};
+                const Qt::KeyboardModifiers modifiers = QGuiApplication::keyboardModifiers();
+#if (QT_VERSION >= QT_VERSION_CHECK(6, 4, 0))
+                const auto event =
+                    new QHoverEvent(QEvent::HoverLeave, scenePos, globalPos, oldPos, modifiers);
+                Q_UNUSED(localPos);
+#elif (QT_VERSION >= QT_VERSION_CHECK(6, 3, 0))
+                const auto event =  new QHoverEvent(QEvent::HoverLeave, localPos, globalPos, oldPos, modifiers);
+                Q_UNUSED(scenePos);
+#else
+                const auto event =  new QHoverEvent(QEvent::HoverLeave, localPos, oldPos, modifiers);
+                Q_UNUSED(scenePos);
+#endif
+                QCoreApplication::postEvent(widget, event);
+            }
+        }
+    });
+}
+
+MainWindow::~MainWindow() = default;
+
+bool MainWindow::event(QEvent *event) {
+    switch (event->type()) {
+        case QEvent::WindowActivate: {
+            auto menu = menuWidget();
+            menu->setProperty("bar-active", true);
+            style()->polish(menu);
+            break;
+        }
+
+        case QEvent::WindowDeactivate: {
+            auto menu = menuWidget();
+            menu->setProperty("bar-active", false);
+            style()->polish(menu);
+            break;
+        }
+
+        default:
+            break;
+    }
+    return QMainWindow::event(event);
 }
 
 void MainWindow::installWindowAgent() {
+    // 1. Setup window agent
     auto agent = new QWK::WidgetWindowAgent(this);
-    if (!agent->setup(this)) {
-        qDebug() << "Frameless handle failed to initialize.";
-        return;
-    }
+    agent->setup(this);
 
-    auto titleLabel = new QLabel();
-    titleLabel->setAlignment(Qt::AlignCenter);
-
-    auto menuBar = []() {
+    // 2. Construct your title bar
+    auto menuBar = [this, agent]() {
         auto menuBar = new QMenuBar();
-        auto file = new QMenu("File(&F)");
-        file->addAction(new QAction("New(&N)"));
-        file->addAction(new QAction("Open(&O)"));
 
-        auto edit = new QMenu("Edit(&E)");
-        edit->addAction(new QAction("Undo(&U)"));
-        edit->addAction(new QAction("Redo(&R)"));
+        // Virtual menu
+        auto file = new QMenu(tr("File(&F)"), menuBar);
+        file->addAction(new QAction(tr("New(&N)"), menuBar));
+        file->addAction(new QAction(tr("Open(&O)"), menuBar));
+        file->addSeparator();
+
+        auto edit = new QMenu(tr("Edit(&E)"), menuBar);
+        edit->addAction(new QAction(tr("Undo(&U)"), menuBar));
+        edit->addAction(new QAction(tr("Redo(&R)"), menuBar));
+
+        // Theme action
+        auto darkAction = new QAction(tr("Dark Theme"), menuBar);
+        darkAction->setCheckable(true);
+        connect(darkAction, &QAction::triggered, this, [this](bool checked) {
+            loadStyleSheet(checked ? Dark : Light); //
+        });
+        connect(this, &MainWindow::themeChanged, darkAction, [this, darkAction]() {
+            darkAction->setChecked(currentTheme == Dark); //
+        });
+
+        // Agent action
+        auto framelessOnAction = new QAction(tr("Enable Frameless"), menuBar);
+        framelessOnAction->setCheckable(true);
+        framelessOnAction->setChecked(true);
+        connect(framelessOnAction, &QAction::triggered, agent, &QWK::WindowAgentBase::setEnabled);
+        connect(agent, &QWK::WindowAgentBase::enabledChanged, framelessOnAction,
+                &QAction::setChecked);
+
+        // Real menu
+        auto settings = new QMenu(tr("Settings(&S)"), menuBar);
+        settings->addAction(darkAction);
+        settings->addSeparator();
+        settings->addAction(framelessOnAction);
 
         menuBar->addMenu(file);
         menuBar->addMenu(edit);
+        menuBar->addMenu(settings);
         return menuBar;
     }();
+    menuBar->setObjectName(QStringLiteral("win-menu-bar"));
 
-    auto iconButton = new QPushButton("I");
-    auto minButton = new QPushButton("鈥�");
-    auto maxButton = new QPushButton("馃棖");
+    auto titleLabel = new QLabel();
+    titleLabel->setAlignment(Qt::AlignCenter);
+    titleLabel->setObjectName(QStringLiteral("win-title-label"));
+
+#ifndef Q_OS_MAC
+    auto iconButton = new QWK::WindowButton();
+    iconButton->setObjectName(QStringLiteral("icon-button"));
+    iconButton->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
+
+    auto minButton = new QWK::WindowButton();
+    minButton->setObjectName(QStringLiteral("min-button"));
+    minButton->setProperty("system-button", true);
+    minButton->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
+
+    auto maxButton = new QWK::WindowButton();
     maxButton->setCheckable(true);
-    auto closeButton = new QPushButton("鉁�");
+    maxButton->setObjectName(QStringLiteral("max-button"));
+    maxButton->setProperty("system-button", true);
+    maxButton->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
+
+    auto closeButton = new QWK::WindowButton();
+    closeButton->setObjectName(QStringLiteral("close-button"));
+    closeButton->setProperty("system-button", true);
+    closeButton->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
+#endif
 
     auto windowBar = new QWK::WindowBar();
+#ifndef Q_OS_MAC
     windowBar->setIconButton(iconButton);
     windowBar->setMinButton(minButton);
     windowBar->setMaxButton(maxButton);
     windowBar->setCloseButton(closeButton);
+#endif
     windowBar->setMenuBar(menuBar);
     windowBar->setTitleLabel(titleLabel);
     windowBar->setHostWidget(this);
 
     agent->setTitleBar(windowBar);
-    agent->setSystemButton(QWK::CoreWindowAgent::WindowIcon, iconButton);
-    agent->setSystemButton(QWK::CoreWindowAgent::Minimize, minButton);
-    agent->setSystemButton(QWK::CoreWindowAgent::Maximize, maxButton);
-    agent->setSystemButton(QWK::CoreWindowAgent::Close, closeButton);
+#ifndef Q_OS_MAC
+    agent->setSystemButton(QWK::WindowAgentBase::WindowIcon, iconButton);
+    agent->setSystemButton(QWK::WindowAgentBase::Minimize, minButton);
+    agent->setSystemButton(QWK::WindowAgentBase::Maximize, maxButton);
+    agent->setSystemButton(QWK::WindowAgentBase::Close, closeButton);
+#endif
     agent->setHitTestVisible(menuBar, true);
 
+    setMenuWidget(windowBar);
+
+    // 3. Adds simulated mouse events to the title bar buttons
+#ifdef Q_OS_WINDOWS
+    // Emulate Window system menu button behaviors
+    connect(iconButton, &QAbstractButton::clicked, agent, [iconButton, agent] {
+        iconButton->setProperty("double-click-close", false);
+
+        // Pick a suitable time threshold
+        QTimer::singleShot(75, agent, [iconButton, agent]() {
+            if (iconButton->property("double-click-close").toBool())
+                return;
+            agent->showSystemMenu(iconButton->mapToGlobal(QPoint{0, iconButton->height()}));
+        });
+    });
+    connect(iconButton, &QWK::WindowButton::doubleClicked, this, [iconButton, this]() {
+        iconButton->setProperty("double-click-close", true);
+        close();
+    });
+#endif
+
+#ifndef Q_OS_MAC
     connect(windowBar, &QWK::WindowBar::minimizeRequested, this, &QWidget::showMinimized);
-    connect(windowBar, &QWK::WindowBar::maximizeRequested, this, [this](bool max) {
+    connect(windowBar, &QWK::WindowBar::maximizeRequested, this, [this, maxButton](bool max) {
         if (max) {
             showMaximized();
         } else {
             showNormal();
         }
+
+        // It's a Qt issue that if a QAbstractButton::clicked triggers a window's maximization,
+        // the button remains to be hovered until the mouse move. As a result, we need to
+        // manually send leave events to the button.
+        emulateLeaveEvent(maxButton);
     });
     connect(windowBar, &QWK::WindowBar::closeRequested, this, &QWidget::close);
-
-    auto clockWidget = new ClockWidget();
-    clockWidget->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
-
-    setMenuWidget(windowBar);
-    setCentralWidget(clockWidget);
-    setWindowTitle("Example MainWindow");
-    resize(1024, 768);
+#endif
 }
+
+void MainWindow::loadStyleSheet(Theme theme) {
+    if (!styleSheet().isEmpty() && theme == currentTheme)
+        return;
+    currentTheme = theme;
+    if (QFile qss(theme == Dark ? QStringLiteral(":/dark-style.qss")
+                                : QStringLiteral(":/light-style.qss"));
+        qss.open(QIODevice::ReadOnly | QIODevice::Text)) {
+        setStyleSheet(QString::fromUtf8(qss.readAll()));
+        Q_EMIT themeChanged();
+    }
+}
\ No newline at end of file

--
Gitblit v1.9.1