| // Copyright 2019 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/extensions/extension_browser_window_helper.h" |
| |
| #include "chrome/browser/extensions/tab_helper.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/ui/browser.h" |
| #include "chrome/browser/ui/browser_command_controller.h" |
| #include "chrome/browser/ui/browser_window.h" |
| #include "chrome/browser/ui/tabs/tab_enums.h" |
| #include "chrome/browser/ui/tabs/tab_strip_model.h" |
| #include "chrome/browser/ui/tabs/tab_utils.h" |
| #include "chrome/browser/ui/web_applications/app_browser_controller.h" |
| #include "content/public/browser/render_frame_host.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/common/url_constants.h" |
| #include "extensions/browser/unloaded_extension_reason.h" |
| #include "url/origin.h" |
| |
| namespace extensions { |
| |
| namespace { |
| |
| // Returns true if the given |web_contents| should be closed when the extension |
| // is unloaded. |
| bool ShouldCloseTabOnExtensionUnload(const Extension* extension, |
| Browser* browser, |
| content::WebContents* web_contents) { |
| // Case 1: A "regular" extension page, e.g. chrome-extension://<id>/page.html. |
| // Note: we check the tuple or precursor tuple in order to close any |
| // windows with opaque origins that were opened by extensions, and may |
| // still be running code. |
| const url::SchemeHostPort& tuple_or_precursor_tuple = |
| web_contents->GetPrimaryMainFrame() |
| ->GetLastCommittedOrigin() |
| .GetTupleOrPrecursorTupleIfOpaque(); |
| if (tuple_or_precursor_tuple.scheme() == extensions::kExtensionScheme && |
| tuple_or_precursor_tuple.host() == extension->id()) { |
| // Edge-case: Chrome URL overrides (such as NTP overrides) are handled |
| // differently (reloaded), and managed by ExtensionWebUI. Ignore them. |
| if (!web_contents->GetLastCommittedURL().SchemeIs( |
| content::kChromeUIScheme)) { |
| return true; |
| } |
| } |
| |
| // Case 2: Check if the page is a page associated with a hosted app, which |
| // can have non-extension schemes. For example, the Gmail hosted app would |
| // have a URL of https://mail.google.com. |
| if (TabHelper::FromWebContents(web_contents)->GetExtensionAppId() == |
| extension->id()) { |
| return true; |
| } |
| |
| return false; |
| } |
| |
| // Unmutes the given |contents| if it was muted by the extension with |
| // |extension_id|. |
| void UnmuteIfMutedByExtension(content::WebContents* contents, |
| const ExtensionId& extension_id) { |
| LastMuteMetadata::CreateForWebContents(contents); // Ensures metadata exists. |
| LastMuteMetadata* const metadata = |
| LastMuteMetadata::FromWebContents(contents); |
| if (metadata->reason == TabMutedReason::EXTENSION && |
| metadata->extension_id == extension_id) { |
| chrome::SetTabAudioMuted(contents, false, TabMutedReason::EXTENSION, |
| extension_id); |
| } |
| } |
| |
| } // namespace |
| |
| ExtensionBrowserWindowHelper::ExtensionBrowserWindowHelper(Browser* browser) |
| : browser_(browser) { |
| registry_observation_.Observe(ExtensionRegistry::Get(browser_->profile())); |
| } |
| |
| ExtensionBrowserWindowHelper::~ExtensionBrowserWindowHelper() = default; |
| |
| void ExtensionBrowserWindowHelper::OnExtensionLoaded( |
| content::BrowserContext* browser_context, |
| const Extension* extension) { |
| browser_->command_controller()->ExtensionStateChanged(); |
| } |
| |
| void ExtensionBrowserWindowHelper::OnExtensionUnloaded( |
| content::BrowserContext* browser_context, |
| const Extension* extension, |
| UnloadedExtensionReason reason) { |
| browser_->command_controller()->ExtensionStateChanged(); |
| |
| // Clean up any tabs from |extension|, unless it was terminated. In the |
| // terminated case (as when the extension crashed), we let the sad tabs stay. |
| if (reason != extensions::UnloadedExtensionReason::TERMINATE) |
| CleanUpTabsOnUnload(extension); |
| |
| // If an extension page was active, the toolbar may need to be updated to hide |
| // the extension name in the location icon. |
| browser_->window()->UpdateToolbar(nullptr); |
| } |
| |
| void ExtensionBrowserWindowHelper::CleanUpTabsOnUnload( |
| const Extension* extension) { |
| TabStripModel* tab_strip_model = browser_->tab_strip_model(); |
| // Iterate backwards as we may remove items while iterating. |
| for (int i = tab_strip_model->count() - 1; i >= 0; --i) { |
| content::WebContents* web_contents = tab_strip_model->GetWebContentsAt(i); |
| if (ShouldCloseTabOnExtensionUnload(extension, browser_, web_contents)) |
| tab_strip_model->CloseWebContentsAt(i, TabCloseTypes::CLOSE_NONE); |
| else |
| UnmuteIfMutedByExtension(web_contents, extension->id()); |
| } |
| } |
| |
| } // namespace extensions |