 Chromium Code Reviews
 Chromium Code Reviews Issue 23812010:
  Implement first part of supporting global extension commands.  (Closed) 
  Base URL: svn://svn.chromium.org/chrome/trunk/src
    
  
    Issue 23812010:
  Implement first part of supporting global extension commands.  (Closed) 
  Base URL: svn://svn.chromium.org/chrome/trunk/src| 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 |