// Copyright 2013 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/lifetime/browser_close_manager.h" #include "chrome/browser/background/background_mode_manager.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/browser_shutdown.h" #include "chrome/browser/download/download_service.h" #include "chrome/browser/download/download_service_factory.h" #include "chrome/browser/profiles/profile_manager.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_iterator.h" #include "chrome/browser/ui/browser_list.h" #include "chrome/browser/ui/browser_window.h" #include "chrome/browser/ui/chrome_pages.h" #include "chrome/browser/ui/scoped_tabbed_browser_displayer.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "content/public/browser/web_contents.h" BrowserCloseManager::BrowserCloseManager() : current_browser_(NULL) {} BrowserCloseManager::~BrowserCloseManager() {} void BrowserCloseManager::StartClosingBrowsers() { // If the session is ending, skip straight to closing the browsers. There's no // time to wait for beforeunload dialogs. if (browser_shutdown::GetShutdownType() == browser_shutdown::END_SESSION) { // Tell everyone that we are shutting down. browser_shutdown::SetTryingToQuit(true); CloseBrowsers(); return; } TryToCloseBrowsers(); } void BrowserCloseManager::CancelBrowserClose() { browser_shutdown::SetTryingToQuit(false); for (chrome::BrowserIterator it; !it.done(); it.Next()) { it->ResetBeforeUnloadHandlers(); } } void BrowserCloseManager::TryToCloseBrowsers() { // If all browser windows can immediately be closed, fall out of this loop and // close the browsers. If any browser window cannot be closed, temporarily // stop closing. CallBeforeUnloadHandlers prompts the user and calls // OnBrowserReportCloseable with the result. If the user confirms the close, // this will trigger TryToCloseBrowsers to try again. for (chrome::BrowserIterator it; !it.done(); it.Next()) { if (it->CallBeforeUnloadHandlers( base::Bind(&BrowserCloseManager::OnBrowserReportCloseable, this))) { current_browser_ = *it; return; } } CheckForDownloadsInProgress(); } void BrowserCloseManager::OnBrowserReportCloseable(bool proceed) { if (!current_browser_) return; current_browser_ = NULL; if (proceed) TryToCloseBrowsers(); else CancelBrowserClose(); } void BrowserCloseManager::CheckForDownloadsInProgress() { int download_count = DownloadService::NonMaliciousDownloadCountAllProfiles(); if (download_count == 0) { CloseBrowsers(); return; } ConfirmCloseWithPendingDownloads( download_count, base::Bind(&BrowserCloseManager::OnReportDownloadsCancellable, this)); } void BrowserCloseManager::ConfirmCloseWithPendingDownloads( int download_count, const base::Callback& callback) { Browser* browser = BrowserList::GetInstance(chrome::GetActiveDesktop())->GetLastActive(); DCHECK(browser); browser->window()->ConfirmBrowserCloseWithPendingDownloads( download_count, Browser::DOWNLOAD_CLOSE_BROWSER_SHUTDOWN, true, callback); } void BrowserCloseManager::OnReportDownloadsCancellable(bool proceed) { if (proceed) { CloseBrowsers(); return; } CancelBrowserClose(); // Open the downloads page for each profile with downloads in progress. std::vector profiles( g_browser_process->profile_manager()->GetLoadedProfiles()); for (std::vector::iterator it = profiles.begin(); it != profiles.end(); ++it) { DownloadService* download_service = DownloadServiceFactory::GetForBrowserContext(*it); if (download_service->NonMaliciousDownloadCount() > 0) { chrome::ScopedTabbedBrowserDisplayer displayer( *it, chrome::GetActiveDesktop()); chrome::ShowDownloads(displayer.browser()); } } } void BrowserCloseManager::CloseBrowsers() { #if defined(ENABLE_SESSION_SERVICE) // Before we close the browsers shutdown all session services. That way an // exit can restore all browsers open before exiting. ProfileManager::ShutdownSessionServices(); #endif if (!browser_shutdown::IsTryingToQuit()) { BackgroundModeManager* background_mode_manager = g_browser_process->background_mode_manager(); if (background_mode_manager) background_mode_manager->SuspendBackgroundMode(); } bool session_ending = browser_shutdown::GetShutdownType() == browser_shutdown::END_SESSION; for (scoped_ptr it_ptr( new chrome::BrowserIterator()); !it_ptr->done();) { Browser* browser = **it_ptr; browser->window()->Close(); if (!session_ending) { it_ptr->Next(); } else { // This path is hit during logoff/power-down. In this case we won't get // a final message and so we force the browser to be deleted. // Close doesn't immediately destroy the browser // (Browser::TabStripEmpty() uses invoke later) but when we're ending the // session we need to make sure the browser is destroyed now. So, invoke // DestroyBrowser to make sure the browser is deleted and cleanup can // happen. while (browser->tab_strip_model()->count()) delete browser->tab_strip_model()->GetWebContentsAt(0); browser->window()->DestroyBrowser(); it_ptr.reset(new chrome::BrowserIterator()); if (!it_ptr->done() && browser == **it_ptr) { // Destroying the browser should have removed it from the browser list. // We should never get here. NOTREACHED(); return; } } } }