| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include <algorithm> | 5 #include <algorithm> |
| 6 #include <set> | 6 #include <set> |
| 7 | 7 |
| 8 #include "chrome/browser/tab_contents/render_view_context_menu.h" | 8 #include "chrome/browser/tab_contents/render_view_context_menu.h" |
| 9 | 9 |
| 10 #include "app/l10n_util.h" | 10 #include "app/l10n_util.h" |
| 11 #include "base/command_line.h" | 11 #include "base/command_line.h" |
| 12 #include "base/logging.h" | 12 #include "base/logging.h" |
| 13 #include "base/metrics/histogram.h" | 13 #include "base/metrics/histogram.h" |
| 14 #include "base/stl_util-inl.h" | 14 #include "base/stl_util-inl.h" |
| 15 #include "base/string_util.h" | 15 #include "base/string_util.h" |
| 16 #include "base/time.h" | 16 #include "base/time.h" |
| 17 #include "chrome/app/chrome_dll_resource.h" | 17 #include "chrome/app/chrome_dll_resource.h" |
| 18 #include "chrome/browser/autocomplete/autocomplete_classifier.h" | 18 #include "chrome/browser/autocomplete/autocomplete_classifier.h" |
| 19 #include "chrome/browser/autocomplete/autocomplete_edit.h" | 19 #include "chrome/browser/autocomplete/autocomplete_edit.h" |
| 20 #include "chrome/browser/autocomplete/autocomplete_match.h" | 20 #include "chrome/browser/autocomplete/autocomplete_match.h" |
| 21 #include "chrome/browser/browser_process.h" | 21 #include "chrome/browser/browser_process.h" |
| 22 #include "chrome/browser/child_process_security_policy.h" | 22 #include "chrome/browser/child_process_security_policy.h" |
| 23 #include "chrome/browser/debugger/devtools_manager.h" | 23 #include "chrome/browser/debugger/devtools_manager.h" |
| 24 #include "chrome/browser/debugger/devtools_window.h" | 24 #include "chrome/browser/debugger/devtools_window.h" |
| 25 #include "chrome/browser/download/download_manager.h" | 25 #include "chrome/browser/download/download_manager.h" |
| 26 #include "chrome/browser/extensions/extension_event_router.h" |
| 26 #include "chrome/browser/extensions/extensions_service.h" | 27 #include "chrome/browser/extensions/extensions_service.h" |
| 27 #include "chrome/browser/fonts_languages_window.h" | 28 #include "chrome/browser/fonts_languages_window.h" |
| 28 #include "chrome/browser/metrics/user_metrics.h" | 29 #include "chrome/browser/metrics/user_metrics.h" |
| 29 #include "chrome/browser/net/browser_url_util.h" | 30 #include "chrome/browser/net/browser_url_util.h" |
| 30 #include "chrome/browser/page_info_window.h" | 31 #include "chrome/browser/page_info_window.h" |
| 31 #include "chrome/browser/platform_util.h" | 32 #include "chrome/browser/platform_util.h" |
| 32 #include "chrome/browser/prefs/pref_member.h" | 33 #include "chrome/browser/prefs/pref_member.h" |
| 33 #include "chrome/browser/prefs/pref_service.h" | 34 #include "chrome/browser/prefs/pref_service.h" |
| 34 #include "chrome/browser/profile.h" | 35 #include "chrome/browser/profile.h" |
| 35 #include "chrome/browser/renderer_host/render_view_host.h" | 36 #include "chrome/browser/renderer_host/render_view_host.h" |
| (...skipping 107 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 143 if (patterns.is_empty()) | 144 if (patterns.is_empty()) |
| 144 return true; | 145 return true; |
| 145 return patterns.ContainsURL(url); | 146 return patterns.ContainsURL(url); |
| 146 } | 147 } |
| 147 | 148 |
| 148 static const GURL& GetDocumentURL(const ContextMenuParams& params) { | 149 static const GURL& GetDocumentURL(const ContextMenuParams& params) { |
| 149 return params.frame_url.is_empty() ? params.page_url : params.frame_url; | 150 return params.frame_url.is_empty() ? params.page_url : params.frame_url; |
| 150 } | 151 } |
| 151 | 152 |
| 152 // Given a list of items, returns the ones that match given the contents | 153 // Given a list of items, returns the ones that match given the contents |
| 153 // of |params|. | 154 // of |params| and the profile. |
| 154 static ExtensionMenuItem::List GetRelevantExtensionItems( | 155 static ExtensionMenuItem::List GetRelevantExtensionItems( |
| 155 const ExtensionMenuItem::List& items, | 156 const ExtensionMenuItem::List& items, |
| 156 const ContextMenuParams& params) { | 157 const ContextMenuParams& params, |
| 158 Profile* profile, |
| 159 bool can_cross_incognito) { |
| 157 ExtensionMenuItem::List result; | 160 ExtensionMenuItem::List result; |
| 158 for (ExtensionMenuItem::List::const_iterator i = items.begin(); | 161 for (ExtensionMenuItem::List::const_iterator i = items.begin(); |
| 159 i != items.end(); ++i) { | 162 i != items.end(); ++i) { |
| 160 const ExtensionMenuItem* item = *i; | 163 const ExtensionMenuItem* item = *i; |
| 161 | 164 |
| 162 if (!ExtensionContextMatch(params, item->contexts())) | 165 if (!ExtensionContextMatch(params, item->contexts())) |
| 163 continue; | 166 continue; |
| 164 | 167 |
| 165 const GURL& document_url = GetDocumentURL(params); | 168 const GURL& document_url = GetDocumentURL(params); |
| 166 if (!ExtensionPatternMatch(item->document_url_patterns(), document_url)) | 169 if (!ExtensionPatternMatch(item->document_url_patterns(), document_url)) |
| 167 continue; | 170 continue; |
| 168 | 171 |
| 169 const GURL& target_url = | 172 const GURL& target_url = |
| 170 params.src_url.is_empty() ? params.link_url : params.src_url; | 173 params.src_url.is_empty() ? params.link_url : params.src_url; |
| 171 if (!ExtensionPatternMatch(item->target_url_patterns(), target_url)) | 174 if (!ExtensionPatternMatch(item->target_url_patterns(), target_url)) |
| 172 continue; | 175 continue; |
| 173 | 176 |
| 174 result.push_back(*i); | 177 if (item->id().profile == profile || can_cross_incognito) |
| 178 result.push_back(*i); |
| 175 } | 179 } |
| 176 return result; | 180 return result; |
| 177 } | 181 } |
| 178 | 182 |
| 179 void RenderViewContextMenu::AppendExtensionItems( | 183 void RenderViewContextMenu::AppendExtensionItems( |
| 180 const std::string& extension_id, int* index) { | 184 const std::string& extension_id, int* index) { |
| 181 ExtensionsService* service = profile_->GetExtensionsService(); | 185 ExtensionsService* service = profile_->GetExtensionsService(); |
| 182 ExtensionMenuManager* manager = service->menu_manager(); | 186 ExtensionMenuManager* manager = service->menu_manager(); |
| 183 const Extension* extension = service->GetExtensionById(extension_id, false); | 187 const Extension* extension = service->GetExtensionById(extension_id, false); |
| 188 bool can_cross_incognito = |
| 189 ExtensionEventRouter::CanCrossIncognito(profile_, extension); |
| 184 DCHECK_GE(*index, 0); | 190 DCHECK_GE(*index, 0); |
| 185 int max_index = | 191 int max_index = |
| 186 IDC_EXTENSIONS_CONTEXT_CUSTOM_LAST - IDC_EXTENSIONS_CONTEXT_CUSTOM_FIRST; | 192 IDC_EXTENSIONS_CONTEXT_CUSTOM_LAST - IDC_EXTENSIONS_CONTEXT_CUSTOM_FIRST; |
| 187 if (!extension || *index >= max_index) | 193 if (!extension || *index >= max_index) |
| 188 return; | 194 return; |
| 189 | 195 |
| 190 // Find matching items. | 196 // Find matching items. |
| 191 const ExtensionMenuItem::List* all_items = manager->MenuItems(extension_id); | 197 const ExtensionMenuItem::List* all_items = manager->MenuItems(extension_id); |
| 192 if (!all_items || all_items->empty()) | 198 if (!all_items || all_items->empty()) |
| 193 return; | 199 return; |
| 194 ExtensionMenuItem::List items = | 200 ExtensionMenuItem::List items = |
| 195 GetRelevantExtensionItems(*all_items, params_); | 201 GetRelevantExtensionItems(*all_items, params_, profile_, |
| 202 can_cross_incognito); |
| 196 if (items.empty()) | 203 if (items.empty()) |
| 197 return; | 204 return; |
| 198 | 205 |
| 199 // If this is the first extension-provided menu item, add a separator. | 206 // If this is the first extension-provided menu item, add a separator. |
| 200 if (*index == 0) | 207 if (*index == 0) |
| 201 menu_model_.AddSeparator(); | 208 menu_model_.AddSeparator(); |
| 202 | 209 |
| 203 int menu_id = IDC_EXTENSIONS_CONTEXT_CUSTOM_FIRST + (*index)++; | 210 int menu_id = IDC_EXTENSIONS_CONTEXT_CUSTOM_FIRST + (*index)++; |
| 204 | 211 |
| 205 // Extensions are only allowed one top-level slot (and it can't be a radio or | 212 // Extensions are only allowed one top-level slot (and it can't be a radio or |
| 206 // checkbox item because we are going to put the extension icon next to it). | 213 // checkbox item because we are going to put the extension icon next to it). |
| 207 // If they have more than that, we automatically push them into a submenu. | 214 // If they have more than that, we automatically push them into a submenu. |
| 208 string16 title; | 215 string16 title; |
| 209 ExtensionMenuItem::List submenu_items; | 216 ExtensionMenuItem::List submenu_items; |
| 210 if (items.size() > 1 || items[0]->type() != ExtensionMenuItem::NORMAL) { | 217 if (items.size() > 1 || items[0]->type() != ExtensionMenuItem::NORMAL) { |
| 211 title = UTF8ToUTF16(extension->name()); | 218 title = UTF8ToUTF16(extension->name()); |
| 212 submenu_items = items; | 219 submenu_items = items; |
| 213 } else { | 220 } else { |
| 214 ExtensionMenuItem* item = items[0]; | 221 ExtensionMenuItem* item = items[0]; |
| 215 extension_item_map_[menu_id] = item->id(); | 222 extension_item_map_[menu_id] = item->id(); |
| 216 title = item->TitleWithReplacement(PrintableSelectionText(), | 223 title = item->TitleWithReplacement(PrintableSelectionText(), |
| 217 kMaxExtensionItemTitleLength); | 224 kMaxExtensionItemTitleLength); |
| 218 submenu_items = GetRelevantExtensionItems(item->children(), params_); | 225 submenu_items = GetRelevantExtensionItems(item->children(), params_, |
| 226 profile_, can_cross_incognito); |
| 219 } | 227 } |
| 220 | 228 |
| 221 // Now add our item(s) to the menu_model_. | 229 // Now add our item(s) to the menu_model_. |
| 222 if (submenu_items.empty()) { | 230 if (submenu_items.empty()) { |
| 223 menu_model_.AddItem(menu_id, title); | 231 menu_model_.AddItem(menu_id, title); |
| 224 } else { | 232 } else { |
| 225 menus::SimpleMenuModel* submenu = new menus::SimpleMenuModel(this); | 233 menus::SimpleMenuModel* submenu = new menus::SimpleMenuModel(this); |
| 226 extension_menu_models_.push_back(submenu); | 234 extension_menu_models_.push_back(submenu); |
| 227 menu_model_.AddSubMenu(menu_id, title, submenu); | 235 menu_model_.AddSubMenu(menu_id, title, submenu); |
| 228 RecursivelyAppendExtensionItems(submenu_items, submenu, index); | 236 RecursivelyAppendExtensionItems(submenu_items, can_cross_incognito, submenu, |
| 237 index); |
| 229 } | 238 } |
| 230 SetExtensionIcon(extension_id); | 239 SetExtensionIcon(extension_id); |
| 231 } | 240 } |
| 232 | 241 |
| 233 void RenderViewContextMenu::RecursivelyAppendExtensionItems( | 242 void RenderViewContextMenu::RecursivelyAppendExtensionItems( |
| 234 const ExtensionMenuItem::List& items, | 243 const ExtensionMenuItem::List& items, |
| 244 bool can_cross_incognito, |
| 235 menus::SimpleMenuModel* menu_model, | 245 menus::SimpleMenuModel* menu_model, |
| 236 int *index) { | 246 int *index) { |
| 237 string16 selection_text = PrintableSelectionText(); | 247 string16 selection_text = PrintableSelectionText(); |
| 238 ExtensionMenuItem::Type last_type = ExtensionMenuItem::NORMAL; | 248 ExtensionMenuItem::Type last_type = ExtensionMenuItem::NORMAL; |
| 239 int radio_group_id = 1; | 249 int radio_group_id = 1; |
| 240 | 250 |
| 241 for (ExtensionMenuItem::List::const_iterator i = items.begin(); | 251 for (ExtensionMenuItem::List::const_iterator i = items.begin(); |
| 242 i != items.end(); ++i) { | 252 i != items.end(); ++i) { |
| 243 ExtensionMenuItem* item = *i; | 253 ExtensionMenuItem* item = *i; |
| 244 | 254 |
| 245 // If last item was of type radio but the current one isn't, auto-insert | 255 // If last item was of type radio but the current one isn't, auto-insert |
| 246 // a separator. The converse case is handled below. | 256 // a separator. The converse case is handled below. |
| 247 if (last_type == ExtensionMenuItem::RADIO && | 257 if (last_type == ExtensionMenuItem::RADIO && |
| 248 item->type() != ExtensionMenuItem::RADIO) { | 258 item->type() != ExtensionMenuItem::RADIO) { |
| 249 menu_model->AddSeparator(); | 259 menu_model->AddSeparator(); |
| 250 last_type = ExtensionMenuItem::SEPARATOR; | 260 last_type = ExtensionMenuItem::SEPARATOR; |
| 251 } | 261 } |
| 252 | 262 |
| 253 int menu_id = IDC_EXTENSIONS_CONTEXT_CUSTOM_FIRST + (*index)++; | 263 int menu_id = IDC_EXTENSIONS_CONTEXT_CUSTOM_FIRST + (*index)++; |
| 254 if (menu_id >= IDC_EXTENSIONS_CONTEXT_CUSTOM_LAST) | 264 if (menu_id >= IDC_EXTENSIONS_CONTEXT_CUSTOM_LAST) |
| 255 return; | 265 return; |
| 256 extension_item_map_[menu_id] = item->id(); | 266 extension_item_map_[menu_id] = item->id(); |
| 257 string16 title = item->TitleWithReplacement(selection_text, | 267 string16 title = item->TitleWithReplacement(selection_text, |
| 258 kMaxExtensionItemTitleLength); | 268 kMaxExtensionItemTitleLength); |
| 259 if (item->type() == ExtensionMenuItem::NORMAL) { | 269 if (item->type() == ExtensionMenuItem::NORMAL) { |
| 260 ExtensionMenuItem::List children = | 270 ExtensionMenuItem::List children = |
| 261 GetRelevantExtensionItems(item->children(), params_); | 271 GetRelevantExtensionItems(item->children(), params_, |
| 272 profile_, can_cross_incognito); |
| 262 if (children.size() == 0) { | 273 if (children.size() == 0) { |
| 263 menu_model->AddItem(menu_id, title); | 274 menu_model->AddItem(menu_id, title); |
| 264 } else { | 275 } else { |
| 265 menus::SimpleMenuModel* submenu = new menus::SimpleMenuModel(this); | 276 menus::SimpleMenuModel* submenu = new menus::SimpleMenuModel(this); |
| 266 extension_menu_models_.push_back(submenu); | 277 extension_menu_models_.push_back(submenu); |
| 267 menu_model->AddSubMenu(menu_id, title, submenu); | 278 menu_model->AddSubMenu(menu_id, title, submenu); |
| 268 RecursivelyAppendExtensionItems(children, submenu, index); | 279 RecursivelyAppendExtensionItems(children, can_cross_incognito, |
| 280 submenu, index); |
| 269 } | 281 } |
| 270 } else if (item->type() == ExtensionMenuItem::CHECKBOX) { | 282 } else if (item->type() == ExtensionMenuItem::CHECKBOX) { |
| 271 menu_model->AddCheckItem(menu_id, title); | 283 menu_model->AddCheckItem(menu_id, title); |
| 272 } else if (item->type() == ExtensionMenuItem::RADIO) { | 284 } else if (item->type() == ExtensionMenuItem::RADIO) { |
| 273 if (i != items.begin() && | 285 if (i != items.begin() && |
| 274 last_type != ExtensionMenuItem::RADIO) { | 286 last_type != ExtensionMenuItem::RADIO) { |
| 275 radio_group_id++; | 287 radio_group_id++; |
| 276 | 288 |
| 277 // Auto-append a separator if needed. | 289 // Auto-append a separator if needed. |
| 278 if (last_type != ExtensionMenuItem::SEPARATOR) | 290 if (last_type != ExtensionMenuItem::SEPARATOR) |
| (...skipping 1196 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1475 profile_->GetPrefs()->GetString(prefs::kAcceptLanguages), | 1487 profile_->GetPrefs()->GetString(prefs::kAcceptLanguages), |
| 1476 g_browser_process->clipboard()); | 1488 g_browser_process->clipboard()); |
| 1477 } | 1489 } |
| 1478 | 1490 |
| 1479 void RenderViewContextMenu::MediaPlayerActionAt( | 1491 void RenderViewContextMenu::MediaPlayerActionAt( |
| 1480 const gfx::Point& location, | 1492 const gfx::Point& location, |
| 1481 const WebMediaPlayerAction& action) { | 1493 const WebMediaPlayerAction& action) { |
| 1482 source_tab_contents_->render_view_host()->MediaPlayerActionAt( | 1494 source_tab_contents_->render_view_host()->MediaPlayerActionAt( |
| 1483 location, action); | 1495 location, action); |
| 1484 } | 1496 } |
| OLD | NEW |