| // Copyright 2014 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 "extensions/browser/guest_view/extension_options/extension_options_guest.h" |
| |
| #include "base/values.h" |
| #include "components/crx_file/id_util.h" |
| #include "content/public/browser/render_process_host.h" |
| #include "content/public/browser/site_instance.h" |
| #include "content/public/browser/web_contents.h" |
| #include "extensions/browser/api/extensions_api_client.h" |
| #include "extensions/browser/extension_function_dispatcher.h" |
| #include "extensions/browser/extension_registry.h" |
| #include "extensions/browser/extension_web_contents_observer.h" |
| #include "extensions/browser/guest_view/extension_options/extension_options_constants.h" |
| #include "extensions/browser/guest_view/extension_options/extension_options_guest_delegate.h" |
| #include "extensions/browser/guest_view/guest_view_manager.h" |
| #include "extensions/common/api/extension_options_internal.h" |
| #include "extensions/common/constants.h" |
| #include "extensions/common/extension.h" |
| #include "extensions/common/extension_messages.h" |
| #include "extensions/common/manifest_handlers/options_page_info.h" |
| #include "extensions/common/permissions/permissions_data.h" |
| #include "extensions/strings/grit/extensions_strings.h" |
| #include "ipc/ipc_message_macros.h" |
| |
| using content::WebContents; |
| using namespace extensions::core_api; |
| |
| namespace extensions { |
| |
| // static |
| const char ExtensionOptionsGuest::Type[] = "extensionoptions"; |
| |
| ExtensionOptionsGuest::ExtensionOptionsGuest( |
| content::BrowserContext* browser_context, |
| int guest_instance_id) |
| : GuestView<ExtensionOptionsGuest>(browser_context, guest_instance_id), |
| extension_options_guest_delegate_( |
| extensions::ExtensionsAPIClient::Get() |
| ->CreateExtensionOptionsGuestDelegate(this)), |
| has_navigated_(false) { |
| } |
| |
| ExtensionOptionsGuest::~ExtensionOptionsGuest() { |
| } |
| |
| // static |
| extensions::GuestViewBase* ExtensionOptionsGuest::Create( |
| content::BrowserContext* browser_context, |
| int guest_instance_id) { |
| return new ExtensionOptionsGuest(browser_context, guest_instance_id); |
| } |
| |
| void ExtensionOptionsGuest::CreateWebContents( |
| const std::string& embedder_extension_id, |
| int embedder_render_process_id, |
| const GURL& embedder_site_url, |
| const base::DictionaryValue& create_params, |
| const WebContentsCreatedCallback& callback) { |
| // Get the extension's base URL. |
| std::string extension_id; |
| create_params.GetString(extensionoptions::kExtensionId, &extension_id); |
| |
| if (!crx_file::id_util::IdIsValid(extension_id)) { |
| callback.Run(NULL); |
| return; |
| } |
| |
| if (crx_file::id_util::IdIsValid(embedder_extension_id) && |
| extension_id != embedder_extension_id) { |
| // Extensions cannot embed other extensions' options pages. |
| callback.Run(NULL); |
| return; |
| } |
| |
| GURL extension_url = |
| extensions::Extension::GetBaseURLFromExtensionId(extension_id); |
| if (!extension_url.is_valid()) { |
| callback.Run(NULL); |
| return; |
| } |
| |
| // Get the options page URL for later use. |
| extensions::ExtensionRegistry* registry = |
| extensions::ExtensionRegistry::Get(browser_context()); |
| const extensions::Extension* extension = |
| registry->enabled_extensions().GetByID(extension_id); |
| if (!extension) { |
| // The ID was valid but the extension didn't exist. Typically this will |
| // happen when an extension is disabled. |
| callback.Run(NULL); |
| return; |
| } |
| |
| options_page_ = extensions::OptionsPageInfo::GetOptionsPage(extension); |
| if (!options_page_.is_valid()) { |
| callback.Run(NULL); |
| return; |
| } |
| |
| // Create a WebContents using the extension URL. The options page's |
| // WebContents should live in the same process as its parent extension's |
| // WebContents, so we can use |extension_url| for creating the SiteInstance. |
| content::SiteInstance* options_site_instance = |
| content::SiteInstance::CreateForURL(browser_context(), extension_url); |
| WebContents::CreateParams params(browser_context(), options_site_instance); |
| params.guest_delegate = this; |
| callback.Run(WebContents::Create(params)); |
| } |
| |
| void ExtensionOptionsGuest::DidAttachToEmbedder() { |
| SetUpAutoSize(); |
| |
| // We should not re-navigate on reattachment. |
| if (has_navigated_) |
| return; |
| |
| web_contents()->GetController().LoadURL(options_page_, |
| content::Referrer(), |
| ui::PAGE_TRANSITION_LINK, |
| std::string()); |
| has_navigated_ = true; |
| } |
| |
| void ExtensionOptionsGuest::DidInitialize() { |
| extension_function_dispatcher_.reset( |
| new extensions::ExtensionFunctionDispatcher(browser_context(), this)); |
| if (extension_options_guest_delegate_) { |
| extension_options_guest_delegate_->DidInitialize(); |
| } |
| } |
| |
| void ExtensionOptionsGuest::DidStopLoading() { |
| scoped_ptr<base::DictionaryValue> args(new base::DictionaryValue()); |
| DispatchEventToEmbedder(new extensions::GuestViewBase::Event( |
| extension_options_internal::OnLoad::kEventName, args.Pass())); |
| } |
| |
| const char* ExtensionOptionsGuest::GetAPINamespace() const { |
| return extensionoptions::kAPINamespace; |
| } |
| |
| int ExtensionOptionsGuest::GetTaskPrefix() const { |
| return IDS_EXTENSION_TASK_MANAGER_EXTENSIONOPTIONS_TAG_PREFIX; |
| } |
| |
| void ExtensionOptionsGuest::GuestSizeChangedDueToAutoSize( |
| const gfx::Size& old_size, |
| const gfx::Size& new_size) { |
| scoped_ptr<base::DictionaryValue> args(new base::DictionaryValue()); |
| args->SetInteger(extensionoptions::kNewWidth, new_size.width()); |
| args->SetInteger(extensionoptions::kNewHeight, new_size.height()); |
| args->SetInteger(extensionoptions::kOldWidth, old_size.width()); |
| args->SetInteger(extensionoptions::kOldHeight, old_size.height()); |
| DispatchEventToEmbedder(new extensions::GuestViewBase::Event( |
| extension_options_internal::OnSizeChanged::kEventName, args.Pass())); |
| } |
| |
| bool ExtensionOptionsGuest::IsAutoSizeSupported() const { |
| return true; |
| } |
| |
| content::WebContents* ExtensionOptionsGuest::GetAssociatedWebContents() const { |
| return web_contents(); |
| } |
| |
| content::WebContents* ExtensionOptionsGuest::OpenURLFromTab( |
| content::WebContents* source, |
| const content::OpenURLParams& params) { |
| if (!extension_options_guest_delegate_) |
| return NULL; |
| |
| // Don't allow external URLs with the CURRENT_TAB disposition be opened in |
| // this guest view, change the disposition to NEW_FOREGROUND_TAB. |
| if ((!params.url.SchemeIs(extensions::kExtensionScheme) || |
| params.url.host() != options_page_.host()) && |
| params.disposition == CURRENT_TAB) { |
| return extension_options_guest_delegate_->OpenURLInNewTab( |
| content::OpenURLParams(params.url, |
| params.referrer, |
| params.frame_tree_node_id, |
| NEW_FOREGROUND_TAB, |
| params.transition, |
| params.is_renderer_initiated)); |
| } |
| return extension_options_guest_delegate_->OpenURLInNewTab(params); |
| } |
| |
| void ExtensionOptionsGuest::CloseContents(content::WebContents* source) { |
| DispatchEventToEmbedder(new extensions::GuestViewBase::Event( |
| extension_options_internal::OnClose::kEventName, |
| make_scoped_ptr(new base::DictionaryValue()))); |
| } |
| |
| bool ExtensionOptionsGuest::HandleContextMenu( |
| const content::ContextMenuParams& params) { |
| if (!extension_options_guest_delegate_) |
| return false; |
| |
| return extension_options_guest_delegate_->HandleContextMenu(params); |
| } |
| |
| bool ExtensionOptionsGuest::ShouldCreateWebContents( |
| content::WebContents* web_contents, |
| int route_id, |
| WindowContainerType window_container_type, |
| const base::string16& frame_name, |
| const GURL& target_url, |
| const std::string& partition_id, |
| content::SessionStorageNamespace* session_storage_namespace) { |
| // This method handles opening links from within the guest. Since this guest |
| // view is used for displaying embedded extension options, we want any |
| // external links to be opened in a new tab, not in a new guest view. |
| // Therefore we just open the URL in a new tab, and since we aren't handling |
| // the new web contents, we return false. |
| // TODO(ericzeng): Open the tab in the background if the click was a |
| // ctrl-click or middle mouse button click |
| if (extension_options_guest_delegate_) { |
| extension_options_guest_delegate_->OpenURLInNewTab( |
| content::OpenURLParams(target_url, |
| content::Referrer(), |
| NEW_FOREGROUND_TAB, |
| ui::PAGE_TRANSITION_LINK, |
| false)); |
| } |
| return false; |
| } |
| |
| bool ExtensionOptionsGuest::OnMessageReceived(const IPC::Message& message) { |
| bool handled = true; |
| IPC_BEGIN_MESSAGE_MAP(ExtensionOptionsGuest, message) |
| IPC_MESSAGE_HANDLER(ExtensionHostMsg_Request, OnRequest) |
| IPC_MESSAGE_UNHANDLED(handled = false) |
| IPC_END_MESSAGE_MAP() |
| return handled; |
| } |
| |
| void ExtensionOptionsGuest::OnRequest( |
| const ExtensionHostMsg_Request_Params& params) { |
| extension_function_dispatcher_->Dispatch(params, |
| web_contents()->GetRenderViewHost()); |
| } |
| |
| void ExtensionOptionsGuest::SetUpAutoSize() { |
| // Read the autosize parameters passed in from the embedder. |
| bool auto_size_enabled = false; |
| attach_params()->GetBoolean(extensionoptions::kAttributeAutoSize, |
| &auto_size_enabled); |
| |
| int max_height = 0; |
| int max_width = 0; |
| attach_params()->GetInteger(extensionoptions::kAttributeMaxHeight, |
| &max_height); |
| attach_params()->GetInteger(extensionoptions::kAttributeMaxWidth, &max_width); |
| |
| int min_height = 0; |
| int min_width = 0; |
| attach_params()->GetInteger(extensionoptions::kAttributeMinHeight, |
| &min_height); |
| attach_params()->GetInteger(extensionoptions::kAttributeMinWidth, &min_width); |
| |
| // Call SetAutoSize to apply all the appropriate validation and clipping of |
| // values. |
| SetAutoSize(auto_size_enabled, |
| gfx::Size(min_width, min_height), |
| gfx::Size(max_width, max_height)); |
| } |
| |
| } // namespace extensions |