| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | // Copyright (c) 2014 GitHub, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-10 19:00:37 -06:00
										 |  |  | #include <iomanip>
 | 
					
						
							|  |  |  | #include <string_view>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 23:39:56 +02:00
										 |  |  | #include <dwmapi.h>
 | 
					
						
							| 
									
										
										
										
											2020-06-25 18:47:50 +02:00
										 |  |  | #include <windows.devices.enumeration.h>
 | 
					
						
							|  |  |  | #include <wrl/client.h>
 | 
					
						
							| 
									
										
										
										
											2018-01-09 11:07:29 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/api/electron_api_system_preferences.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-14 16:00:46 -05:00
										 |  |  | #include "base/containers/fixed_flat_map.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-25 18:47:50 +02:00
										 |  |  | #include "base/win/core_winrt_util.h"
 | 
					
						
							| 
									
										
										
										
											2020-07-13 18:13:34 -07:00
										 |  |  | #include "base/win/windows_types.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | #include "base/win/wrapped_window_proc.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/common/color_util.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | #include "ui/base/win/shell.h"
 | 
					
						
							| 
									
										
										
										
											2016-10-10 14:31:06 -07:00
										 |  |  | #include "ui/gfx/color_utils.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | #include "ui/gfx/win/hwnd_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-20 02:42:41 +10:00
										 |  |  | const wchar_t kSystemPreferencesWindowClass[] = | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |     L"Electron_SystemPreferencesHostWindow"; | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 18:47:50 +02:00
										 |  |  | using ABI::Windows::Devices::Enumeration::DeviceAccessStatus; | 
					
						
							|  |  |  | using ABI::Windows::Devices::Enumeration::DeviceClass; | 
					
						
							|  |  |  | using ABI::Windows::Devices::Enumeration::IDeviceAccessInformation; | 
					
						
							|  |  |  | using ABI::Windows::Devices::Enumeration::IDeviceAccessInformationStatics; | 
					
						
							|  |  |  | using Microsoft::WRL::ComPtr; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DeviceAccessStatus GetDeviceAccessStatus(DeviceClass device_class) { | 
					
						
							|  |  |  |   ComPtr<IDeviceAccessInformationStatics> dev_access_info_statics; | 
					
						
							|  |  |  |   HRESULT hr = base::win::GetActivationFactory< | 
					
						
							|  |  |  |       IDeviceAccessInformationStatics, | 
					
						
							|  |  |  |       RuntimeClass_Windows_Devices_Enumeration_DeviceAccessInformation>( | 
					
						
							|  |  |  |       &dev_access_info_statics); | 
					
						
							|  |  |  |   if (FAILED(hr)) { | 
					
						
							|  |  |  |     VLOG(1) << "IDeviceAccessInformationStatics failed: " << hr; | 
					
						
							|  |  |  |     return DeviceAccessStatus::DeviceAccessStatus_Allowed; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ComPtr<IDeviceAccessInformation> dev_access_info; | 
					
						
							|  |  |  |   hr = dev_access_info_statics->CreateFromDeviceClass(device_class, | 
					
						
							|  |  |  |                                                       &dev_access_info); | 
					
						
							|  |  |  |   if (FAILED(hr)) { | 
					
						
							|  |  |  |     VLOG(1) << "IDeviceAccessInformation failed: " << hr; | 
					
						
							|  |  |  |     return DeviceAccessStatus::DeviceAccessStatus_Allowed; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto status = DeviceAccessStatus::DeviceAccessStatus_Unspecified; | 
					
						
							|  |  |  |   dev_access_info->get_CurrentStatus(&status); | 
					
						
							|  |  |  |   return status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string ConvertDeviceAccessStatus(DeviceAccessStatus value) { | 
					
						
							|  |  |  |   switch (value) { | 
					
						
							|  |  |  |     case DeviceAccessStatus::DeviceAccessStatus_Unspecified: | 
					
						
							|  |  |  |       return "not-determined"; | 
					
						
							|  |  |  |     case DeviceAccessStatus::DeviceAccessStatus_Allowed: | 
					
						
							|  |  |  |       return "granted"; | 
					
						
							|  |  |  |     case DeviceAccessStatus::DeviceAccessStatus_DeniedBySystem: | 
					
						
							|  |  |  |       return "restricted"; | 
					
						
							|  |  |  |     case DeviceAccessStatus::DeviceAccessStatus_DeniedByUser: | 
					
						
							|  |  |  |       return "denied"; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return "unknown"; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace api { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool SystemPreferences::IsAeroGlassEnabled() { | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string hexColorDWORDToRGBA(DWORD color) { | 
					
						
							| 
									
										
										
										
											2018-01-09 11:07:29 -08:00
										 |  |  |   DWORD rgba = color << 8 | color >> 24; | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |   std::ostringstream stream; | 
					
						
							| 
									
										
										
										
											2018-01-09 11:07:29 -08:00
										 |  |  |   stream << std::hex << std::setw(8) << std::setfill('0') << rgba; | 
					
						
							|  |  |  |   return stream.str(); | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::string SystemPreferences::GetAccentColor() { | 
					
						
							|  |  |  |   DWORD color = 0; | 
					
						
							|  |  |  |   BOOL opaque = FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-06 23:39:56 +02:00
										 |  |  |   if (FAILED(DwmGetColorizationColor(&color, &opaque))) { | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |     return ""; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return hexColorDWORDToRGBA(color); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-21 14:31:03 -07:00
										 |  |  | std::string SystemPreferences::GetColor(gin_helper::ErrorThrower thrower, | 
					
						
							|  |  |  |                                         const std::string& color) { | 
					
						
							| 
									
										
										
										
											2024-01-10 19:00:37 -06:00
										 |  |  |   static constexpr auto Lookup = base::MakeFixedFlatMap<std::string_view, int>({ | 
					
						
							|  |  |  |       {"3d-dark-shadow", COLOR_3DDKSHADOW}, | 
					
						
							|  |  |  |       {"3d-face", COLOR_3DFACE}, | 
					
						
							|  |  |  |       {"3d-highlight", COLOR_3DHIGHLIGHT}, | 
					
						
							|  |  |  |       {"3d-light", COLOR_3DLIGHT}, | 
					
						
							|  |  |  |       {"3d-shadow", COLOR_3DSHADOW}, | 
					
						
							|  |  |  |       {"active-border", COLOR_ACTIVEBORDER}, | 
					
						
							|  |  |  |       {"active-caption", COLOR_ACTIVECAPTION}, | 
					
						
							|  |  |  |       {"active-caption-gradient", COLOR_GRADIENTACTIVECAPTION}, | 
					
						
							|  |  |  |       {"app-workspace", COLOR_APPWORKSPACE}, | 
					
						
							|  |  |  |       {"button-text", COLOR_BTNTEXT}, | 
					
						
							|  |  |  |       {"caption-text", COLOR_CAPTIONTEXT}, | 
					
						
							|  |  |  |       {"desktop", COLOR_DESKTOP}, | 
					
						
							|  |  |  |       {"disabled-text", COLOR_GRAYTEXT}, | 
					
						
							|  |  |  |       {"highlight", COLOR_HIGHLIGHT}, | 
					
						
							|  |  |  |       {"highlight-text", COLOR_HIGHLIGHTTEXT}, | 
					
						
							|  |  |  |       {"hotlight", COLOR_HOTLIGHT}, | 
					
						
							|  |  |  |       {"inactive-border", COLOR_INACTIVEBORDER}, | 
					
						
							|  |  |  |       {"inactive-caption", COLOR_INACTIVECAPTION}, | 
					
						
							|  |  |  |       {"inactive-caption-gradient", COLOR_GRADIENTINACTIVECAPTION}, | 
					
						
							|  |  |  |       {"inactive-caption-text", COLOR_INACTIVECAPTIONTEXT}, | 
					
						
							|  |  |  |       {"info-background", COLOR_INFOBK}, | 
					
						
							|  |  |  |       {"info-text", COLOR_INFOTEXT}, | 
					
						
							|  |  |  |       {"menu", COLOR_MENU}, | 
					
						
							|  |  |  |       {"menu-highlight", COLOR_MENUHILIGHT}, | 
					
						
							|  |  |  |       {"menu-text", COLOR_MENUTEXT}, | 
					
						
							|  |  |  |       {"menubar", COLOR_MENUBAR}, | 
					
						
							|  |  |  |       {"scrollbar", COLOR_SCROLLBAR}, | 
					
						
							|  |  |  |       {"window", COLOR_WINDOW}, | 
					
						
							|  |  |  |       {"window-frame", COLOR_WINDOWFRAME}, | 
					
						
							|  |  |  |       {"window-text", COLOR_WINDOWTEXT}, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2023-06-14 16:00:46 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (const auto* iter = Lookup.find(color); iter != Lookup.end()) | 
					
						
							| 
									
										
										
										
											2023-09-28 23:56:16 +01:00
										 |  |  |     return ToRGBAHex(color_utils::GetSysSkColor(iter->second)); | 
					
						
							| 
									
										
										
										
											2023-06-14 16:00:46 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   thrower.ThrowError("Unknown color: " + color); | 
					
						
							|  |  |  |   return ""; | 
					
						
							| 
									
										
										
										
											2016-10-10 14:31:06 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-25 18:47:50 +02:00
										 |  |  | std::string SystemPreferences::GetMediaAccessStatus( | 
					
						
							| 
									
										
										
										
											2020-07-28 11:03:30 -07:00
										 |  |  |     gin_helper::ErrorThrower thrower, | 
					
						
							|  |  |  |     const std::string& media_type) { | 
					
						
							| 
									
										
										
										
											2020-06-25 18:47:50 +02:00
										 |  |  |   if (media_type == "camera") { | 
					
						
							|  |  |  |     return ConvertDeviceAccessStatus( | 
					
						
							|  |  |  |         GetDeviceAccessStatus(DeviceClass::DeviceClass_VideoCapture)); | 
					
						
							|  |  |  |   } else if (media_type == "microphone") { | 
					
						
							|  |  |  |     return ConvertDeviceAccessStatus( | 
					
						
							|  |  |  |         GetDeviceAccessStatus(DeviceClass::DeviceClass_AudioCapture)); | 
					
						
							|  |  |  |   } else if (media_type == "screen") { | 
					
						
							|  |  |  |     return ConvertDeviceAccessStatus( | 
					
						
							|  |  |  |         DeviceAccessStatus::DeviceAccessStatus_Allowed); | 
					
						
							|  |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2020-07-28 11:03:30 -07:00
										 |  |  |     thrower.ThrowError("Invalid media type"); | 
					
						
							| 
									
										
										
										
											2020-06-25 18:47:50 +02:00
										 |  |  |     return std::string(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | void SystemPreferences::InitializeWindow() { | 
					
						
							| 
									
										
										
										
											2024-06-20 16:12:16 -04:00
										 |  |  |   if (electron::IsUtilityProcess()) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2017-02-02 09:34:20 -08:00
										 |  |  |   // Wait until app is ready before creating sys color listener
 | 
					
						
							|  |  |  |   // Creating this listener before the app is ready causes global shortcuts
 | 
					
						
							|  |  |  |   // to not fire
 | 
					
						
							|  |  |  |   if (Browser::Get()->is_ready()) | 
					
						
							| 
									
										
										
										
											2021-06-07 19:00:05 -07:00
										 |  |  |     color_change_listener_ = | 
					
						
							|  |  |  |         std::make_unique<gfx::ScopedSysColorChangeListener>(this); | 
					
						
							| 
									
										
										
										
											2017-02-02 09:34:20 -08:00
										 |  |  |   else | 
					
						
							|  |  |  |     Browser::Get()->AddObserver(this); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |   WNDCLASSEX window_class; | 
					
						
							|  |  |  |   base::win::InitializeWindowClass( | 
					
						
							| 
									
										
										
										
											2016-09-20 02:42:41 +10:00
										 |  |  |       kSystemPreferencesWindowClass, | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |       &base::win::WrappedWindowProc<SystemPreferences::WndProcStatic>, 0, 0, 0, | 
					
						
							| 
									
										
										
										
											2023-10-03 21:26:35 +02:00
										 |  |  |       nullptr, nullptr, nullptr, nullptr, nullptr, &window_class); | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |   instance_ = window_class.hInstance; | 
					
						
							|  |  |  |   atom_ = RegisterClassEx(&window_class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Create an offscreen window for receiving broadcast messages for the system
 | 
					
						
							|  |  |  |   // colorization color.  Create a hidden WS_POPUP window instead of an
 | 
					
						
							|  |  |  |   // HWND_MESSAGE window, because only top-level windows such as popups can
 | 
					
						
							|  |  |  |   // receive broadcast messages like "WM_DWMCOLORIZATIONCOLORCHANGED".
 | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |   window_ = CreateWindow(MAKEINTATOM(atom_), 0, WS_POPUP, 0, 0, 0, 0, 0, 0, | 
					
						
							|  |  |  |                          instance_, 0); | 
					
						
							| 
									
										
										
										
											2020-07-13 18:13:34 -07:00
										 |  |  |   gfx::CheckWindowCreated(window_, ::GetLastError()); | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |   gfx::SetWindowUserData(window_, this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LRESULT CALLBACK SystemPreferences::WndProcStatic(HWND hwnd, | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |                                                   UINT message, | 
					
						
							|  |  |  |                                                   WPARAM wparam, | 
					
						
							|  |  |  |                                                   LPARAM lparam) { | 
					
						
							| 
									
										
										
										
											2020-10-26 11:56:31 -07:00
										 |  |  |   auto* msg_wnd = reinterpret_cast<SystemPreferences*>( | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |       GetWindowLongPtr(hwnd, GWLP_USERDATA)); | 
					
						
							|  |  |  |   if (msg_wnd) | 
					
						
							|  |  |  |     return msg_wnd->WndProc(hwnd, message, wparam, lparam); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return ::DefWindowProc(hwnd, message, wparam, lparam); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LRESULT CALLBACK SystemPreferences::WndProc(HWND hwnd, | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |                                             UINT message, | 
					
						
							|  |  |  |                                             WPARAM wparam, | 
					
						
							|  |  |  |                                             LPARAM lparam) { | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |   if (message == WM_DWMCOLORIZATIONCOLORCHANGED) { | 
					
						
							| 
									
										
										
										
											2018-04-17 21:55:30 -04:00
										 |  |  |     DWORD new_color = (DWORD)wparam; | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  |     std::string new_color_string = hexColorDWORDToRGBA(new_color); | 
					
						
							|  |  |  |     if (new_color_string != current_color_) { | 
					
						
							|  |  |  |       Emit("accent-color-changed", hexColorDWORDToRGBA(new_color)); | 
					
						
							|  |  |  |       current_color_ = new_color_string; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return ::DefWindowProc(hwnd, message, wparam, lparam); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-10 10:20:51 -07:00
										 |  |  | void SystemPreferences::OnSysColorChange() { | 
					
						
							| 
									
										
										
										
											2016-10-11 09:35:14 -07:00
										 |  |  |   Emit("color-changed"); | 
					
						
							| 
									
										
										
										
											2016-10-07 09:38:19 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-22 23:28:41 -07:00
										 |  |  | void SystemPreferences::OnFinishLaunching(base::Value::Dict launch_info) { | 
					
						
							| 
									
										
										
										
											2021-06-07 19:00:05 -07:00
										 |  |  |   color_change_listener_ = | 
					
						
							|  |  |  |       std::make_unique<gfx::ScopedSysColorChangeListener>(this); | 
					
						
							| 
									
										
										
										
											2017-02-02 09:34:20 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 02:00:22 +10:00
										 |  |  | }  // namespace api
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |