// Copyright (c) 2012 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/ui/unload_controller.h" #include "base/location.h" #include "base/single_thread_task_runner.h" #include "base/thread_task_runner_handle.h" #include "chrome/browser/chrome_notification_types.h" #include "chrome/browser/devtools/devtools_window.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_tabstrip.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/notification_source.h" #include "content/public/browser/notification_types.h" #include "content/public/browser/render_view_host.h" #include "content/public/browser/web_contents.h" namespace chrome { //////////////////////////////////////////////////////////////////////////////// // UnloadController, public: UnloadController::UnloadController(Browser* browser) : browser_(browser), is_attempting_to_close_browser_(false), weak_factory_(this) { browser_->tab_strip_model()->AddObserver(this); } UnloadController::~UnloadController() { browser_->tab_strip_model()->RemoveObserver(this); } bool UnloadController::CanCloseContents(content::WebContents* contents) { // Don't try to close the tab when the whole browser is being closed, since // that avoids the fast shutdown path where we just kill all the renderers. if (is_attempting_to_close_browser_) ClearUnloadState(contents, true); return !is_attempting_to_close_browser_ || is_calling_before_unload_handlers(); } // static bool UnloadController::ShouldRunUnloadEventsHelper( content::WebContents* contents) { // If |contents| is being inspected, devtools needs to intercept beforeunload // events. return DevToolsWindow::GetInstanceForInspectedWebContents(contents) != NULL; } // static bool UnloadController::RunUnloadEventsHelper(content::WebContents* contents) { // If there's a devtools window attached to |contents|, // we would like devtools to call its own beforeunload handlers first, // and then call beforeunload handlers for |contents|. // See DevToolsWindow::InterceptPageBeforeUnload for details. if (DevToolsWindow::InterceptPageBeforeUnload(contents)) { return true; } // If the WebContents is not connected yet, then there's no unload // handler we can fire even if the WebContents has an unload listener. // One case where we hit this is in a tab that has an infinite loop // before load. if (contents->NeedToFireBeforeUnload()) { // If the page has unload listeners, then we tell the renderer to fire // them. Once they have fired, we'll get a message back saying whether // to proceed closing the page or not, which sends us back to this method // with the NeedToFireBeforeUnload bit cleared. contents->DispatchBeforeUnload(false); return true; } return false; } bool UnloadController::BeforeUnloadFired(content::WebContents* contents, bool proceed) { if (!proceed) DevToolsWindow::OnPageCloseCanceled(contents); if (!is_attempting_to_close_browser_) { if (!proceed) contents->SetClosedByUserGesture(false); return proceed; } if (!proceed) { CancelWindowClose(); contents->SetClosedByUserGesture(false); return false; } if (RemoveFromSet(&tabs_needing_before_unload_fired_, contents)) { // Now that beforeunload has fired, put the tab on the queue to fire // unload. tabs_needing_unload_fired_.insert(contents); ProcessPendingTabs(); // We want to handle firing the unload event ourselves since we want to // fire all the beforeunload events before attempting to fire the unload // events should the user cancel closing the browser. return false; } return true; } bool UnloadController::ShouldCloseWindow() { if (HasCompletedUnloadProcessing()) return true; // Special case for when we quit an application. The devtools window can // close if it's beforeunload event has already fired which will happen due // to the interception of it's content's beforeunload. if (browser_->is_devtools() && DevToolsWindow::HasFiredBeforeUnloadEventForDevToolsBrowser(browser_)) { return true; } // The behavior followed here varies based on the current phase of the // operation and whether a batched shutdown is in progress. // // If there are tabs with outstanding beforeunload handlers: // 1. If a batched shutdown is in progress: return false. // This is to prevent interference with batched shutdown already in // progress. // 2. Otherwise: start sending beforeunload events and return false. // // Otherwise, If there are no tabs with outstanding beforeunload handlers: // 3. If a batched shutdown is in progress: start sending unload events and // return false. // 4. Otherwise: return true. is_attempting_to_close_browser_ = true; // Cases 1 and 4. bool need_beforeunload_fired = TabsNeedBeforeUnloadFired(); if (need_beforeunload_fired == is_calling_before_unload_handlers()) return !need_beforeunload_fired; // Cases 2 and 3. on_close_confirmed_.Reset(); ProcessPendingTabs(); return false; } bool UnloadController::CallBeforeUnloadHandlers( const base::Callback& on_close_confirmed) { // The devtools browser gets its beforeunload events as the results of // intercepting events from the inspected tab, so don't send them here as // well. if (browser_->is_devtools() || HasCompletedUnloadProcessing() || !TabsNeedBeforeUnloadFired()) return false; is_attempting_to_close_browser_ = true; on_close_confirmed_ = on_close_confirmed; ProcessPendingTabs(); return true; } void UnloadController::ResetBeforeUnloadHandlers() { if (!is_calling_before_unload_handlers()) return; CancelWindowClose(); } bool UnloadController::TabsNeedBeforeUnloadFired() { if (tabs_needing_before_unload_fired_.empty()) { for (int i = 0; i < browser_->tab_strip_model()->count(); ++i) { content::WebContents* contents = browser_->tab_strip_model()->GetWebContentsAt(i); bool should_fire_beforeunload = contents->NeedToFireBeforeUnload() || DevToolsWindow::NeedsToInterceptBeforeUnload(contents); if (!ContainsKey(tabs_needing_unload_fired_, contents) && should_fire_beforeunload) { tabs_needing_before_unload_fired_.insert(contents); } } } return !tabs_needing_before_unload_fired_.empty(); } void UnloadController::CancelWindowClose() { // Closing of window can be canceled from a beforeunload handler. DCHECK(is_attempting_to_close_browser_); tabs_needing_before_unload_fired_.clear(); for (UnloadListenerSet::iterator it = tabs_needing_unload_fired_.begin(); it != tabs_needing_unload_fired_.end(); ++it) { DevToolsWindow::OnPageCloseCanceled(*it); } tabs_needing_unload_fired_.clear(); if (is_calling_before_unload_handlers()) { base::Callback on_close_confirmed = on_close_confirmed_; on_close_confirmed_.Reset(); on_close_confirmed.Run(false); } is_attempting_to_close_browser_ = false; content::NotificationService::current()->Notify( chrome::NOTIFICATION_BROWSER_CLOSE_CANCELLED, content::Source(browser_), content::NotificationService::NoDetails()); } //////////////////////////////////////////////////////////////////////////////// // UnloadController, content::NotificationObserver implementation: void UnloadController::Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) { switch (type) { case content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED: if (is_attempting_to_close_browser_) { ClearUnloadState(content::Source(source).ptr(), false); // See comment for ClearUnloadState(). } break; default: NOTREACHED() << "Got a notification we didn't register for."; } } //////////////////////////////////////////////////////////////////////////////// // UnloadController, TabStripModelObserver implementation: void UnloadController::TabInsertedAt(content::WebContents* contents, int index, bool foreground) { TabAttachedImpl(contents); } void UnloadController::TabDetachedAt(content::WebContents* contents, int index) { TabDetachedImpl(contents); } void UnloadController::TabReplacedAt(TabStripModel* tab_strip_model, content::WebContents* old_contents, content::WebContents* new_contents, int index) { TabDetachedImpl(old_contents); TabAttachedImpl(new_contents); } void UnloadController::TabStripEmpty() { // Set is_attempting_to_close_browser_ here, so that extensions, etc, do not // attempt to add tabs to the browser before it closes. is_attempting_to_close_browser_ = true; } //////////////////////////////////////////////////////////////////////////////// // UnloadController, private: void UnloadController::TabAttachedImpl(content::WebContents* contents) { // If the tab crashes in the beforeunload or unload handler, it won't be // able to ack. But we know we can close it. registrar_.Add( this, content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, content::Source(contents)); } void UnloadController::TabDetachedImpl(content::WebContents* contents) { if (is_attempting_to_close_browser_) ClearUnloadState(contents, false); registrar_.Remove(this, content::NOTIFICATION_WEB_CONTENTS_DISCONNECTED, content::Source(contents)); } void UnloadController::ProcessPendingTabs() { if (!is_attempting_to_close_browser_) { // Because we might invoke this after a delay it's possible for the value of // is_attempting_to_close_browser_ to have changed since we scheduled the // task. return; } if (HasCompletedUnloadProcessing()) { // We've finished all the unload events and can proceed to close the // browser. browser_->OnWindowClosing(); return; } // Process beforeunload tabs first. When that queue is empty, process // unload tabs. if (!tabs_needing_before_unload_fired_.empty()) { content::WebContents* web_contents = *(tabs_needing_before_unload_fired_.begin()); // Null check render_view_host here as this gets called on a PostTask and // the tab's render_view_host may have been nulled out. if (web_contents->GetRenderViewHost()) { // If there's a devtools window attached to |web_contents|, // we would like devtools to call its own beforeunload handlers first, // and then call beforeunload handlers for |web_contents|. // See DevToolsWindow::InterceptPageBeforeUnload for details. if (!DevToolsWindow::InterceptPageBeforeUnload(web_contents)) web_contents->DispatchBeforeUnload(false); } else { ClearUnloadState(web_contents, true); } } else if (is_calling_before_unload_handlers()) { base::Callback on_close_confirmed = on_close_confirmed_; // Reset |on_close_confirmed_| in case the callback tests // |is_calling_before_unload_handlers()|, we want to return that calling // is complete. if (tabs_needing_unload_fired_.empty()) on_close_confirmed_.Reset(); on_close_confirmed.Run(true); } else if (!tabs_needing_unload_fired_.empty()) { // We've finished firing all beforeunload events and can proceed with unload // events. // TODO(ojan): We should add a call to browser_shutdown::OnShutdownStarting // somewhere around here so that we have accurate measurements of shutdown // time. // TODO(ojan): We can probably fire all the unload events in parallel and // get a perf benefit from that in the cases where the tab hangs in it's // unload handler or takes a long time to page in. content::WebContents* web_contents = *(tabs_needing_unload_fired_.begin()); // Null check render_view_host here as this gets called on a PostTask and // the tab's render_view_host may have been nulled out. if (web_contents->GetRenderViewHost()) { web_contents->ClosePage(); } else { ClearUnloadState(web_contents, true); } } else { NOTREACHED(); } } bool UnloadController::HasCompletedUnloadProcessing() const { return is_attempting_to_close_browser_ && tabs_needing_before_unload_fired_.empty() && tabs_needing_unload_fired_.empty(); } bool UnloadController::RemoveFromSet(UnloadListenerSet* set, content::WebContents* web_contents) { DCHECK(is_attempting_to_close_browser_); UnloadListenerSet::iterator iter = std::find(set->begin(), set->end(), web_contents); if (iter != set->end()) { set->erase(iter); return true; } return false; } void UnloadController::ClearUnloadState(content::WebContents* web_contents, bool process_now) { if (is_attempting_to_close_browser_) { RemoveFromSet(&tabs_needing_before_unload_fired_, web_contents); RemoveFromSet(&tabs_needing_unload_fired_, web_contents); if (process_now) { ProcessPendingTabs(); } else { base::ThreadTaskRunnerHandle::Get()->PostTask( FROM_HERE, base::Bind(&UnloadController::ProcessPendingTabs, weak_factory_.GetWeakPtr())); } } } } // namespace chrome