// 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/extensions/extension_install_ui_default.h" #include "base/bind.h" #include "base/strings/utf_string_conversions.h" #include "chrome/browser/chrome_notification_types.h" #include "chrome/browser/extensions/extension_install_prompt.h" #include "chrome/browser/extensions/theme_installed_infobar_delegate.h" #include "chrome/browser/infobars/infobar_service.h" #include "chrome/browser/prefs/incognito_mode_prefs.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/search/search.h" #include "chrome/browser/themes/theme_service.h" #include "chrome/browser/themes/theme_service_factory.h" #include "chrome/browser/ui/app_list/app_list_service.h" #include "chrome/browser/ui/app_list/app_list_util.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/browser_dialogs.h" #include "chrome/browser/ui/browser_finder.h" #include "chrome/browser/ui/browser_navigator.h" #include "chrome/browser/ui/browser_tabstrip.h" #include "chrome/browser/ui/browser_window.h" #include "chrome/browser/ui/host_desktop.h" #include "chrome/browser/ui/scoped_tabbed_browser_displayer.h" #include "chrome/browser/ui/simple_message_box.h" #include "chrome/browser/ui/singleton_tabs.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "chrome/common/url_constants.h" #include "components/infobars/core/confirm_infobar_delegate.h" #include "components/infobars/core/infobar.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/notification_service.h" #include "content/public/browser/web_contents.h" #include "extensions/common/extension.h" #include "grit/generated_resources.h" #include "grit/theme_resources.h" #include "ui/base/l10n/l10n_util.h" #include "ui/base/resource/resource_bundle.h" #if defined(USE_ASH) #include "ash/shell.h" #endif using content::BrowserThread; using content::WebContents; using extensions::Extension; namespace { // Helpers -------------------------------------------------------------------- Browser* FindOrCreateVisibleBrowser(Profile* profile) { // TODO(mpcomplete): remove this workaround for http://crbug.com/244246 // after fixing http://crbug.com/38676. if (!IncognitoModePrefs::CanOpenBrowser(profile)) return NULL; chrome::ScopedTabbedBrowserDisplayer displayer( profile, chrome::GetActiveDesktop()); Browser* browser = displayer.browser(); if (browser->tab_strip_model()->count() == 0) chrome::AddTabAt(browser, GURL(), -1, true); return browser; } void ShowExtensionInstalledBubble(const extensions::Extension* extension, Profile* profile, const SkBitmap& icon) { Browser* browser = FindOrCreateVisibleBrowser(profile); if (browser) chrome::ShowExtensionInstalledBubble(extension, browser, icon); } // ErrorInfoBarDelegate ------------------------------------------------------- // Helper class to put up an infobar when installation fails. class ErrorInfoBarDelegate : public ConfirmInfoBarDelegate { public: // Creates an error infobar and delegate and adds the infobar to // |infobar_service|. static void Create(InfoBarService* infobar_service, const extensions::CrxInstallerError& error); private: explicit ErrorInfoBarDelegate(const extensions::CrxInstallerError& error); virtual ~ErrorInfoBarDelegate(); // ConfirmInfoBarDelegate: virtual base::string16 GetMessageText() const OVERRIDE; virtual int GetButtons() const OVERRIDE; virtual base::string16 GetLinkText() const OVERRIDE; virtual bool LinkClicked(WindowOpenDisposition disposition) OVERRIDE; extensions::CrxInstallerError error_; DISALLOW_COPY_AND_ASSIGN(ErrorInfoBarDelegate); }; // static void ErrorInfoBarDelegate::Create(InfoBarService* infobar_service, const extensions::CrxInstallerError& error) { infobar_service->AddInfoBar(ConfirmInfoBarDelegate::CreateInfoBar( scoped_ptr(new ErrorInfoBarDelegate(error)))); } ErrorInfoBarDelegate::ErrorInfoBarDelegate( const extensions::CrxInstallerError& error) : ConfirmInfoBarDelegate(), error_(error) { } ErrorInfoBarDelegate::~ErrorInfoBarDelegate() { } base::string16 ErrorInfoBarDelegate::GetMessageText() const { return error_.message(); } int ErrorInfoBarDelegate::GetButtons() const { return BUTTON_OK; } base::string16 ErrorInfoBarDelegate::GetLinkText() const { return (error_.type() == extensions::CrxInstallerError::ERROR_OFF_STORE) ? l10n_util::GetStringUTF16(IDS_LEARN_MORE) : base::string16(); } bool ErrorInfoBarDelegate::LinkClicked(WindowOpenDisposition disposition) { InfoBarService::WebContentsFromInfoBar(infobar())->OpenURL( content::OpenURLParams( GURL("http://support.google.com/chrome_webstore/?p=crx_warning"), content::Referrer(), (disposition == CURRENT_TAB) ? NEW_FOREGROUND_TAB : disposition, content::PAGE_TRANSITION_LINK, false)); return false; } } // namespace // ExtensionInstallUI --------------------------------------------------------- // static ExtensionInstallUI* ExtensionInstallUI::Create(Profile* profile) { return new ExtensionInstallUIDefault(profile); } // static void ExtensionInstallUI::OpenAppInstalledUI(Profile* profile, const std::string& app_id) { #if defined(OS_CHROMEOS) AppListService::Get(chrome::HOST_DESKTOP_TYPE_ASH)-> ShowForProfile(profile); content::NotificationService::current()->Notify( chrome::NOTIFICATION_APP_INSTALLED_TO_APPLIST, content::Source(profile), content::Details(&app_id)); #else Browser* browser = FindOrCreateVisibleBrowser(profile); if (browser) { GURL url(chrome::IsInstantExtendedAPIEnabled() ? chrome::kChromeUIAppsURL : chrome::kChromeUINewTabURL); chrome::NavigateParams params( chrome::GetSingletonTabNavigateParams(browser, url)); chrome::Navigate(¶ms); content::NotificationService::current()->Notify( chrome::NOTIFICATION_APP_INSTALLED_TO_NTP, content::Source(params.target_contents), content::Details(&app_id)); } #endif } // static ExtensionInstallPrompt* ExtensionInstallUI::CreateInstallPromptWithBrowser( Browser* browser) { content::WebContents* web_contents = NULL; if (browser) web_contents = browser->tab_strip_model()->GetActiveWebContents(); return new ExtensionInstallPrompt(web_contents); } // static ExtensionInstallPrompt* ExtensionInstallUI::CreateInstallPromptWithProfile( Profile* profile) { Browser* browser = chrome::FindLastActiveWithProfile(profile, chrome::GetActiveDesktop()); if (browser) return CreateInstallPromptWithBrowser(browser); // No browser window is open yet. Create a free-standing dialog associated // with |profile|. return new ExtensionInstallPrompt(profile, NULL, NULL); } // ExtensionInstallUIDefault -------------------------------------------------- ExtensionInstallUIDefault::ExtensionInstallUIDefault(Profile* profile) : ExtensionInstallUI(profile), previous_using_system_theme_(false), use_app_installed_bubble_(false) { // |profile| can be NULL during tests. if (profile) { // Remember the current theme in case the user presses undo. const Extension* previous_theme = ThemeServiceFactory::GetThemeForProfile(profile); if (previous_theme) previous_theme_id_ = previous_theme->id(); previous_using_system_theme_ = ThemeServiceFactory::GetForProfile(profile)->UsingSystemTheme(); } } ExtensionInstallUIDefault::~ExtensionInstallUIDefault() {} void ExtensionInstallUIDefault::OnInstallSuccess(const Extension* extension, SkBitmap* icon) { if (skip_post_install_ui()) return; if (!profile()) { // TODO(zelidrag): Figure out what exact conditions cause crash // http://crbug.com/159437 and write browser test to cover it. NOTREACHED(); return; } if (extension->is_theme()) { ThemeInstalledInfoBarDelegate::Create( extension, profile(), previous_theme_id_, previous_using_system_theme_); return; } // Extensions aren't enabled by default in incognito so we confirm // the install in a normal window. Profile* current_profile = profile()->GetOriginalProfile(); if (extension->is_app()) { bool use_bubble = false; #if defined(TOOLKIT_VIEWS) || defined(OS_MACOSX) use_bubble = use_app_installed_bubble_; #endif if (IsAppLauncherEnabled()) { // TODO(tapted): ExtensionInstallUI should retain the desktop type from // the browser used to initiate the flow. http://crbug.com/308360. AppListService::Get(chrome::GetActiveDesktop())-> ShowForProfile(current_profile); content::NotificationService::current()->Notify( chrome::NOTIFICATION_APP_INSTALLED_TO_APPLIST, content::Source(current_profile), content::Details(&extension->id())); return; } if (use_bubble) { ShowExtensionInstalledBubble(extension, current_profile, *icon); return; } ExtensionInstallUI::OpenAppInstalledUI(current_profile, extension->id()); return; } ShowExtensionInstalledBubble(extension, current_profile, *icon); } void ExtensionInstallUIDefault::OnInstallFailure( const extensions::CrxInstallerError& error) { DCHECK_CURRENTLY_ON(BrowserThread::UI); if (disable_failure_ui_for_tests() || skip_post_install_ui()) return; Browser* browser = chrome::FindLastActiveWithProfile(profile(), chrome::GetActiveDesktop()); if (!browser) // Can be NULL in unittests. return; WebContents* web_contents = browser->tab_strip_model()->GetActiveWebContents(); if (!web_contents) return; ErrorInfoBarDelegate::Create(InfoBarService::FromWebContents(web_contents), error); } void ExtensionInstallUIDefault::SetUseAppInstalledBubble(bool use_bubble) { use_app_installed_bubble_ = use_bubble; }