From 430d245bb94c634047e7b65232eb3e50e2b6e49a Mon Sep 17 00:00:00 2001
From: Zhao Yuhang <2546789017@qq.com>
Date: 摹曛, 21 12月 2023 22:00:37 +0800
Subject: [PATCH] add more attrib

---
 src/stylesupport/styleagent_p.h     |    1 
 examples/mainwindow/CMakeLists.txt  |    2 
 src/stylesupport/styleagent_win.cpp |   56 ++++++++++++++++--
 examples/mainwindow/mainwindow.h    |    8 ++
 src/core/shared/qwkwindowsextra_p.h |    5 +
 examples/mainwindow/dark-style.qss  |    6 +
 examples/mainwindow/light-style.qss |    6 +
 examples/mainwindow/mainwindow.cpp  |   83 +++++++++++++++++++++++----
 8 files changed, 142 insertions(+), 25 deletions(-)

diff --git a/examples/mainwindow/CMakeLists.txt b/examples/mainwindow/CMakeLists.txt
index c29f2b8..ea3c66b 100644
--- a/examples/mainwindow/CMakeLists.txt
+++ b/examples/mainwindow/CMakeLists.txt
@@ -5,7 +5,7 @@
 qwk_add_example(${PROJECT_NAME}
     SOURCES ${_src} mainwindow.qrc ../shared/resources/shared.qrc
     QT_LINKS Core Gui Widgets
-    LINKS QWKWidgets WidgetFrame
+    LINKS QWKWidgets QWKStyleSupport WidgetFrame
 )
 
 set_target_properties(${PROJECT_NAME} PROPERTIES
diff --git a/examples/mainwindow/dark-style.qss b/examples/mainwindow/dark-style.qss
index 095b900..ef77219 100644
--- a/examples/mainwindow/dark-style.qss
+++ b/examples/mainwindow/dark-style.qss
@@ -142,7 +142,11 @@
 
 /* Window */
 
-MainWindow {
+MainWindow[custom-style=true] {
+    background-color: transparent;
+}
+
+MainWindow[custom-style=false] {
     background-color: #1E1E1E;
 }
 
diff --git a/examples/mainwindow/light-style.qss b/examples/mainwindow/light-style.qss
index eef4066..6c15e62 100644
--- a/examples/mainwindow/light-style.qss
+++ b/examples/mainwindow/light-style.qss
@@ -140,7 +140,11 @@
 
 /* Window */
 
-MainWindow {
+MainWindow[custom-style=true] {
+    background-color: transparent;
+}
+
+MainWindow[custom-style=false] {
     background-color: #F3F3F3;
 }
 
diff --git a/examples/mainwindow/mainwindow.cpp b/examples/mainwindow/mainwindow.cpp
index 52b2880..173be17 100644
--- a/examples/mainwindow/mainwindow.cpp
+++ b/examples/mainwindow/mainwindow.cpp
@@ -4,11 +4,14 @@
 #include <QtCore/QFile>
 #include <QtCore/QTime>
 #include <QtCore/QTimer>
+#include <QtGui/QPainter>
+#include <QtGui/QActionGroup>
 #include <QtWidgets/QApplication>
 #include <QtWidgets/QStyle>
 #include <QtWidgets/QPushButton>
 
 #include <QWKWidgets/widgetwindowagent.h>
+#include <QWKStyleSupport/styleagent.h>
 
 #include <widgetframe/windowbar.h>
 #include <widgetframe/windowbutton.h>
@@ -30,6 +33,8 @@
 
 MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent) {
     installWindowAgent();
+
+    styleAgent = new QWK::StyleAgent(this);
 
     auto clockWidget = new ClockWidget();
     clockWidget->setObjectName(QStringLiteral("clock-widget"));
@@ -104,11 +109,11 @@
 
 void MainWindow::installWindowAgent() {
     // 1. Setup window agent
-    auto agent = new QWK::WidgetWindowAgent(this);
-    agent->setup(this);
+    windowAgent = new QWK::WidgetWindowAgent(this);
+    windowAgent->setup(this);
 
     // 2. Construct your title bar
-    auto menuBar = [this, agent]() {
+    auto menuBar = [this]() {
         auto menuBar = new QMenuBar();
 
         // Virtual menu
@@ -122,7 +127,7 @@
         edit->addAction(new QAction(tr("Redo(&R)"), menuBar));
 
         // Theme action
-        auto darkAction = new QAction(tr("Dark Theme"), menuBar);
+        auto darkAction = new QAction(tr("Enable dark theme"), menuBar);
         darkAction->setCheckable(true);
         connect(darkAction, &QAction::triggered, this, [this](bool checked) {
             loadStyleSheet(checked ? Dark : Light); //
@@ -131,10 +136,60 @@
             darkAction->setChecked(currentTheme == Dark); //
         });
 
+#ifdef Q_OS_WIN
+        auto dwmBlurAction = new QAction(tr("Enable DWM blur"), menuBar);
+        dwmBlurAction->setCheckable(true);
+        connect(dwmBlurAction, &QAction::triggered, this, [this](bool checked){
+            QWindow *w = windowHandle();
+            styleAgent->setWindowAttribute(w, QStringLiteral("dwm-blur"), checked);
+            setProperty("custom-style", checked);
+            style()->polish(this);
+        });
+
+        auto acrylicAction = new QAction(tr("Enable acrylic material"), menuBar);
+        acrylicAction->setCheckable(true);
+        connect(acrylicAction, &QAction::triggered, this, [this](bool checked){
+            QWindow *w = windowHandle();
+            styleAgent->setWindowAttribute(w, QStringLiteral("acrylic-material"), QColor());
+            setProperty("custom-style", checked);
+            style()->polish(this);
+        });
+
+        auto micaAction = new QAction(tr("Enable mica"), menuBar);
+        micaAction->setCheckable(true);
+        connect(micaAction, &QAction::triggered, this, [this](bool checked){
+            QWindow *w = windowHandle();
+            styleAgent->setWindowAttribute(w, QStringLiteral("mica"), checked);
+            setProperty("custom-style", checked);
+            style()->polish(this);
+        });
+
+        auto micaAltAction = new QAction(tr("Enable mica alt"), menuBar);
+        micaAltAction->setCheckable(true);
+        connect(micaAltAction, &QAction::triggered, this, [this](bool checked){
+            QWindow *w = windowHandle();
+            styleAgent->setWindowAttribute(w, QStringLiteral("mica-alt"), checked);
+            setProperty("custom-style", checked);
+            style()->polish(this);
+        });
+
+        auto winStyleGroup = new QActionGroup(menuBar);
+        winStyleGroup->addAction(dwmBlurAction);
+        winStyleGroup->addAction(acrylicAction);
+        winStyleGroup->addAction(micaAction);
+        winStyleGroup->addAction(micaAltAction);
+#endif
+
         // Real menu
         auto settings = new QMenu(tr("Settings(&S)"), menuBar);
         settings->addAction(darkAction);
+#ifdef Q_OS_WIN
         settings->addSeparator();
+        settings->addAction(dwmBlurAction);
+        settings->addAction(acrylicAction);
+        settings->addAction(micaAction);
+        settings->addAction(micaAltAction);
+#endif
 
         menuBar->addMenu(file);
         menuBar->addMenu(edit);
@@ -180,28 +235,28 @@
     windowBar->setTitleLabel(titleLabel);
     windowBar->setHostWidget(this);
 
-    agent->setTitleBar(windowBar);
+    windowAgent->setTitleBar(windowBar);
 #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);
+    windowAgent->setSystemButton(QWK::WindowAgentBase::WindowIcon, iconButton);
+    windowAgent->setSystemButton(QWK::WindowAgentBase::Minimize, minButton);
+    windowAgent->setSystemButton(QWK::WindowAgentBase::Maximize, maxButton);
+    windowAgent->setSystemButton(QWK::WindowAgentBase::Close, closeButton);
 #endif
-    agent->setHitTestVisible(menuBar, true);
+    windowAgent->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] {
+    connect(iconButton, &QAbstractButton::clicked, windowAgent, [this, iconButton] {
         iconButton->setProperty("double-click-close", false);
 
         // Pick a suitable time threshold
-        QTimer::singleShot(75, agent, [iconButton, agent]() {
+        QTimer::singleShot(75, windowAgent, [this, iconButton]() {
             if (iconButton->property("double-click-close").toBool())
                 return;
-            agent->showSystemMenu(iconButton->mapToGlobal(QPoint{0, iconButton->height()}));
+            windowAgent->showSystemMenu(iconButton->mapToGlobal(QPoint{0, iconButton->height()}));
         });
     });
     connect(iconButton, &QWK::WindowButton::doubleClicked, this, [iconButton, this]() {
@@ -238,4 +293,4 @@
         setStyleSheet(QString::fromUtf8(qss.readAll()));
         Q_EMIT themeChanged();
     }
-}
\ No newline at end of file
+}
diff --git a/examples/mainwindow/mainwindow.h b/examples/mainwindow/mainwindow.h
index 5511105..48d73cf 100644
--- a/examples/mainwindow/mainwindow.h
+++ b/examples/mainwindow/mainwindow.h
@@ -3,6 +3,11 @@
 
 #include <QtWidgets/QMainWindow>
 
+namespace QWK {
+    class WidgetWindowAgent;
+    class StyleAgent;
+}
+
 class MainWindow : public QMainWindow {
     Q_OBJECT
 public:
@@ -26,6 +31,9 @@
     void loadStyleSheet(Theme theme);
 
     Theme currentTheme{};
+
+    QWK::WidgetWindowAgent *windowAgent;
+    QWK::StyleAgent *styleAgent;
 };
 
 #endif // MAINWINDOW_H
diff --git a/src/core/shared/qwkwindowsextra_p.h b/src/core/shared/qwkwindowsextra_p.h
index c4b5002..850f0a2 100644
--- a/src/core/shared/qwkwindowsextra_p.h
+++ b/src/core/shared/qwkwindowsextra_p.h
@@ -16,6 +16,9 @@
 
 #include <QWKCore/qwindowkit_windows.h>
 
+#include <QtGui/QStyleHints>
+#include <QtGui/QPalette>
+
 #include <QtCore/private/qsystemlibrary_p.h>
 #include <QtCore/private/qwinregistry_p.h>
 
@@ -146,6 +149,7 @@
             DYNAMIC_API_DECLARE(DwmGetWindowAttribute);
             DYNAMIC_API_DECLARE(DwmSetWindowAttribute);
             DYNAMIC_API_DECLARE(DwmExtendFrameIntoClientArea);
+            DYNAMIC_API_DECLARE(DwmEnableBlurBehindWindow);
             DYNAMIC_API_DECLARE(GetDpiForWindow);
             DYNAMIC_API_DECLARE(GetSystemMetricsForDpi);
             DYNAMIC_API_DECLARE(GetDpiForMonitor);
@@ -177,6 +181,7 @@
                 DYNAMIC_API_RESOLVE(dwmapi, DwmGetWindowAttribute);
                 DYNAMIC_API_RESOLVE(dwmapi, DwmSetWindowAttribute);
                 DYNAMIC_API_RESOLVE(dwmapi, DwmExtendFrameIntoClientArea);
+                DYNAMIC_API_RESOLVE(dwmapi, DwmEnableBlurBehindWindow);
 
                 QSystemLibrary winmm(QStringLiteral("winmm"));
                 DYNAMIC_API_RESOLVE(winmm, timeGetDevCaps);
diff --git a/src/stylesupport/styleagent_p.h b/src/stylesupport/styleagent_p.h
index b59266d..3e36aaf 100644
--- a/src/stylesupport/styleagent_p.h
+++ b/src/stylesupport/styleagent_p.h
@@ -2,6 +2,7 @@
 #define STYLEAGENTPRIVATE_H
 
 #include <QWKStyleSupport/styleagent.h>
+#include <QtCore/QHash>
 
 namespace QWK {
 
diff --git a/src/stylesupport/styleagent_win.cpp b/src/stylesupport/styleagent_win.cpp
index a685586..67b6532 100644
--- a/src/stylesupport/styleagent_win.cpp
+++ b/src/stylesupport/styleagent_win.cpp
@@ -2,6 +2,7 @@
 
 #include <QtCore/QSet>
 #include <QtCore/QVariant>
+#include <QtGui/QColor>
 
 #include <QWKCore/private/qwkwindowsextra_p.h>
 #include <QWKCore/private/nativeeventfilter_p.h>
@@ -88,13 +89,7 @@
         const auto hwnd = reinterpret_cast<HWND>(window->winId());
         const DynamicApis &apis = DynamicApis::instance();
 
-        if (key == QStringLiteral("frame-shadow")) {
-            if (attribute.toBool()) {
-                // TODO: set off
-            } else {
-                // TODO: set on
-            }
-        } else if (key == QStringLiteral("mica")) {
+        if (key == QStringLiteral("mica")) {
             if (!isWin11OrGreater()) {
                 return false;
             }
@@ -151,7 +146,10 @@
             }
             return true;
         } else if (key == QStringLiteral("acrylic-material")) {
-            if (attribute.type() == QVariant::Color) {
+            if (!isWin10OrGreater()) {
+                return false;
+            }
+            if (attribute.userType() == QMetaType::QColor) {
                 // We need to extend the window frame into the whole client area to be able
                 // to see the blurred window background.
                 static constexpr const MARGINS margins = {-1, -1, -1, -1};
@@ -195,6 +193,48 @@
                 apis.pDwmExtendFrameIntoClientArea(hwnd, &margins);
             }
             return true;
+        } else if (key == QStringLiteral("dwm-blur")) {
+            if (attribute.toBool()) {
+                // We need to extend the window frame into the whole client area to be able
+                // to see the blurred window background.
+                static constexpr const MARGINS margins = {-1, -1, -1, -1};
+                apis.pDwmExtendFrameIntoClientArea(hwnd, &margins);
+                if (isWin8OrGreater()) {
+                    ACCENT_POLICY policy{};
+                    policy.dwAccentState = ACCENT_ENABLE_BLURBEHIND;
+                    policy.dwAccentFlags = ACCENT_NONE;
+                    WINDOWCOMPOSITIONATTRIBDATA wcad{};
+                    wcad.Attrib = WCA_ACCENT_POLICY;
+                    wcad.pvData = &policy;
+                    wcad.cbData = sizeof(policy);
+                    apis.pSetWindowCompositionAttribute(hwnd, &wcad);
+                } else {
+                    DWM_BLURBEHIND bb{};
+                    bb.fEnable = TRUE;
+                    bb.fTransitionOnMaximized = TRUE;
+                    bb.dwFlags = DWM_BB_ENABLE | DWM_BB_TRANSITIONONMAXIMIZED;
+                    apis.pDwmEnableBlurBehindWindow(hwnd, &bb);
+                }
+            } else {
+                if (isWin8OrGreater()) {
+                    ACCENT_POLICY policy{};
+                    policy.dwAccentState = ACCENT_DISABLED;
+                    policy.dwAccentFlags = ACCENT_NONE;
+                    WINDOWCOMPOSITIONATTRIBDATA wcad{};
+                    wcad.Attrib = WCA_ACCENT_POLICY;
+                    wcad.pvData = &policy;
+                    wcad.cbData = sizeof(policy);
+                    apis.pSetWindowCompositionAttribute(hwnd, &wcad);
+                } else {
+                    DWM_BLURBEHIND bb{};
+                    bb.fEnable = FALSE;
+                    bb.dwFlags = DWM_BB_ENABLE;
+                    apis.pDwmEnableBlurBehindWindow(hwnd, &bb);
+                }
+                static constexpr const MARGINS margins = {0, 0, 0, 0};
+                apis.pDwmExtendFrameIntoClientArea(hwnd, &margins);
+            }
+            return true;
         }
         return false;
     }

--
Gitblit v1.9.1