| 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#endifconstexpr 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 factorint 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);}// staticLRESULT 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);}LRESULTWin32Window::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));  }}
 |