• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "chrome/browser/extensions/extension_action_manager.h"
6 
7 #include "chrome/browser/extensions/api/system_indicator/system_indicator_manager_factory.h"
8 #include "chrome/browser/extensions/extension_action.h"
9 #include "chrome/browser/profiles/profile.h"
10 #include "components/keyed_service/content/browser_context_dependency_manager.h"
11 #include "extensions/browser/extension_registry.h"
12 #include "extensions/browser/extension_system.h"
13 #include "extensions/browser/extensions_browser_client.h"
14 #include "extensions/common/constants.h"
15 #include "extensions/common/manifest_handlers/icons_handler.h"
16 
17 namespace extensions {
18 
19 namespace {
20 
21 // BrowserContextKeyedServiceFactory for ExtensionActionManager.
22 class ExtensionActionManagerFactory : public BrowserContextKeyedServiceFactory {
23  public:
24   // BrowserContextKeyedServiceFactory implementation:
GetForBrowserContext(content::BrowserContext * context)25   static ExtensionActionManager* GetForBrowserContext(
26       content::BrowserContext* context) {
27     return static_cast<ExtensionActionManager*>(
28         GetInstance()->GetServiceForBrowserContext(context, true));
29   }
30 
31   static ExtensionActionManagerFactory* GetInstance();
32 
33  private:
34   friend struct DefaultSingletonTraits<ExtensionActionManagerFactory>;
35 
ExtensionActionManagerFactory()36   ExtensionActionManagerFactory()
37       : BrowserContextKeyedServiceFactory(
38           "ExtensionActionManager",
39           BrowserContextDependencyManager::GetInstance()) {
40   }
41 
BuildServiceInstanceFor(content::BrowserContext * profile) const42   virtual KeyedService* BuildServiceInstanceFor(
43       content::BrowserContext* profile) const OVERRIDE {
44     return new ExtensionActionManager(static_cast<Profile*>(profile));
45   }
46 
GetBrowserContextToUse(content::BrowserContext * context) const47   virtual content::BrowserContext* GetBrowserContextToUse(
48       content::BrowserContext* context) const OVERRIDE {
49     return ExtensionsBrowserClient::Get()->GetOriginalContext(context);
50   }
51 };
52 
53 ExtensionActionManagerFactory*
GetInstance()54 ExtensionActionManagerFactory::GetInstance() {
55   return Singleton<ExtensionActionManagerFactory>::get();
56 }
57 
58 }  // namespace
59 
ExtensionActionManager(Profile * profile)60 ExtensionActionManager::ExtensionActionManager(Profile* profile)
61     : profile_(profile), extension_registry_observer_(this) {
62   CHECK_EQ(profile, profile->GetOriginalProfile())
63       << "Don't instantiate this with an incognito profile.";
64   extension_registry_observer_.Add(ExtensionRegistry::Get(profile_));
65 }
66 
~ExtensionActionManager()67 ExtensionActionManager::~ExtensionActionManager() {
68   // Don't assert that the ExtensionAction maps are empty because Extensions are
69   // sometimes (only in tests?) not unloaded before the Profile is destroyed.
70 }
71 
Get(content::BrowserContext * context)72 ExtensionActionManager* ExtensionActionManager::Get(
73     content::BrowserContext* context) {
74   return ExtensionActionManagerFactory::GetForBrowserContext(context);
75 }
76 
OnExtensionUnloaded(content::BrowserContext * browser_context,const Extension * extension,UnloadedExtensionInfo::Reason reason)77 void ExtensionActionManager::OnExtensionUnloaded(
78     content::BrowserContext* browser_context,
79     const Extension* extension,
80     UnloadedExtensionInfo::Reason reason) {
81   page_actions_.erase(extension->id());
82   browser_actions_.erase(extension->id());
83   system_indicators_.erase(extension->id());
84 }
85 
86 namespace {
87 
88 // Loads resources missing from |action| (i.e. title, icons) from the "icons"
89 // key of |extension|'s manifest.
PopulateMissingValues(const Extension & extension,ExtensionAction * action)90 void PopulateMissingValues(const Extension& extension,
91                            ExtensionAction* action) {
92   // If the title is missing from |action|, set it to |extension|'s name.
93   if (action->GetTitle(ExtensionAction::kDefaultTabId).empty())
94     action->SetTitle(ExtensionAction::kDefaultTabId, extension.name());
95 
96   scoped_ptr<ExtensionIconSet> default_icon(new ExtensionIconSet());
97   if (action->default_icon())
98     *default_icon = *action->default_icon();
99 
100   const ExtensionIconSet& extension_icons = IconsInfo::GetIcons(&extension);
101   std::string largest_icon = extension_icons.Get(
102       extension_misc::EXTENSION_ICON_GIGANTOR,
103       ExtensionIconSet::MATCH_SMALLER);
104 
105   if (!largest_icon.empty()) {
106     int largest_icon_size = extension_icons.GetIconSizeFromPath(largest_icon);
107     // Replace any missing extension action icons with the largest icon
108     // retrieved from |extension|'s manifest so long as the largest icon is
109     // larger than the current key.
110     for (int i = extension_misc::kNumExtensionActionIconSizes - 1;
111          i >= 0; --i) {
112       int size = extension_misc::kExtensionActionIconSizes[i].size;
113       if (default_icon->Get(size, ExtensionIconSet::MATCH_BIGGER).empty()
114           && largest_icon_size > size) {
115         default_icon->Add(size, largest_icon);
116         break;
117       }
118     }
119     action->set_default_icon(default_icon.Pass());
120   }
121 }
122 
123 // Returns map[extension_id] if that entry exists. Otherwise, if
124 // action_info!=NULL, creates an ExtensionAction from it, fills in the map, and
125 // returns that.  Otherwise (action_info==NULL), returns NULL.
GetOrCreateOrNull(std::map<std::string,linked_ptr<ExtensionAction>> * map,const Extension & extension,ActionInfo::Type action_type,const ActionInfo * action_info,Profile * profile)126 ExtensionAction* GetOrCreateOrNull(
127     std::map<std::string, linked_ptr<ExtensionAction> >* map,
128     const Extension& extension,
129     ActionInfo::Type action_type,
130     const ActionInfo* action_info,
131     Profile* profile) {
132   std::map<std::string, linked_ptr<ExtensionAction> >::const_iterator it =
133       map->find(extension.id());
134   if (it != map->end())
135     return it->second.get();
136   if (!action_info)
137     return NULL;
138 
139   // Only create action info for enabled extensions.
140   // This avoids bugs where actions are recreated just after being removed
141   // in response to OnExtensionUnloaded().
142   if (!ExtensionRegistry::Get(profile)
143       ->enabled_extensions().Contains(extension.id())) {
144     return NULL;
145   }
146 
147   linked_ptr<ExtensionAction> action(new ExtensionAction(
148       extension.id(), action_type, *action_info));
149   (*map)[extension.id()] = action;
150   PopulateMissingValues(extension, action.get());
151   return action.get();
152 }
153 
154 }  // namespace
155 
GetPageAction(const Extension & extension) const156 ExtensionAction* ExtensionActionManager::GetPageAction(
157     const Extension& extension) const {
158   return GetOrCreateOrNull(&page_actions_, extension,
159                            ActionInfo::TYPE_PAGE,
160                            ActionInfo::GetPageActionInfo(&extension),
161                            profile_);
162 }
163 
GetBrowserAction(const Extension & extension) const164 ExtensionAction* ExtensionActionManager::GetBrowserAction(
165     const Extension& extension) const {
166   return GetOrCreateOrNull(&browser_actions_, extension,
167                            ActionInfo::TYPE_BROWSER,
168                            ActionInfo::GetBrowserActionInfo(&extension),
169                            profile_);
170 }
171 
GetBestFitAction(const Extension & extension,ActionInfo::Type type) const172 scoped_ptr<ExtensionAction> ExtensionActionManager::GetBestFitAction(
173     const Extension& extension,
174     ActionInfo::Type type) const {
175   const ActionInfo* info = ActionInfo::GetBrowserActionInfo(&extension);
176   if (!info)
177     info = ActionInfo::GetPageActionInfo(&extension);
178 
179   // Create a new ExtensionAction of |type| with |extension|'s ActionInfo.
180   // If no ActionInfo exists for |extension|, create and return a new action
181   // with a blank ActionInfo.
182   // Populate any missing values from |extension|'s manifest.
183   scoped_ptr<ExtensionAction> new_action(new ExtensionAction(
184       extension.id(), type, info ? *info : ActionInfo()));
185   PopulateMissingValues(extension, new_action.get());
186   return new_action.Pass();
187 }
188 
GetSystemIndicator(const Extension & extension) const189 ExtensionAction* ExtensionActionManager::GetSystemIndicator(
190     const Extension& extension) const {
191   // If it does not already exist, create the SystemIndicatorManager for the
192   // given profile.  This could return NULL if the system indicator area is
193   // unavailable on the current system.  If so, return NULL to signal that
194   // the system indicator area is unusable.
195   if (!SystemIndicatorManagerFactory::GetForProfile(profile_))
196     return NULL;
197 
198   return GetOrCreateOrNull(&system_indicators_, extension,
199                            ActionInfo::TYPE_SYSTEM_INDICATOR,
200                            ActionInfo::GetSystemIndicatorInfo(&extension),
201                            profile_);
202 }
203 
GetExtensionAction(const Extension & extension) const204 ExtensionAction* ExtensionActionManager::GetExtensionAction(
205     const Extension& extension) const {
206   ExtensionAction* action = GetBrowserAction(extension);
207   return action ? action : GetPageAction(extension);
208 }
209 
210 }  // namespace extensions
211