| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 | 
							- #include "win32_window.h"
 
- #include <dwmapi.h>
 
- #include <flutter_windows.h>
 
- #include "resource.h"
 
- namespace {
 
- /// Window attribute that enables dark mode window decorations.
 
- ///
 
- /// Redefined in case the developer's machine has a Windows SDK older than
 
- /// version 10.0.22000.0.
 
- /// See: https://docs.microsoft.com/windows/win32/api/dwmapi/ne-dwmapi-dwmwindowattribute
 
- #ifndef DWMWA_USE_IMMERSIVE_DARK_MODE
 
- #define DWMWA_USE_IMMERSIVE_DARK_MODE 20
 
- #endif
 
- constexpr const wchar_t kWindowClassName[] = L"FLUTTER_RUNNER_WIN32_WINDOW";
 
- /// Registry key for app theme preference.
 
- ///
 
- /// A value of 0 indicates apps should use dark mode. A non-zero or missing
 
- /// value indicates apps should use light mode.
 
- constexpr const wchar_t kGetPreferredBrightnessRegKey[] =
 
-   L"Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\Personalize";
 
- constexpr const wchar_t kGetPreferredBrightnessRegValue[] = L"AppsUseLightTheme";
 
- // The number of Win32Window objects that currently exist.
 
- static int g_active_window_count = 0;
 
- using EnableNonClientDpiScaling = BOOL __stdcall(HWND hwnd);
 
- // Scale helper to convert logical scaler values to physical using passed in
 
- // scale factor
 
- int Scale(int source, double scale_factor) {
 
-   return static_cast<int>(source * scale_factor);
 
- }
 
- // Dynamically loads the |EnableNonClientDpiScaling| from the User32 module.
 
- // This API is only needed for PerMonitor V1 awareness mode.
 
- void EnableFullDpiSupportIfAvailable(HWND hwnd) {
 
-   HMODULE user32_module = LoadLibraryA("User32.dll");
 
-   if (!user32_module) {
 
-     return;
 
-   }
 
-   auto enable_non_client_dpi_scaling =
 
-       reinterpret_cast<EnableNonClientDpiScaling*>(
 
-           GetProcAddress(user32_module, "EnableNonClientDpiScaling"));
 
-   if (enable_non_client_dpi_scaling != nullptr) {
 
-     enable_non_client_dpi_scaling(hwnd);
 
-   }
 
-   FreeLibrary(user32_module);
 
- }
 
- }  // namespace
 
- // Manages the Win32Window's window class registration.
 
- class WindowClassRegistrar {
 
-  public:
 
-   ~WindowClassRegistrar() = default;
 
-   // Returns the singleton registrar instance.
 
-   static WindowClassRegistrar* GetInstance() {
 
-     if (!instance_) {
 
-       instance_ = new WindowClassRegistrar();
 
-     }
 
-     return instance_;
 
-   }
 
-   // Returns the name of the window class, registering the class if it hasn't
 
-   // previously been registered.
 
-   const wchar_t* GetWindowClass();
 
-   // Unregisters the window class. Should only be called if there are no
 
-   // instances of the window.
 
-   void UnregisterWindowClass();
 
-  private:
 
-   WindowClassRegistrar() = default;
 
-   static WindowClassRegistrar* instance_;
 
-   bool class_registered_ = false;
 
- };
 
- WindowClassRegistrar* WindowClassRegistrar::instance_ = nullptr;
 
- const wchar_t* WindowClassRegistrar::GetWindowClass() {
 
-   if (!class_registered_) {
 
-     WNDCLASS window_class{};
 
-     window_class.hCursor = LoadCursor(nullptr, IDC_ARROW);
 
-     window_class.lpszClassName = kWindowClassName;
 
-     window_class.style = CS_HREDRAW | CS_VREDRAW;
 
-     window_class.cbClsExtra = 0;
 
-     window_class.cbWndExtra = 0;
 
-     window_class.hInstance = GetModuleHandle(nullptr);
 
-     window_class.hIcon =
 
-         LoadIcon(window_class.hInstance, MAKEINTRESOURCE(IDI_APP_ICON));
 
-     window_class.hbrBackground = 0;
 
-     window_class.lpszMenuName = nullptr;
 
-     window_class.lpfnWndProc = Win32Window::WndProc;
 
-     RegisterClass(&window_class);
 
-     class_registered_ = true;
 
-   }
 
-   return kWindowClassName;
 
- }
 
- void WindowClassRegistrar::UnregisterWindowClass() {
 
-   UnregisterClass(kWindowClassName, nullptr);
 
-   class_registered_ = false;
 
- }
 
- Win32Window::Win32Window() {
 
-   ++g_active_window_count;
 
- }
 
- Win32Window::~Win32Window() {
 
-   --g_active_window_count;
 
-   Destroy();
 
- }
 
- bool Win32Window::Create(const std::wstring& title,
 
-                          const Point& origin,
 
-                          const Size& size) {
 
-   Destroy();
 
-   const wchar_t* window_class =
 
-       WindowClassRegistrar::GetInstance()->GetWindowClass();
 
-   const POINT target_point = {static_cast<LONG>(origin.x),
 
-                               static_cast<LONG>(origin.y)};
 
-   HMONITOR monitor = MonitorFromPoint(target_point, MONITOR_DEFAULTTONEAREST);
 
-   UINT dpi = FlutterDesktopGetDpiForMonitor(monitor);
 
-   double scale_factor = dpi / 96.0;
 
-   HWND window = CreateWindow(
 
-       window_class, title.c_str(), WS_OVERLAPPEDWINDOW,
 
-       Scale(origin.x, scale_factor), Scale(origin.y, scale_factor),
 
-       Scale(size.width, scale_factor), Scale(size.height, scale_factor),
 
-       nullptr, nullptr, GetModuleHandle(nullptr), this);
 
-   if (!window) {
 
-     return false;
 
-   }
 
-   UpdateTheme(window);
 
-   return OnCreate();
 
- }
 
- bool Win32Window::Show() {
 
-   return ShowWindow(window_handle_, SW_SHOWNORMAL);
 
- }
 
- // static
 
- LRESULT CALLBACK Win32Window::WndProc(HWND const window,
 
-                                       UINT const message,
 
-                                       WPARAM const wparam,
 
-                                       LPARAM const lparam) noexcept {
 
-   if (message == WM_NCCREATE) {
 
-     auto window_struct = reinterpret_cast<CREATESTRUCT*>(lparam);
 
-     SetWindowLongPtr(window, GWLP_USERDATA,
 
-                      reinterpret_cast<LONG_PTR>(window_struct->lpCreateParams));
 
-     auto that = static_cast<Win32Window*>(window_struct->lpCreateParams);
 
-     EnableFullDpiSupportIfAvailable(window);
 
-     that->window_handle_ = window;
 
-   } else if (Win32Window* that = GetThisFromHandle(window)) {
 
-     return that->MessageHandler(window, message, wparam, lparam);
 
-   }
 
-   return DefWindowProc(window, message, wparam, lparam);
 
- }
 
- LRESULT
 
- Win32Window::MessageHandler(HWND hwnd,
 
-                             UINT const message,
 
-                             WPARAM const wparam,
 
-                             LPARAM const lparam) noexcept {
 
-   switch (message) {
 
-     case WM_DESTROY:
 
-       window_handle_ = nullptr;
 
-       Destroy();
 
-       if (quit_on_close_) {
 
-         PostQuitMessage(0);
 
-       }
 
-       return 0;
 
-     case WM_DPICHANGED: {
 
-       auto newRectSize = reinterpret_cast<RECT*>(lparam);
 
-       LONG newWidth = newRectSize->right - newRectSize->left;
 
-       LONG newHeight = newRectSize->bottom - newRectSize->top;
 
-       SetWindowPos(hwnd, nullptr, newRectSize->left, newRectSize->top, newWidth,
 
-                    newHeight, SWP_NOZORDER | SWP_NOACTIVATE);
 
-       return 0;
 
-     }
 
-     case WM_SIZE: {
 
-       RECT rect = GetClientArea();
 
-       if (child_content_ != nullptr) {
 
-         // Size and position the child window.
 
-         MoveWindow(child_content_, rect.left, rect.top, rect.right - rect.left,
 
-                    rect.bottom - rect.top, TRUE);
 
-       }
 
-       return 0;
 
-     }
 
-     case WM_ACTIVATE:
 
-       if (child_content_ != nullptr) {
 
-         SetFocus(child_content_);
 
-       }
 
-       return 0;
 
-     case WM_DWMCOLORIZATIONCOLORCHANGED:
 
-       UpdateTheme(hwnd);
 
-       return 0;
 
-   }
 
-   return DefWindowProc(window_handle_, message, wparam, lparam);
 
- }
 
- void Win32Window::Destroy() {
 
-   OnDestroy();
 
-   if (window_handle_) {
 
-     DestroyWindow(window_handle_);
 
-     window_handle_ = nullptr;
 
-   }
 
-   if (g_active_window_count == 0) {
 
-     WindowClassRegistrar::GetInstance()->UnregisterWindowClass();
 
-   }
 
- }
 
- Win32Window* Win32Window::GetThisFromHandle(HWND const window) noexcept {
 
-   return reinterpret_cast<Win32Window*>(
 
-       GetWindowLongPtr(window, GWLP_USERDATA));
 
- }
 
- void Win32Window::SetChildContent(HWND content) {
 
-   child_content_ = content;
 
-   SetParent(content, window_handle_);
 
-   RECT frame = GetClientArea();
 
-   MoveWindow(content, frame.left, frame.top, frame.right - frame.left,
 
-              frame.bottom - frame.top, true);
 
-   SetFocus(child_content_);
 
- }
 
- RECT Win32Window::GetClientArea() {
 
-   RECT frame;
 
-   GetClientRect(window_handle_, &frame);
 
-   return frame;
 
- }
 
- HWND Win32Window::GetHandle() {
 
-   return window_handle_;
 
- }
 
- void Win32Window::SetQuitOnClose(bool quit_on_close) {
 
-   quit_on_close_ = quit_on_close;
 
- }
 
- bool Win32Window::OnCreate() {
 
-   // No-op; provided for subclasses.
 
-   return true;
 
- }
 
- void Win32Window::OnDestroy() {
 
-   // No-op; provided for subclasses.
 
- }
 
- void Win32Window::UpdateTheme(HWND const window) {
 
-   DWORD light_mode;
 
-   DWORD light_mode_size = sizeof(light_mode);
 
-   LSTATUS result = RegGetValue(HKEY_CURRENT_USER, kGetPreferredBrightnessRegKey,
 
-                                kGetPreferredBrightnessRegValue,
 
-                                RRF_RT_REG_DWORD, nullptr, &light_mode,
 
-                                &light_mode_size);
 
-   if (result == ERROR_SUCCESS) {
 
-     BOOL enable_dark_mode = light_mode == 0;
 
-     DwmSetWindowAttribute(window, DWMWA_USE_IMMERSIVE_DARK_MODE,
 
-                           &enable_dark_mode, sizeof(enable_dark_mode));
 
-   }
 
- }
 
 
  |