• 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/menu_manager.h"
6 
7 #include <algorithm>
8 
9 #include "base/json/json_writer.h"
10 #include "base/logging.h"
11 #include "base/stl_util.h"
12 #include "base/strings/string_util.h"
13 #include "base/strings/utf_string_conversions.h"
14 #include "base/values.h"
15 #include "chrome/browser/chrome_notification_types.h"
16 #include "chrome/browser/extensions/extension_service.h"
17 #include "chrome/browser/extensions/extension_tab_util.h"
18 #include "chrome/browser/extensions/menu_manager_factory.h"
19 #include "chrome/browser/extensions/tab_helper.h"
20 #include "chrome/browser/profiles/profile.h"
21 #include "chrome/common/extensions/api/chrome_web_view_internal.h"
22 #include "chrome/common/extensions/api/context_menus.h"
23 #include "content/public/browser/notification_details.h"
24 #include "content/public/browser/notification_service.h"
25 #include "content/public/browser/notification_source.h"
26 #include "content/public/browser/web_contents.h"
27 #include "content/public/common/context_menu_params.h"
28 #include "extensions/browser/event_router.h"
29 #include "extensions/browser/extension_registry.h"
30 #include "extensions/browser/extension_system.h"
31 #include "extensions/browser/guest_view/web_view/web_view_guest.h"
32 #include "extensions/browser/state_store.h"
33 #include "extensions/common/extension.h"
34 #include "extensions/common/manifest_handlers/background_info.h"
35 #include "ui/gfx/favicon_size.h"
36 #include "ui/gfx/text_elider.h"
37 
38 using content::WebContents;
39 using extensions::ExtensionSystem;
40 
41 namespace extensions {
42 
43 namespace context_menus = api::context_menus;
44 namespace chrome_web_view = api::chrome_web_view_internal;
45 
46 namespace {
47 
48 // Keys for serialization to and from Value to store in the preferences.
49 const char kContextMenusKey[] = "context_menus";
50 
51 const char kCheckedKey[] = "checked";
52 const char kContextsKey[] = "contexts";
53 const char kDocumentURLPatternsKey[] = "document_url_patterns";
54 const char kEnabledKey[] = "enabled";
55 const char kIncognitoKey[] = "incognito";
56 const char kParentUIDKey[] = "parent_uid";
57 const char kStringUIDKey[] = "string_uid";
58 const char kTargetURLPatternsKey[] = "target_url_patterns";
59 const char kTitleKey[] = "title";
60 const char kTypeKey[] = "type";
61 
SetIdKeyValue(base::DictionaryValue * properties,const char * key,const MenuItem::Id & id)62 void SetIdKeyValue(base::DictionaryValue* properties,
63                    const char* key,
64                    const MenuItem::Id& id) {
65   if (id.uid == 0)
66     properties->SetString(key, id.string_uid);
67   else
68     properties->SetInteger(key, id.uid);
69 }
70 
MenuItemsFromValue(const std::string & extension_id,base::Value * value)71 MenuItem::List MenuItemsFromValue(const std::string& extension_id,
72                                   base::Value* value) {
73   MenuItem::List items;
74 
75   base::ListValue* list = NULL;
76   if (!value || !value->GetAsList(&list))
77     return items;
78 
79   for (size_t i = 0; i < list->GetSize(); ++i) {
80     base::DictionaryValue* dict = NULL;
81     if (!list->GetDictionary(i, &dict))
82       continue;
83     MenuItem* item = MenuItem::Populate(
84         extension_id, *dict, NULL);
85     if (!item)
86       continue;
87     items.push_back(item);
88   }
89   return items;
90 }
91 
MenuItemsToValue(const MenuItem::List & items)92 scoped_ptr<base::Value> MenuItemsToValue(const MenuItem::List& items) {
93   scoped_ptr<base::ListValue> list(new base::ListValue());
94   for (size_t i = 0; i < items.size(); ++i)
95     list->Append(items[i]->ToValue().release());
96   return scoped_ptr<base::Value>(list.release());
97 }
98 
GetStringList(const base::DictionaryValue & dict,const std::string & key,std::vector<std::string> * out)99 bool GetStringList(const base::DictionaryValue& dict,
100                    const std::string& key,
101                    std::vector<std::string>* out) {
102   if (!dict.HasKey(key))
103     return true;
104 
105   const base::ListValue* list = NULL;
106   if (!dict.GetListWithoutPathExpansion(key, &list))
107     return false;
108 
109   for (size_t i = 0; i < list->GetSize(); ++i) {
110     std::string pattern;
111     if (!list->GetString(i, &pattern))
112       return false;
113     out->push_back(pattern);
114   }
115 
116   return true;
117 }
118 
119 }  // namespace
120 
MenuItem(const Id & id,const std::string & title,bool checked,bool enabled,Type type,const ContextList & contexts)121 MenuItem::MenuItem(const Id& id,
122                    const std::string& title,
123                    bool checked,
124                    bool enabled,
125                    Type type,
126                    const ContextList& contexts)
127     : id_(id),
128       title_(title),
129       type_(type),
130       checked_(checked),
131       enabled_(enabled),
132       contexts_(contexts) {}
133 
~MenuItem()134 MenuItem::~MenuItem() {
135   STLDeleteElements(&children_);
136 }
137 
ReleaseChild(const Id & child_id,bool recursive)138 MenuItem* MenuItem::ReleaseChild(const Id& child_id,
139                                  bool recursive) {
140   for (List::iterator i = children_.begin(); i != children_.end(); ++i) {
141     MenuItem* child = NULL;
142     if ((*i)->id() == child_id) {
143       child = *i;
144       children_.erase(i);
145       return child;
146     } else if (recursive) {
147       child = (*i)->ReleaseChild(child_id, recursive);
148       if (child)
149         return child;
150     }
151   }
152   return NULL;
153 }
154 
GetFlattenedSubtree(MenuItem::List * list)155 void MenuItem::GetFlattenedSubtree(MenuItem::List* list) {
156   list->push_back(this);
157   for (List::iterator i = children_.begin(); i != children_.end(); ++i)
158     (*i)->GetFlattenedSubtree(list);
159 }
160 
RemoveAllDescendants()161 std::set<MenuItem::Id> MenuItem::RemoveAllDescendants() {
162   std::set<Id> result;
163   for (List::iterator i = children_.begin(); i != children_.end(); ++i) {
164     MenuItem* child = *i;
165     result.insert(child->id());
166     std::set<Id> removed = child->RemoveAllDescendants();
167     result.insert(removed.begin(), removed.end());
168   }
169   STLDeleteElements(&children_);
170   return result;
171 }
172 
TitleWithReplacement(const base::string16 & selection,size_t max_length) const173 base::string16 MenuItem::TitleWithReplacement(const base::string16& selection,
174                                               size_t max_length) const {
175   base::string16 result = base::UTF8ToUTF16(title_);
176   // TODO(asargent) - Change this to properly handle %% escaping so you can
177   // put "%s" in titles that won't get substituted.
178   ReplaceSubstringsAfterOffset(&result, 0, base::ASCIIToUTF16("%s"), selection);
179 
180   if (result.length() > max_length)
181     result = gfx::TruncateString(result, max_length, gfx::WORD_BREAK);
182   return result;
183 }
184 
SetChecked(bool checked)185 bool MenuItem::SetChecked(bool checked) {
186   if (type_ != CHECKBOX && type_ != RADIO)
187     return false;
188   checked_ = checked;
189   return true;
190 }
191 
AddChild(MenuItem * item)192 void MenuItem::AddChild(MenuItem* item) {
193   item->parent_id_.reset(new Id(id_));
194   children_.push_back(item);
195 }
196 
ToValue() const197 scoped_ptr<base::DictionaryValue> MenuItem::ToValue() const {
198   scoped_ptr<base::DictionaryValue> value(new base::DictionaryValue);
199   // Should only be called for extensions with event pages, which only have
200   // string IDs for items.
201   DCHECK_EQ(0, id_.uid);
202   value->SetString(kStringUIDKey, id_.string_uid);
203   value->SetBoolean(kIncognitoKey, id_.incognito);
204   value->SetInteger(kTypeKey, type_);
205   if (type_ != SEPARATOR)
206     value->SetString(kTitleKey, title_);
207   if (type_ == CHECKBOX || type_ == RADIO)
208     value->SetBoolean(kCheckedKey, checked_);
209   value->SetBoolean(kEnabledKey, enabled_);
210   value->Set(kContextsKey, contexts_.ToValue().release());
211   if (parent_id_) {
212     DCHECK_EQ(0, parent_id_->uid);
213     value->SetString(kParentUIDKey, parent_id_->string_uid);
214   }
215   value->Set(kDocumentURLPatternsKey,
216              document_url_patterns_.ToValue().release());
217   value->Set(kTargetURLPatternsKey, target_url_patterns_.ToValue().release());
218   return value.Pass();
219 }
220 
221 // static
Populate(const std::string & extension_id,const base::DictionaryValue & value,std::string * error)222 MenuItem* MenuItem::Populate(const std::string& extension_id,
223                              const base::DictionaryValue& value,
224                              std::string* error) {
225   bool incognito = false;
226   if (!value.GetBoolean(kIncognitoKey, &incognito))
227     return NULL;
228   Id id(incognito, MenuItem::ExtensionKey(extension_id));
229   if (!value.GetString(kStringUIDKey, &id.string_uid))
230     return NULL;
231   int type_int;
232   Type type = NORMAL;
233   if (!value.GetInteger(kTypeKey, &type_int))
234     return NULL;
235   type = static_cast<Type>(type_int);
236   std::string title;
237   if (type != SEPARATOR && !value.GetString(kTitleKey, &title))
238     return NULL;
239   bool checked = false;
240   if ((type == CHECKBOX || type == RADIO) &&
241       !value.GetBoolean(kCheckedKey, &checked)) {
242     return NULL;
243   }
244   bool enabled = true;
245   if (!value.GetBoolean(kEnabledKey, &enabled))
246     return NULL;
247   ContextList contexts;
248   const base::Value* contexts_value = NULL;
249   if (!value.Get(kContextsKey, &contexts_value))
250     return NULL;
251   if (!contexts.Populate(*contexts_value))
252     return NULL;
253 
254   scoped_ptr<MenuItem> result(new MenuItem(
255       id, title, checked, enabled, type, contexts));
256 
257   std::vector<std::string> document_url_patterns;
258   if (!GetStringList(value, kDocumentURLPatternsKey, &document_url_patterns))
259     return NULL;
260   std::vector<std::string> target_url_patterns;
261   if (!GetStringList(value, kTargetURLPatternsKey, &target_url_patterns))
262     return NULL;
263 
264   if (!result->PopulateURLPatterns(&document_url_patterns,
265                                    &target_url_patterns,
266                                    error)) {
267     return NULL;
268   }
269 
270   // parent_id is filled in from the value, but it might not be valid. It's left
271   // to be validated upon being added (via AddChildItem) to the menu manager.
272   scoped_ptr<Id> parent_id(
273       new Id(incognito, MenuItem::ExtensionKey(extension_id)));
274   if (value.HasKey(kParentUIDKey)) {
275     if (!value.GetString(kParentUIDKey, &parent_id->string_uid))
276       return NULL;
277     result->parent_id_.swap(parent_id);
278   }
279   return result.release();
280 }
281 
PopulateURLPatterns(std::vector<std::string> * document_url_patterns,std::vector<std::string> * target_url_patterns,std::string * error)282 bool MenuItem::PopulateURLPatterns(
283     std::vector<std::string>* document_url_patterns,
284     std::vector<std::string>* target_url_patterns,
285     std::string* error) {
286   if (document_url_patterns) {
287     if (!document_url_patterns_.Populate(
288             *document_url_patterns, URLPattern::SCHEME_ALL, true, error)) {
289       return false;
290     }
291   }
292   if (target_url_patterns) {
293     if (!target_url_patterns_.Populate(
294             *target_url_patterns, URLPattern::SCHEME_ALL, true, error)) {
295       return false;
296     }
297   }
298   return true;
299 }
300 
301 // static
302 const char MenuManager::kOnContextMenus[] = "contextMenus";
303 const char MenuManager::kOnWebviewContextMenus[] =
304     "webViewInternal.contextMenus";
305 
MenuManager(content::BrowserContext * context,StateStore * store)306 MenuManager::MenuManager(content::BrowserContext* context, StateStore* store)
307     : extension_registry_observer_(this),
308       browser_context_(context),
309       store_(store) {
310   extension_registry_observer_.Add(ExtensionRegistry::Get(browser_context_));
311   registrar_.Add(this, chrome::NOTIFICATION_PROFILE_DESTROYED,
312                  content::NotificationService::AllSources());
313   if (store_)
314     store_->RegisterKey(kContextMenusKey);
315 }
316 
~MenuManager()317 MenuManager::~MenuManager() {
318   MenuItemMap::iterator i;
319   for (i = context_items_.begin(); i != context_items_.end(); ++i) {
320     STLDeleteElements(&(i->second));
321   }
322 }
323 
324 // static
Get(content::BrowserContext * context)325 MenuManager* MenuManager::Get(content::BrowserContext* context) {
326   return MenuManagerFactory::GetForBrowserContext(context);
327 }
328 
ExtensionIds()329 std::set<MenuItem::ExtensionKey> MenuManager::ExtensionIds() {
330   std::set<MenuItem::ExtensionKey> id_set;
331   for (MenuItemMap::const_iterator i = context_items_.begin();
332        i != context_items_.end(); ++i) {
333     id_set.insert(i->first);
334   }
335   return id_set;
336 }
337 
MenuItems(const MenuItem::ExtensionKey & key)338 const MenuItem::List* MenuManager::MenuItems(
339     const MenuItem::ExtensionKey& key) {
340   MenuItemMap::iterator i = context_items_.find(key);
341   if (i != context_items_.end()) {
342     return &(i->second);
343   }
344   return NULL;
345 }
346 
AddContextItem(const Extension * extension,MenuItem * item)347 bool MenuManager::AddContextItem(const Extension* extension, MenuItem* item) {
348   const MenuItem::ExtensionKey& key = item->id().extension_key;
349   // The item must have a non-empty extension id, and not have already been
350   // added.
351   if (key.empty() || ContainsKey(items_by_id_, item->id()))
352     return false;
353 
354   DCHECK_EQ(extension->id(), key.extension_id);
355 
356   bool first_item = !ContainsKey(context_items_, key);
357   context_items_[key].push_back(item);
358   items_by_id_[item->id()] = item;
359 
360   if (item->type() == MenuItem::RADIO) {
361     if (item->checked())
362       RadioItemSelected(item);
363     else
364       SanitizeRadioList(context_items_[key]);
365   }
366 
367   // If this is the first item for this extension, start loading its icon.
368   if (first_item)
369     icon_manager_.LoadIcon(browser_context_, extension);
370 
371   return true;
372 }
373 
AddChildItem(const MenuItem::Id & parent_id,MenuItem * child)374 bool MenuManager::AddChildItem(const MenuItem::Id& parent_id,
375                                MenuItem* child) {
376   MenuItem* parent = GetItemById(parent_id);
377   if (!parent || parent->type() != MenuItem::NORMAL ||
378       parent->incognito() != child->incognito() ||
379       parent->extension_id() != child->extension_id() ||
380       ContainsKey(items_by_id_, child->id()))
381     return false;
382   parent->AddChild(child);
383   items_by_id_[child->id()] = child;
384 
385   if (child->type() == MenuItem::RADIO)
386     SanitizeRadioList(parent->children());
387   return true;
388 }
389 
DescendantOf(MenuItem * item,const MenuItem::Id & ancestor_id)390 bool MenuManager::DescendantOf(MenuItem* item,
391                                const MenuItem::Id& ancestor_id) {
392   // Work our way up the tree until we find the ancestor or NULL.
393   MenuItem::Id* id = item->parent_id();
394   while (id != NULL) {
395     DCHECK(*id != item->id());  // Catch circular graphs.
396     if (*id == ancestor_id)
397       return true;
398     MenuItem* next = GetItemById(*id);
399     if (!next) {
400       NOTREACHED();
401       return false;
402     }
403     id = next->parent_id();
404   }
405   return false;
406 }
407 
ChangeParent(const MenuItem::Id & child_id,const MenuItem::Id * parent_id)408 bool MenuManager::ChangeParent(const MenuItem::Id& child_id,
409                                const MenuItem::Id* parent_id) {
410   MenuItem* child = GetItemById(child_id);
411   MenuItem* new_parent = parent_id ? GetItemById(*parent_id) : NULL;
412   if ((parent_id && (child_id == *parent_id)) || !child ||
413       (!new_parent && parent_id != NULL) ||
414       (new_parent && (DescendantOf(new_parent, child_id) ||
415                       child->incognito() != new_parent->incognito() ||
416                       child->extension_id() != new_parent->extension_id())))
417     return false;
418 
419   MenuItem::Id* old_parent_id = child->parent_id();
420   if (old_parent_id != NULL) {
421     MenuItem* old_parent = GetItemById(*old_parent_id);
422     if (!old_parent) {
423       NOTREACHED();
424       return false;
425     }
426     MenuItem* taken =
427       old_parent->ReleaseChild(child_id, false /* non-recursive search*/);
428     DCHECK(taken == child);
429     SanitizeRadioList(old_parent->children());
430   } else {
431     // This is a top-level item, so we need to pull it out of our list of
432     // top-level items.
433     const MenuItem::ExtensionKey& child_key = child->id().extension_key;
434     MenuItemMap::iterator i = context_items_.find(child_key);
435     if (i == context_items_.end()) {
436       NOTREACHED();
437       return false;
438     }
439     MenuItem::List& list = i->second;
440     MenuItem::List::iterator j = std::find(list.begin(), list.end(), child);
441     if (j == list.end()) {
442       NOTREACHED();
443       return false;
444     }
445     list.erase(j);
446     SanitizeRadioList(list);
447   }
448 
449   if (new_parent) {
450     new_parent->AddChild(child);
451     SanitizeRadioList(new_parent->children());
452   } else {
453     const MenuItem::ExtensionKey& child_key = child->id().extension_key;
454     context_items_[child_key].push_back(child);
455     child->parent_id_.reset(NULL);
456     SanitizeRadioList(context_items_[child_key]);
457   }
458   return true;
459 }
460 
RemoveContextMenuItem(const MenuItem::Id & id)461 bool MenuManager::RemoveContextMenuItem(const MenuItem::Id& id) {
462   if (!ContainsKey(items_by_id_, id))
463     return false;
464 
465   MenuItem* menu_item = GetItemById(id);
466   DCHECK(menu_item);
467   const MenuItem::ExtensionKey extension_key = id.extension_key;
468   MenuItemMap::iterator i = context_items_.find(extension_key);
469   if (i == context_items_.end()) {
470     NOTREACHED();
471     return false;
472   }
473 
474   bool result = false;
475   std::set<MenuItem::Id> items_removed;
476   MenuItem::List& list = i->second;
477   MenuItem::List::iterator j;
478   for (j = list.begin(); j < list.end(); ++j) {
479     // See if the current top-level item is a match.
480     if ((*j)->id() == id) {
481       items_removed = (*j)->RemoveAllDescendants();
482       items_removed.insert(id);
483       delete *j;
484       list.erase(j);
485       result = true;
486       SanitizeRadioList(list);
487       break;
488     } else {
489       // See if the item to remove was found as a descendant of the current
490       // top-level item.
491       MenuItem* child = (*j)->ReleaseChild(id, true /* recursive */);
492       if (child) {
493         items_removed = child->RemoveAllDescendants();
494         items_removed.insert(id);
495         SanitizeRadioList(GetItemById(*child->parent_id())->children());
496         delete child;
497         result = true;
498         break;
499       }
500     }
501   }
502   DCHECK(result);  // The check at the very top should have prevented this.
503 
504   // Clear entries from the items_by_id_ map.
505   std::set<MenuItem::Id>::iterator removed_iter;
506   for (removed_iter = items_removed.begin();
507        removed_iter != items_removed.end();
508        ++removed_iter) {
509     items_by_id_.erase(*removed_iter);
510   }
511 
512   if (list.empty()) {
513     context_items_.erase(extension_key);
514     icon_manager_.RemoveIcon(extension_key.extension_id);
515   }
516   return result;
517 }
518 
RemoveAllContextItems(const MenuItem::ExtensionKey & extension_key)519 void MenuManager::RemoveAllContextItems(
520     const MenuItem::ExtensionKey& extension_key) {
521   MenuItem::List::iterator i;
522   for (i = context_items_[extension_key].begin();
523        i != context_items_[extension_key].end();
524        ++i) {
525     MenuItem* item = *i;
526     items_by_id_.erase(item->id());
527 
528     // Remove descendants from this item and erase them from the lookup cache.
529     std::set<MenuItem::Id> removed_ids = item->RemoveAllDescendants();
530     std::set<MenuItem::Id>::const_iterator j;
531     for (j = removed_ids.begin(); j != removed_ids.end(); ++j) {
532       items_by_id_.erase(*j);
533     }
534   }
535   STLDeleteElements(&context_items_[extension_key]);
536   context_items_.erase(extension_key);
537   icon_manager_.RemoveIcon(extension_key.extension_id);
538 }
539 
GetItemById(const MenuItem::Id & id) const540 MenuItem* MenuManager::GetItemById(const MenuItem::Id& id) const {
541   std::map<MenuItem::Id, MenuItem*>::const_iterator i =
542       items_by_id_.find(id);
543   if (i != items_by_id_.end())
544     return i->second;
545   else
546     return NULL;
547 }
548 
RadioItemSelected(MenuItem * item)549 void MenuManager::RadioItemSelected(MenuItem* item) {
550   // If this is a child item, we need to get a handle to the list from its
551   // parent. Otherwise get a handle to the top-level list.
552   const MenuItem::List* list = NULL;
553   if (item->parent_id()) {
554     MenuItem* parent = GetItemById(*item->parent_id());
555     if (!parent) {
556       NOTREACHED();
557       return;
558     }
559     list = &(parent->children());
560   } else {
561     const MenuItem::ExtensionKey& key = item->id().extension_key;
562     if (context_items_.find(key) == context_items_.end()) {
563       NOTREACHED();
564       return;
565     }
566     list = &context_items_[key];
567   }
568 
569   // Find where |item| is in the list.
570   MenuItem::List::const_iterator item_location;
571   for (item_location = list->begin(); item_location != list->end();
572        ++item_location) {
573     if (*item_location == item)
574       break;
575   }
576   if (item_location == list->end()) {
577     NOTREACHED();  // We should have found the item.
578     return;
579   }
580 
581   // Iterate backwards from |item| and uncheck any adjacent radio items.
582   MenuItem::List::const_iterator i;
583   if (item_location != list->begin()) {
584     i = item_location;
585     do {
586       --i;
587       if ((*i)->type() != MenuItem::RADIO)
588         break;
589       (*i)->SetChecked(false);
590     } while (i != list->begin());
591   }
592 
593   // Now iterate forwards from |item| and uncheck any adjacent radio items.
594   for (i = item_location + 1; i != list->end(); ++i) {
595     if ((*i)->type() != MenuItem::RADIO)
596       break;
597     (*i)->SetChecked(false);
598   }
599 }
600 
AddURLProperty(base::DictionaryValue * dictionary,const std::string & key,const GURL & url)601 static void AddURLProperty(base::DictionaryValue* dictionary,
602                            const std::string& key, const GURL& url) {
603   if (!url.is_empty())
604     dictionary->SetString(key, url.possibly_invalid_spec());
605 }
606 
ExecuteCommand(content::BrowserContext * context,WebContents * web_contents,const content::ContextMenuParams & params,const MenuItem::Id & menu_item_id)607 void MenuManager::ExecuteCommand(content::BrowserContext* context,
608                                  WebContents* web_contents,
609                                  const content::ContextMenuParams& params,
610                                  const MenuItem::Id& menu_item_id) {
611   EventRouter* event_router = EventRouter::Get(context);
612   if (!event_router)
613     return;
614 
615   MenuItem* item = GetItemById(menu_item_id);
616   if (!item)
617     return;
618 
619   // ExtensionService/Extension can be NULL in unit tests :(
620   ExtensionService* service =
621       ExtensionSystem::Get(browser_context_)->extension_service();
622   const Extension* extension =
623       service ? service->extensions()->GetByID(item->extension_id()) : NULL;
624 
625   if (item->type() == MenuItem::RADIO)
626     RadioItemSelected(item);
627 
628   scoped_ptr<base::ListValue> args(new base::ListValue());
629 
630   base::DictionaryValue* properties = new base::DictionaryValue();
631   SetIdKeyValue(properties, "menuItemId", item->id());
632   if (item->parent_id())
633     SetIdKeyValue(properties, "parentMenuItemId", *item->parent_id());
634 
635   switch (params.media_type) {
636     case blink::WebContextMenuData::MediaTypeImage:
637       properties->SetString("mediaType", "image");
638       break;
639     case blink::WebContextMenuData::MediaTypeVideo:
640       properties->SetString("mediaType", "video");
641       break;
642     case blink::WebContextMenuData::MediaTypeAudio:
643       properties->SetString("mediaType", "audio");
644       break;
645     default:  {}  // Do nothing.
646   }
647 
648   AddURLProperty(properties, "linkUrl", params.unfiltered_link_url);
649   AddURLProperty(properties, "srcUrl", params.src_url);
650   AddURLProperty(properties, "pageUrl", params.page_url);
651   AddURLProperty(properties, "frameUrl", params.frame_url);
652 
653   if (params.selection_text.length() > 0)
654     properties->SetString("selectionText", params.selection_text);
655 
656   properties->SetBoolean("editable", params.is_editable);
657 
658   WebViewGuest* webview_guest = WebViewGuest::FromWebContents(web_contents);
659   if (webview_guest) {
660     // This is used in web_view_internalcustom_bindings.js.
661     // The property is not exposed to developer API.
662     properties->SetInteger("webviewInstanceId",
663                            webview_guest->view_instance_id());
664   }
665 
666   args->Append(properties);
667 
668   // Add the tab info to the argument list.
669   // No tab info in a platform app.
670   if (!extension || !extension->is_platform_app()) {
671     // Note: web_contents are NULL in unit tests :(
672     if (web_contents) {
673       args->Append(ExtensionTabUtil::CreateTabValue(web_contents));
674     } else {
675       args->Append(new base::DictionaryValue());
676     }
677   }
678 
679   if (item->type() == MenuItem::CHECKBOX ||
680       item->type() == MenuItem::RADIO) {
681     bool was_checked = item->checked();
682     properties->SetBoolean("wasChecked", was_checked);
683 
684     // RADIO items always get set to true when you click on them, but CHECKBOX
685     // items get their state toggled.
686     bool checked =
687         (item->type() == MenuItem::RADIO) ? true : !was_checked;
688 
689     item->SetChecked(checked);
690     properties->SetBoolean("checked", item->checked());
691 
692     if (extension)
693       WriteToStorage(extension, item->id().extension_key);
694   }
695 
696   // Note: web_contents are NULL in unit tests :(
697   if (web_contents && extensions::TabHelper::FromWebContents(web_contents)) {
698     extensions::TabHelper::FromWebContents(web_contents)->
699         active_tab_permission_granter()->GrantIfRequested(extension);
700   }
701 
702   {
703     // Dispatch to menu item's .onclick handler.
704     scoped_ptr<Event> event(
705         new Event(webview_guest ? kOnWebviewContextMenus
706                                 : kOnContextMenus,
707                   scoped_ptr<base::ListValue>(args->DeepCopy())));
708     event->restrict_to_browser_context = context;
709     event->user_gesture = EventRouter::USER_GESTURE_ENABLED;
710     event_router->DispatchEventToExtension(item->extension_id(), event.Pass());
711   }
712   {
713     // Dispatch to .contextMenus.onClicked handler.
714     scoped_ptr<Event> event(
715         new Event(webview_guest ? chrome_web_view::OnClicked::kEventName
716                                 : context_menus::OnClicked::kEventName,
717                   args.Pass()));
718     event->restrict_to_browser_context = context;
719     event->user_gesture = EventRouter::USER_GESTURE_ENABLED;
720     if (webview_guest)
721       event->filter_info.SetInstanceID(webview_guest->view_instance_id());
722     event_router->DispatchEventToExtension(item->extension_id(), event.Pass());
723   }
724 }
725 
SanitizeRadioList(const MenuItem::List & item_list)726 void MenuManager::SanitizeRadioList(const MenuItem::List& item_list) {
727   MenuItem::List::const_iterator i = item_list.begin();
728   while (i != item_list.end()) {
729     if ((*i)->type() != MenuItem::RADIO) {
730       ++i;
731       break;
732     }
733 
734     // Uncheck any checked radio items in the run, and at the end reset
735     // the appropriate one to checked. If no check radio items were found,
736     // then check the first radio item in the run.
737     MenuItem::List::const_iterator last_checked = item_list.end();
738     MenuItem::List::const_iterator radio_run_iter;
739     for (radio_run_iter = i; radio_run_iter != item_list.end();
740         ++radio_run_iter) {
741       if ((*radio_run_iter)->type() != MenuItem::RADIO) {
742         break;
743       }
744 
745       if ((*radio_run_iter)->checked()) {
746         last_checked = radio_run_iter;
747         (*radio_run_iter)->SetChecked(false);
748       }
749     }
750 
751     if (last_checked != item_list.end())
752       (*last_checked)->SetChecked(true);
753     else
754       (*i)->SetChecked(true);
755 
756     i = radio_run_iter;
757   }
758 }
759 
ItemUpdated(const MenuItem::Id & id)760 bool MenuManager::ItemUpdated(const MenuItem::Id& id) {
761   if (!ContainsKey(items_by_id_, id))
762     return false;
763 
764   MenuItem* menu_item = GetItemById(id);
765   DCHECK(menu_item);
766 
767   if (menu_item->parent_id()) {
768     SanitizeRadioList(GetItemById(*menu_item->parent_id())->children());
769   } else {
770     MenuItemMap::iterator i =
771         context_items_.find(menu_item->id().extension_key);
772     if (i == context_items_.end()) {
773       NOTREACHED();
774       return false;
775     }
776     SanitizeRadioList(i->second);
777   }
778 
779   return true;
780 }
781 
WriteToStorage(const Extension * extension,const MenuItem::ExtensionKey & extension_key)782 void MenuManager::WriteToStorage(const Extension* extension,
783                                  const MenuItem::ExtensionKey& extension_key) {
784   if (!BackgroundInfo::HasLazyBackgroundPage(extension))
785     return;
786   // <webview> menu items are transient and not stored in storage.
787   if (extension_key.webview_instance_id)
788     return;
789   const MenuItem::List* top_items = MenuItems(extension_key);
790   MenuItem::List all_items;
791   if (top_items) {
792     for (MenuItem::List::const_iterator i = top_items->begin();
793          i != top_items->end(); ++i) {
794       DCHECK(!(*i)->id().extension_key.webview_instance_id);
795       (*i)->GetFlattenedSubtree(&all_items);
796     }
797   }
798 
799   if (store_) {
800     store_->SetExtensionValue(extension->id(), kContextMenusKey,
801                               MenuItemsToValue(all_items));
802   }
803 }
804 
ReadFromStorage(const std::string & extension_id,scoped_ptr<base::Value> value)805 void MenuManager::ReadFromStorage(const std::string& extension_id,
806                                   scoped_ptr<base::Value> value) {
807   const Extension* extension = ExtensionSystem::Get(browser_context_)
808                                    ->extension_service()
809                                    ->extensions()
810                                    ->GetByID(extension_id);
811   if (!extension)
812     return;
813 
814   MenuItem::List items = MenuItemsFromValue(extension_id, value.get());
815   for (size_t i = 0; i < items.size(); ++i) {
816     bool added = false;
817 
818     if (items[i]->parent_id()) {
819       // Parent IDs are stored in the parent_id field for convenience, but
820       // they have not yet been validated. Separate them out here.
821       // Because of the order in which we store items in the prefs, parents will
822       // precede children, so we should already know about any parent items.
823       scoped_ptr<MenuItem::Id> parent_id;
824       parent_id.swap(items[i]->parent_id_);
825       added = AddChildItem(*parent_id, items[i]);
826     } else {
827       added = AddContextItem(extension, items[i]);
828     }
829 
830     if (!added)
831       delete items[i];
832   }
833 }
834 
OnExtensionLoaded(content::BrowserContext * browser_context,const Extension * extension)835 void MenuManager::OnExtensionLoaded(content::BrowserContext* browser_context,
836                                     const Extension* extension) {
837   if (store_ && BackgroundInfo::HasLazyBackgroundPage(extension)) {
838     store_->GetExtensionValue(
839         extension->id(),
840         kContextMenusKey,
841         base::Bind(
842             &MenuManager::ReadFromStorage, AsWeakPtr(), extension->id()));
843   }
844 }
845 
OnExtensionUnloaded(content::BrowserContext * browser_context,const Extension * extension,UnloadedExtensionInfo::Reason reason)846 void MenuManager::OnExtensionUnloaded(content::BrowserContext* browser_context,
847                                       const Extension* extension,
848                                       UnloadedExtensionInfo::Reason reason) {
849   MenuItem::ExtensionKey extension_key(extension->id());
850   if (ContainsKey(context_items_, extension_key)) {
851     RemoveAllContextItems(extension_key);
852   }
853 }
854 
Observe(int type,const content::NotificationSource & source,const content::NotificationDetails & details)855 void MenuManager::Observe(int type,
856                           const content::NotificationSource& source,
857                           const content::NotificationDetails& details) {
858   DCHECK_EQ(chrome::NOTIFICATION_PROFILE_DESTROYED, type);
859   Profile* profile = content::Source<Profile>(source).ptr();
860   // We cannot use profile_->HasOffTheRecordProfile as it may already be
861   // false at this point, if for example the incognito profile was destroyed
862   // using DestroyOffTheRecordProfile.
863   if (profile->GetOriginalProfile() == browser_context_ &&
864       profile->GetOriginalProfile() != profile) {
865     RemoveAllIncognitoContextItems();
866   }
867 }
868 
GetIconForExtension(const std::string & extension_id)869 const SkBitmap& MenuManager::GetIconForExtension(
870     const std::string& extension_id) {
871   return icon_manager_.GetIcon(extension_id);
872 }
873 
RemoveAllIncognitoContextItems()874 void MenuManager::RemoveAllIncognitoContextItems() {
875   // Get all context menu items with "incognito" set to "split".
876   std::set<MenuItem::Id> items_to_remove;
877   std::map<MenuItem::Id, MenuItem*>::const_iterator iter;
878   for (iter = items_by_id_.begin();
879        iter != items_by_id_.end();
880        ++iter) {
881     if (iter->first.incognito)
882       items_to_remove.insert(iter->first);
883   }
884 
885   std::set<MenuItem::Id>::iterator remove_iter;
886   for (remove_iter = items_to_remove.begin();
887        remove_iter != items_to_remove.end();
888        ++remove_iter)
889     RemoveContextMenuItem(*remove_iter);
890 }
891 
ExtensionKey()892 MenuItem::ExtensionKey::ExtensionKey() : webview_instance_id(0) {}
893 
ExtensionKey(const std::string & extension_id,int webview_instance_id)894 MenuItem::ExtensionKey::ExtensionKey(const std::string& extension_id,
895                                      int webview_instance_id)
896     : extension_id(extension_id), webview_instance_id(webview_instance_id) {}
897 
ExtensionKey(const std::string & extension_id)898 MenuItem::ExtensionKey::ExtensionKey(const std::string& extension_id)
899     : extension_id(extension_id), webview_instance_id(0) {}
900 
operator ==(const ExtensionKey & other) const901 bool MenuItem::ExtensionKey::operator==(const ExtensionKey& other) const {
902   return extension_id == other.extension_id &&
903          webview_instance_id == other.webview_instance_id;
904 }
905 
operator <(const ExtensionKey & other) const906 bool MenuItem::ExtensionKey::operator<(const ExtensionKey& other) const {
907   if (extension_id != other.extension_id)
908     return extension_id < other.extension_id;
909 
910   return webview_instance_id < other.webview_instance_id;
911 }
912 
operator !=(const ExtensionKey & other) const913 bool MenuItem::ExtensionKey::operator!=(const ExtensionKey& other) const {
914   return !(*this == other);
915 }
916 
empty() const917 bool MenuItem::ExtensionKey::empty() const {
918   return extension_id.empty() && !webview_instance_id;
919 }
920 
Id()921 MenuItem::Id::Id() : incognito(false), uid(0) {}
922 
Id(bool incognito,const MenuItem::ExtensionKey & extension_key)923 MenuItem::Id::Id(bool incognito, const MenuItem::ExtensionKey& extension_key)
924     : incognito(incognito), extension_key(extension_key), uid(0) {}
925 
~Id()926 MenuItem::Id::~Id() {
927 }
928 
operator ==(const Id & other) const929 bool MenuItem::Id::operator==(const Id& other) const {
930   return (incognito == other.incognito &&
931           extension_key == other.extension_key && uid == other.uid &&
932           string_uid == other.string_uid);
933 }
934 
operator !=(const Id & other) const935 bool MenuItem::Id::operator!=(const Id& other) const {
936   return !(*this == other);
937 }
938 
operator <(const Id & other) const939 bool MenuItem::Id::operator<(const Id& other) const {
940   if (incognito < other.incognito)
941     return true;
942   if (incognito == other.incognito) {
943     if (extension_key < other.extension_key)
944       return true;
945     if (extension_key == other.extension_key) {
946       if (uid < other.uid)
947         return true;
948       if (uid == other.uid)
949         return string_uid < other.string_uid;
950     }
951   }
952   return false;
953 }
954 
955 }  // namespace extensions
956