Chromium Code Reviews| Index: chrome/browser/ui/views/extensions/extension_commands_global_registry_views.cc |
| diff --git a/chrome/browser/ui/views/extensions/extension_commands_global_registry_views.cc b/chrome/browser/ui/views/extensions/extension_commands_global_registry_views.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..9e7d2520440eada5a0c53b98f5a3235d04894867 |
| --- /dev/null |
| +++ b/chrome/browser/ui/views/extensions/extension_commands_global_registry_views.cc |
| @@ -0,0 +1,131 @@ |
| +// Copyright (c) 2013 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +#include "chrome/browser/ui/views/extensions/extension_commands_global_registry_views.h" |
| + |
| +#include "base/lazy_instance.h" |
| +#include "chrome/browser/chrome_notification_types.h" |
| +#include "chrome/browser/extensions/api/commands/command_service.h" |
| +#include "chrome/browser/extensions/extension_keybinding_registry.h" |
| +#include "chrome/browser/extensions/extension_service.h" |
| +#include "chrome/browser/extensions/global_shortcut_listener.h" |
| +#include "chrome/browser/profiles/profile.h" |
| +#include "chrome/common/extensions/extension.h" |
| + |
| +namespace extensions { |
| + |
| +ExtensionCommandsGlobalRegistryViews::ExtensionCommandsGlobalRegistryViews( |
| + Profile* profile) |
| + : ExtensionKeybindingRegistry( |
| + profile, ExtensionKeybindingRegistry::ALL_EXTENSIONS, NULL), |
| + profile_(profile) { |
| + Init(); |
| +} |
| + |
| +ExtensionCommandsGlobalRegistryViews::~ExtensionCommandsGlobalRegistryViews() { |
| + EventTargets::const_iterator iter; |
| + for (iter = event_targets_.begin(); iter != event_targets_.end(); ++iter) { |
| + GlobalShortcutListener::GetInstance()->UnregisterAccelerator( |
| + iter->first, this); |
| + } |
| +} |
| + |
| +static base::LazyInstance< |
| + ProfileKeyedAPIFactory<ExtensionCommandsGlobalRegistryViews> > |
| +g_factory = LAZY_INSTANCE_INITIALIZER; |
| + |
| +// static |
| +ProfileKeyedAPIFactory<ExtensionCommandsGlobalRegistryViews>* |
| +ExtensionCommandsGlobalRegistryViews::GetFactoryInstance() { |
| + return &g_factory.Get(); |
| +} |
| + |
| +template<> |
| +void ProfileKeyedAPIFactory< |
| + ExtensionCommandsGlobalRegistryViews>::DeclareFactoryDependencies() { |
| + DependsOn(ExtensionSystemFactory::GetInstance()); |
| +} |
| + |
| +// static |
| +ExtensionCommandsGlobalRegistryViews* |
| +ExtensionCommandsGlobalRegistryViews::Get(Profile* profile) { |
| + return ProfileKeyedAPIFactory< |
| + ExtensionCommandsGlobalRegistryViews>::GetForProfile(profile); |
| +} |
| + |
| + |
| +void ExtensionCommandsGlobalRegistryViews::AddExtensionKeybinding( |
| + const extensions::Extension* extension, |
| + const std::string& command_name) { |
| + // This object only handles named commands, not browser/page actions. |
| + if (ShouldIgnoreCommand(command_name)) |
| + return; |
| + |
| + extensions::CommandService* command_service = |
| + extensions::CommandService::Get(profile_); |
| + // Add all the active keybindings (except page actions and browser actions, |
| + // which are handled elsewhere). |
| + extensions::CommandMap commands; |
| + if (!command_service->GetNamedCommands( |
| + extension->id(), |
| + extensions::CommandService::ACTIVE_ONLY, |
| + extensions::CommandService::GLOBAL, |
| + &commands)) |
| + return; |
| + |
| + extensions::CommandMap::const_iterator iter = commands.begin(); |
| + for (; iter != commands.end(); ++iter) { |
| + if (!command_name.empty() && (iter->second.command_name() != command_name)) |
| + continue; |
| + |
| + LOG(ERROR) << "Adding global keybinding for " << extension->name().c_str() |
| + << " " << command_name.c_str() |
| + << " key: " << iter->second.accelerator().GetShortcutText(); |
| + |
| + event_targets_[iter->second.accelerator()] = |
| + std::make_pair(extension->id(), iter->second.command_name()); |
| + |
| + GlobalShortcutListener::GetInstance()->RegisterAccelerator( |
| + iter->second.accelerator(), this); |
| + } |
| +} |
| + |
| +void ExtensionCommandsGlobalRegistryViews::RemoveExtensionKeybinding( |
| + const extensions::Extension* extension, |
| + const std::string& command_name) { |
| + LOG(ERROR) << "Removing keybinding for " << extension->name().c_str() |
| + << " " << command_name.c_str(); |
| + |
| + // This object only handles named commands, not browser/page actions. |
|
zhchbin
2013/09/24 02:41:22
We should handle *global* named commands only. Cur
Finnur
2013/09/24 10:53:23
I don't see the code here as being incorrect. The
zhchbin
2013/09/24 11:25:27
ExtensionKeybindingRegistry is observing chrome::N
Finnur
2013/09/25 13:11:40
I see. I am currently investigating a crash (not r
zhchbin
2013/09/25 13:27:13
I would begin as soon as I can! So many homework f
Finnur
2013/09/26 13:53:08
OK, found some time to think about this some more.
zhchbin
2013/09/26 14:44:59
Hm... I mean to said that we can check whether the
Finnur
2013/09/26 15:00:59
No, that's what I'm saying: The Command in questio
zhchbin
2013/09/26 15:13:05
:) OK let's stop talking about this and go on.
(
|
| + if (ShouldIgnoreCommand(command_name)) |
| + return; |
| + |
| + EventTargets::iterator iter = event_targets_.begin(); |
| + while (iter != event_targets_.end()) { |
| + if (iter->second.first != extension->id() || |
| + (!command_name.empty() && (iter->second.second != command_name))) { |
| + ++iter; |
| + continue; // Not the extension or command we asked for. |
| + } |
| + |
| + GlobalShortcutListener::GetInstance()->UnregisterAccelerator( |
| + iter->first, this); |
| + |
| + EventTargets::iterator old = iter++; |
| + event_targets_.erase(old); |
| + } |
| +} |
| + |
| +void ExtensionCommandsGlobalRegistryViews::OnKeyPressed( |
| + const ui::Accelerator& accelerator) { |
| + EventTargets::iterator it = event_targets_.find(accelerator); |
| + if (it == event_targets_.end()) { |
| + NOTREACHED(); // Shouldn't get this event for something not registered. |
| + return; |
| + } |
| + |
| + CommandExecuted(it->second.first, it->second.second); |
| +} |
| + |
| +} // namespace extensions |