// 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 "shell/browser/api/electron_api_global_shortcut.h" #include <string> #include <vector> #include "base/stl_util.h" #include "base/strings/utf_string_conversions.h" #include "chrome/common/extensions/command.h" #include "gin/dictionary.h" #include "gin/object_template_builder.h" #include "shell/browser/api/electron_api_system_preferences.h" #include "shell/browser/browser.h" #include "shell/common/gin_converters/accelerator_converter.h" #include "shell/common/gin_converters/callback_converter.h" #include "shell/common/node_includes.h" #if defined(OS_MAC) #include "base/mac/mac_util.h" #endif using extensions::Command; using extensions::GlobalShortcutListener; namespace { #if defined(OS_MAC) bool RegisteringMediaKeyForUntrustedClient(const ui::Accelerator& accelerator) { if (base::mac::IsAtLeastOS10_14()) { constexpr ui::KeyboardCode mediaKeys[] = { ui::VKEY_MEDIA_PLAY_PAUSE, ui::VKEY_MEDIA_NEXT_TRACK, ui::VKEY_MEDIA_PREV_TRACK, ui::VKEY_MEDIA_STOP, ui::VKEY_VOLUME_UP, ui::VKEY_VOLUME_DOWN, ui::VKEY_VOLUME_MUTE}; if (std::find(std::begin(mediaKeys), std::end(mediaKeys), accelerator.key_code()) != std::end(mediaKeys)) { bool trusted = electron::api::SystemPreferences::IsTrustedAccessibilityClient(false); if (!trusted) return true; } } return false; } bool MapHasMediaKeys( const std::map<ui::Accelerator, base::Closure>& accelerator_map) { auto media_key = std::find_if( accelerator_map.begin(), accelerator_map.end(), [](const auto& ac) { return Command::IsMediaKey(ac.first); }); return media_key != accelerator_map.end(); } #endif } // namespace namespace electron { namespace api { gin::WrapperInfo GlobalShortcut::kWrapperInfo = {gin::kEmbedderNativeGin}; GlobalShortcut::GlobalShortcut(v8::Isolate* isolate) {} GlobalShortcut::~GlobalShortcut() { UnregisterAll(); } void GlobalShortcut::OnKeyPressed(const ui::Accelerator& accelerator) { if (accelerator_callback_map_.find(accelerator) == accelerator_callback_map_.end()) { // This should never occur, because if it does, GlobalShortcutListener // notifies us with wrong accelerator. NOTREACHED(); return; } accelerator_callback_map_[accelerator].Run(); } bool GlobalShortcut::RegisterAll( const std::vector<ui::Accelerator>& accelerators, const base::Closure& callback) { if (!electron::Browser::Get()->is_ready()) { gin_helper::ErrorThrower(JavascriptEnvironment::GetIsolate()) .ThrowError("globalShortcut cannot be used before the app is ready"); return false; } std::vector<ui::Accelerator> registered; for (auto& accelerator : accelerators) { if (!Register(accelerator, callback)) { // unregister all shortcuts if any failed UnregisterSome(registered); return false; } registered.push_back(accelerator); } return true; } bool GlobalShortcut::Register(const ui::Accelerator& accelerator, const base::Closure& callback) { if (!electron::Browser::Get()->is_ready()) { gin_helper::ErrorThrower(JavascriptEnvironment::GetIsolate()) .ThrowError("globalShortcut cannot be used before the app is ready"); return false; } #if defined(OS_MAC) if (Command::IsMediaKey(accelerator)) { if (RegisteringMediaKeyForUntrustedClient(accelerator)) return false; GlobalShortcutListener::SetShouldUseInternalMediaKeyHandling(false); } #endif if (!GlobalShortcutListener::GetInstance()->RegisterAccelerator(accelerator, this)) { return false; } accelerator_callback_map_[accelerator] = callback; return true; } void GlobalShortcut::Unregister(const ui::Accelerator& accelerator) { if (!electron::Browser::Get()->is_ready()) { gin_helper::ErrorThrower(JavascriptEnvironment::GetIsolate()) .ThrowError("globalShortcut cannot be used before the app is ready"); return; } if (accelerator_callback_map_.erase(accelerator) == 0) return; #if defined(OS_MAC) if (Command::IsMediaKey(accelerator) && !MapHasMediaKeys(accelerator_callback_map_)) { GlobalShortcutListener::SetShouldUseInternalMediaKeyHandling(true); } #endif GlobalShortcutListener::GetInstance()->UnregisterAccelerator(accelerator, this); } void GlobalShortcut::UnregisterSome( const std::vector<ui::Accelerator>& accelerators) { for (auto& accelerator : accelerators) { Unregister(accelerator); } } bool GlobalShortcut::IsRegistered(const ui::Accelerator& accelerator) { return base::Contains(accelerator_callback_map_, accelerator); } void GlobalShortcut::UnregisterAll() { if (!electron::Browser::Get()->is_ready()) { gin_helper::ErrorThrower(JavascriptEnvironment::GetIsolate()) .ThrowError("globalShortcut cannot be used before the app is ready"); return; } accelerator_callback_map_.clear(); GlobalShortcutListener::GetInstance()->UnregisterAccelerators(this); } // static gin::Handle<GlobalShortcut> GlobalShortcut::Create(v8::Isolate* isolate) { return gin::CreateHandle(isolate, new GlobalShortcut(isolate)); } // static gin::ObjectTemplateBuilder GlobalShortcut::GetObjectTemplateBuilder( v8::Isolate* isolate) { return gin::Wrappable<GlobalShortcut>::GetObjectTemplateBuilder(isolate) .SetMethod("registerAll", &GlobalShortcut::RegisterAll) .SetMethod("register", &GlobalShortcut::Register) .SetMethod("isRegistered", &GlobalShortcut::IsRegistered) .SetMethod("unregister", &GlobalShortcut::Unregister) .SetMethod("unregisterAll", &GlobalShortcut::UnregisterAll); } const char* GlobalShortcut::GetTypeName() { return "GlobalShortcut"; } } // namespace api } // namespace electron namespace { void Initialize(v8::Local<v8::Object> exports, v8::Local<v8::Value> unused, v8::Local<v8::Context> context, void* priv) { v8::Isolate* isolate = context->GetIsolate(); gin::Dictionary dict(isolate, exports); dict.Set("globalShortcut", electron::api::GlobalShortcut::Create(isolate)); } } // namespace NODE_LINKED_MODULE_CONTEXT_AWARE(electron_browser_global_shortcut, Initialize)