123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229 |
- // Copyright (c) 2014 GitHub, Inc.
- // Use of this source code is governed by the MIT license that can be
- // found in the LICENSE file.
- #include <iomanip>
- #include <string_view>
- #include <dwmapi.h>
- #include <windows.devices.enumeration.h>
- #include <wrl/client.h>
- #include "shell/browser/api/electron_api_system_preferences.h"
- #include "base/containers/fixed_flat_map.h"
- #include "base/win/core_winrt_util.h"
- #include "base/win/windows_types.h"
- #include "base/win/wrapped_window_proc.h"
- #include "shell/common/color_util.h"
- #include "shell/common/process_util.h"
- #include "ui/gfx/color_utils.h"
- #include "ui/gfx/win/hwnd_util.h"
- namespace electron {
- namespace {
- const wchar_t kSystemPreferencesWindowClass[] =
- L"Electron_SystemPreferencesHostWindow";
- 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";
- }
- }
- } // namespace
- namespace api {
- bool SystemPreferences::IsAeroGlassEnabled() {
- return true;
- }
- std::string hexColorDWORDToRGBA(DWORD color) {
- DWORD rgba = color << 8 | color >> 24;
- std::ostringstream stream;
- stream << std::hex << std::setw(8) << std::setfill('0') << rgba;
- return stream.str();
- }
- std::string SystemPreferences::GetAccentColor() {
- DWORD color = 0;
- BOOL opaque = FALSE;
- if (FAILED(DwmGetColorizationColor(&color, &opaque))) {
- return "";
- }
- return hexColorDWORDToRGBA(color);
- }
- std::string SystemPreferences::GetColor(gin_helper::ErrorThrower thrower,
- const std::string& color) {
- 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},
- });
- if (const auto* iter = Lookup.find(color); iter != Lookup.end())
- return ToRGBAHex(color_utils::GetSysSkColor(iter->second));
- thrower.ThrowError("Unknown color: " + color);
- return "";
- }
- std::string SystemPreferences::GetMediaAccessStatus(
- gin_helper::ErrorThrower thrower,
- const std::string& media_type) {
- 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 {
- thrower.ThrowError("Invalid media type");
- return {};
- }
- }
- void SystemPreferences::InitializeWindow() {
- if (electron::IsUtilityProcess())
- return;
- // 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())
- color_change_listener_ =
- std::make_unique<gfx::ScopedSysColorChangeListener>(this);
- else
- Browser::Get()->AddObserver(this);
- WNDCLASSEX window_class;
- base::win::InitializeWindowClass(
- kSystemPreferencesWindowClass,
- &base::win::WrappedWindowProc<SystemPreferences::WndProcStatic>, 0, 0, 0,
- nullptr, nullptr, nullptr, nullptr, nullptr, &window_class);
- 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".
- window_ = CreateWindow(MAKEINTATOM(atom_), 0, WS_POPUP, 0, 0, 0, 0, 0, 0,
- instance_, 0);
- gfx::CheckWindowCreated(window_, ::GetLastError());
- gfx::SetWindowUserData(window_, this);
- }
- LRESULT CALLBACK SystemPreferences::WndProcStatic(HWND hwnd,
- UINT message,
- WPARAM wparam,
- LPARAM lparam) {
- auto* msg_wnd = reinterpret_cast<SystemPreferences*>(
- 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,
- UINT message,
- WPARAM wparam,
- LPARAM lparam) {
- if (message == WM_DWMCOLORIZATIONCOLORCHANGED) {
- DWORD new_color = (DWORD)wparam;
- 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);
- }
- void SystemPreferences::OnSysColorChange() {
- Emit("color-changed");
- }
- void SystemPreferences::OnFinishLaunching(base::Value::Dict launch_info) {
- color_change_listener_ =
- std::make_unique<gfx::ScopedSysColorChangeListener>(this);
- }
- } // namespace api
- } // namespace electron
|