// 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/plugins/plugin_infobar_delegates.h" #include "base/path_service.h" #include "base/strings/utf_string_conversions.h" #include "chrome/browser/content_settings/host_content_settings_map.h" #include "chrome/browser/google/google_util.h" #include "chrome/browser/infobars/infobar.h" #include "chrome/browser/infobars/infobar_service.h" #include "chrome/browser/lifetime/application_lifetime.h" #include "chrome/browser/plugins/chrome_plugin_service_filter.h" #include "chrome/browser/plugins/plugin_metadata.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/shell_integration.h" #include "chrome/browser/ui/browser_commands.h" #include "chrome/common/render_messages.h" #include "chrome/common/url_constants.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/render_process_host.h" #include "content/public/browser/render_view_host.h" #include "content/public/browser/user_metrics.h" #include "content/public/browser/web_contents.h" #include "grit/generated_resources.h" #include "grit/locale_settings.h" #include "grit/theme_resources.h" #include "ui/base/l10n/l10n_util.h" #if defined(ENABLE_PLUGIN_INSTALLATION) #if defined(OS_WIN) #include "base/win/metro.h" #endif #include "chrome/browser/plugins/plugin_installer.h" #endif #if defined(OS_WIN) #include <shellapi.h> #include "ui/base/win/shell.h" #if defined(USE_AURA) #include "ui/aura/remote_root_window_host_win.h" #endif #endif using content::UserMetricsAction; // PluginInfoBarDelegate ------------------------------------------------------ PluginInfoBarDelegate::PluginInfoBarDelegate(const std::string& identifier) : ConfirmInfoBarDelegate(), identifier_(identifier) { } PluginInfoBarDelegate::~PluginInfoBarDelegate() { } bool PluginInfoBarDelegate::LinkClicked(WindowOpenDisposition disposition) { web_contents()->OpenURL(content::OpenURLParams( GURL(GetLearnMoreURL()), content::Referrer(), (disposition == CURRENT_TAB) ? NEW_FOREGROUND_TAB : disposition, content::PAGE_TRANSITION_LINK, false)); return false; } void PluginInfoBarDelegate::LoadBlockedPlugins() { content::RenderViewHost* host = web_contents()->GetRenderViewHost(); ChromePluginServiceFilter::GetInstance()->AuthorizeAllPlugins( host->GetProcess()->GetID()); host->Send(new ChromeViewMsg_LoadBlockedPlugins( host->GetRoutingID(), identifier_)); } int PluginInfoBarDelegate::GetIconID() const { return IDR_INFOBAR_PLUGIN_INSTALL; } base::string16 PluginInfoBarDelegate::GetLinkText() const { return l10n_util::GetStringUTF16(IDS_LEARN_MORE); } // UnauthorizedPluginInfoBarDelegate ------------------------------------------ // static void UnauthorizedPluginInfoBarDelegate::Create( InfoBarService* infobar_service, HostContentSettingsMap* content_settings, const base::string16& name, const std::string& identifier) { infobar_service->AddInfoBar(ConfirmInfoBarDelegate::CreateInfoBar( scoped_ptr<ConfirmInfoBarDelegate>(new UnauthorizedPluginInfoBarDelegate( content_settings, name, identifier)))); content::RecordAction(UserMetricsAction("BlockedPluginInfobar.Shown")); std::string utf8_name(UTF16ToUTF8(name)); if (utf8_name == PluginMetadata::kJavaGroupName) { content::RecordAction(UserMetricsAction("BlockedPluginInfobar.Shown.Java")); } else if (utf8_name == PluginMetadata::kQuickTimeGroupName) { content::RecordAction( UserMetricsAction("BlockedPluginInfobar.Shown.QuickTime")); } else if (utf8_name == PluginMetadata::kShockwaveGroupName) { content::RecordAction( UserMetricsAction("BlockedPluginInfobar.Shown.Shockwave")); } else if (utf8_name == PluginMetadata::kRealPlayerGroupName) { content::RecordAction( UserMetricsAction("BlockedPluginInfobar.Shown.RealPlayer")); } else if (utf8_name == PluginMetadata::kWindowsMediaPlayerGroupName) { content::RecordAction( UserMetricsAction("BlockedPluginInfobar.Shown.WindowsMediaPlayer")); } } UnauthorizedPluginInfoBarDelegate::UnauthorizedPluginInfoBarDelegate( HostContentSettingsMap* content_settings, const base::string16& name, const std::string& identifier) : PluginInfoBarDelegate(identifier), content_settings_(content_settings), name_(name) { } UnauthorizedPluginInfoBarDelegate::~UnauthorizedPluginInfoBarDelegate() { content::RecordAction(UserMetricsAction("BlockedPluginInfobar.Closed")); } std::string UnauthorizedPluginInfoBarDelegate::GetLearnMoreURL() const { return chrome::kBlockedPluginLearnMoreURL; } base::string16 UnauthorizedPluginInfoBarDelegate::GetMessageText() const { return l10n_util::GetStringFUTF16(IDS_PLUGIN_NOT_AUTHORIZED, name_); } base::string16 UnauthorizedPluginInfoBarDelegate::GetButtonLabel( InfoBarButton button) const { return l10n_util::GetStringUTF16((button == BUTTON_OK) ? IDS_PLUGIN_ENABLE_TEMPORARILY : IDS_PLUGIN_ENABLE_ALWAYS); } bool UnauthorizedPluginInfoBarDelegate::Accept() { content::RecordAction( UserMetricsAction("BlockedPluginInfobar.AllowThisTime")); LoadBlockedPlugins(); return true; } bool UnauthorizedPluginInfoBarDelegate::Cancel() { content::RecordAction(UserMetricsAction("BlockedPluginInfobar.AlwaysAllow")); const GURL& url = web_contents()->GetURL(); content_settings_->AddExceptionForURL(url, url, CONTENT_SETTINGS_TYPE_PLUGINS, CONTENT_SETTING_ALLOW); LoadBlockedPlugins(); return true; } void UnauthorizedPluginInfoBarDelegate::InfoBarDismissed() { content::RecordAction(UserMetricsAction("BlockedPluginInfobar.Dismissed")); } bool UnauthorizedPluginInfoBarDelegate::LinkClicked( WindowOpenDisposition disposition) { content::RecordAction(UserMetricsAction("BlockedPluginInfobar.LearnMore")); return PluginInfoBarDelegate::LinkClicked(disposition); } #if defined(ENABLE_PLUGIN_INSTALLATION) // OutdatedPluginInfoBarDelegate ---------------------------------------------- void OutdatedPluginInfoBarDelegate::Create( InfoBarService* infobar_service, PluginInstaller* installer, scoped_ptr<PluginMetadata> plugin_metadata) { // Copy the name out of |plugin_metadata| now, since the Pass() call below // will make it impossible to get at. base::string16 name(plugin_metadata->name()); infobar_service->AddInfoBar(ConfirmInfoBarDelegate::CreateInfoBar( scoped_ptr<ConfirmInfoBarDelegate>(new OutdatedPluginInfoBarDelegate( installer, plugin_metadata.Pass(), l10n_util::GetStringFUTF16( (installer->state() == PluginInstaller::INSTALLER_STATE_IDLE) ? IDS_PLUGIN_OUTDATED_PROMPT : IDS_PLUGIN_DOWNLOADING, name))))); } OutdatedPluginInfoBarDelegate::OutdatedPluginInfoBarDelegate( PluginInstaller* installer, scoped_ptr<PluginMetadata> plugin_metadata, const base::string16& message) : PluginInfoBarDelegate(plugin_metadata->identifier()), WeakPluginInstallerObserver(installer), plugin_metadata_(plugin_metadata.Pass()), message_(message) { content::RecordAction(UserMetricsAction("OutdatedPluginInfobar.Shown")); std::string name = UTF16ToUTF8(plugin_metadata_->name()); if (name == PluginMetadata::kJavaGroupName) { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.Shown.Java")); } else if (name == PluginMetadata::kQuickTimeGroupName) { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.Shown.QuickTime")); } else if (name == PluginMetadata::kShockwaveGroupName) { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.Shown.Shockwave")); } else if (name == PluginMetadata::kRealPlayerGroupName) { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.Shown.RealPlayer")); } else if (name == PluginMetadata::kSilverlightGroupName) { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.Shown.Silverlight")); } else if (name == PluginMetadata::kAdobeReaderGroupName) { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.Shown.Reader")); } } OutdatedPluginInfoBarDelegate::~OutdatedPluginInfoBarDelegate() { content::RecordAction(UserMetricsAction("OutdatedPluginInfobar.Closed")); } std::string OutdatedPluginInfoBarDelegate::GetLearnMoreURL() const { return chrome::kOutdatedPluginLearnMoreURL; } base::string16 OutdatedPluginInfoBarDelegate::GetMessageText() const { return message_; } base::string16 OutdatedPluginInfoBarDelegate::GetButtonLabel( InfoBarButton button) const { return l10n_util::GetStringUTF16((button == BUTTON_OK) ? IDS_PLUGIN_UPDATE : IDS_PLUGIN_ENABLE_TEMPORARILY); } bool OutdatedPluginInfoBarDelegate::Accept() { DCHECK_EQ(PluginInstaller::INSTALLER_STATE_IDLE, installer()->state()); content::RecordAction(UserMetricsAction("OutdatedPluginInfobar.Update")); // A call to any of |OpenDownloadURL()| or |StartInstalling()| will // result in deleting ourselves. Accordingly, we make sure to // not pass a reference to an object that can go away. GURL plugin_url(plugin_metadata_->plugin_url()); if (plugin_metadata_->url_for_display()) installer()->OpenDownloadURL(plugin_url, web_contents()); else installer()->StartInstalling(plugin_url, web_contents()); return false; } bool OutdatedPluginInfoBarDelegate::Cancel() { content::RecordAction( UserMetricsAction("OutdatedPluginInfobar.AllowThisTime")); LoadBlockedPlugins(); return true; } void OutdatedPluginInfoBarDelegate::InfoBarDismissed() { content::RecordAction(UserMetricsAction("OutdatedPluginInfobar.Dismissed")); } bool OutdatedPluginInfoBarDelegate::LinkClicked( WindowOpenDisposition disposition) { content::RecordAction(UserMetricsAction("OutdatedPluginInfobar.LearnMore")); return PluginInfoBarDelegate::LinkClicked(disposition); } void OutdatedPluginInfoBarDelegate::DownloadStarted() { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_DOWNLOADING, plugin_metadata_->name())); } void OutdatedPluginInfoBarDelegate::DownloadError(const std::string& message) { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_DOWNLOAD_ERROR_SHORT, plugin_metadata_->name())); } void OutdatedPluginInfoBarDelegate::DownloadCancelled() { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_DOWNLOAD_CANCELLED, plugin_metadata_->name())); } void OutdatedPluginInfoBarDelegate::DownloadFinished() { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_UPDATING, plugin_metadata_->name())); } void OutdatedPluginInfoBarDelegate::OnlyWeakObserversLeft() { infobar()->RemoveSelf(); } void OutdatedPluginInfoBarDelegate::ReplaceWithInfoBar( const base::string16& message) { // Return early if the message doesn't change. This is important in case the // PluginInstaller is still iterating over its observers (otherwise we would // keep replacing infobar delegates infinitely). if ((message_ == message) || !infobar()->owner()) return; PluginInstallerInfoBarDelegate::Replace( infobar(), installer(), plugin_metadata_->Clone(), false, message); } // PluginInstallerInfoBarDelegate --------------------------------------------- void PluginInstallerInfoBarDelegate::Create( InfoBarService* infobar_service, PluginInstaller* installer, scoped_ptr<PluginMetadata> plugin_metadata, const InstallCallback& callback) { base::string16 name(plugin_metadata->name()); #if defined(OS_WIN) if (base::win::IsMetroProcess()) { PluginMetroModeInfoBarDelegate::Create( infobar_service, PluginMetroModeInfoBarDelegate::MISSING_PLUGIN, name); return; } #endif infobar_service->AddInfoBar(ConfirmInfoBarDelegate::CreateInfoBar( scoped_ptr<ConfirmInfoBarDelegate>(new PluginInstallerInfoBarDelegate( installer, plugin_metadata.Pass(), callback, true, l10n_util::GetStringFUTF16( (installer->state() == PluginInstaller::INSTALLER_STATE_IDLE) ? IDS_PLUGININSTALLER_INSTALLPLUGIN_PROMPT : IDS_PLUGIN_DOWNLOADING, name))))); } void PluginInstallerInfoBarDelegate::Replace( InfoBar* infobar, PluginInstaller* installer, scoped_ptr<PluginMetadata> plugin_metadata, bool new_install, const base::string16& message) { DCHECK(infobar->owner()); infobar->owner()->ReplaceInfoBar(infobar, ConfirmInfoBarDelegate::CreateInfoBar(scoped_ptr<ConfirmInfoBarDelegate>( new PluginInstallerInfoBarDelegate( installer, plugin_metadata.Pass(), PluginInstallerInfoBarDelegate::InstallCallback(), new_install, message)))); } PluginInstallerInfoBarDelegate::PluginInstallerInfoBarDelegate( PluginInstaller* installer, scoped_ptr<PluginMetadata> plugin_metadata, const InstallCallback& callback, bool new_install, const base::string16& message) : ConfirmInfoBarDelegate(), WeakPluginInstallerObserver(installer), plugin_metadata_(plugin_metadata.Pass()), callback_(callback), new_install_(new_install), message_(message) { } PluginInstallerInfoBarDelegate::~PluginInstallerInfoBarDelegate() { } int PluginInstallerInfoBarDelegate::GetIconID() const { return IDR_INFOBAR_PLUGIN_INSTALL; } base::string16 PluginInstallerInfoBarDelegate::GetMessageText() const { return message_; } int PluginInstallerInfoBarDelegate::GetButtons() const { return callback_.is_null() ? BUTTON_NONE : BUTTON_OK; } base::string16 PluginInstallerInfoBarDelegate::GetButtonLabel( InfoBarButton button) const { DCHECK_EQ(BUTTON_OK, button); return l10n_util::GetStringUTF16(IDS_PLUGININSTALLER_INSTALLPLUGIN_BUTTON); } bool PluginInstallerInfoBarDelegate::Accept() { callback_.Run(plugin_metadata_.get()); return false; } base::string16 PluginInstallerInfoBarDelegate::GetLinkText() const { return l10n_util::GetStringUTF16(new_install_ ? IDS_PLUGININSTALLER_PROBLEMSINSTALLING : IDS_PLUGININSTALLER_PROBLEMSUPDATING); } bool PluginInstallerInfoBarDelegate::LinkClicked( WindowOpenDisposition disposition) { GURL url(plugin_metadata_->help_url()); if (url.is_empty()) { url = google_util::AppendGoogleLocaleParam(GURL( "https://www.google.com/support/chrome/bin/answer.py?answer=142064")); } web_contents()->OpenURL(content::OpenURLParams( url, content::Referrer(), (disposition == CURRENT_TAB) ? NEW_FOREGROUND_TAB : disposition, content::PAGE_TRANSITION_LINK, false)); return false; } void PluginInstallerInfoBarDelegate::DownloadStarted() { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_DOWNLOADING, plugin_metadata_->name())); } void PluginInstallerInfoBarDelegate::DownloadCancelled() { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_DOWNLOAD_CANCELLED, plugin_metadata_->name())); } void PluginInstallerInfoBarDelegate::DownloadError(const std::string& message) { ReplaceWithInfoBar(l10n_util::GetStringFUTF16(IDS_PLUGIN_DOWNLOAD_ERROR_SHORT, plugin_metadata_->name())); } void PluginInstallerInfoBarDelegate::DownloadFinished() { ReplaceWithInfoBar( l10n_util::GetStringFUTF16( new_install_ ? IDS_PLUGIN_INSTALLING : IDS_PLUGIN_UPDATING, plugin_metadata_->name())); } void PluginInstallerInfoBarDelegate::OnlyWeakObserversLeft() { infobar()->RemoveSelf(); } void PluginInstallerInfoBarDelegate::ReplaceWithInfoBar( const base::string16& message) { // Return early if the message doesn't change. This is important in case the // PluginInstaller is still iterating over its observers (otherwise we would // keep replacing infobar delegates infinitely). if ((message_ == message) || !infobar()->owner()) return; Replace(infobar(), installer(), plugin_metadata_->Clone(), new_install_, message); } #if defined(OS_WIN) // PluginMetroModeInfoBarDelegate --------------------------------------------- // static void PluginMetroModeInfoBarDelegate::Create( InfoBarService* infobar_service, PluginMetroModeInfoBarDelegate::Mode mode, const base::string16& name) { infobar_service->AddInfoBar(ConfirmInfoBarDelegate::CreateInfoBar( scoped_ptr<ConfirmInfoBarDelegate>( new PluginMetroModeInfoBarDelegate(mode, name)))); } PluginMetroModeInfoBarDelegate::PluginMetroModeInfoBarDelegate( PluginMetroModeInfoBarDelegate::Mode mode, const base::string16& name) : ConfirmInfoBarDelegate(), mode_(mode), name_(name) { } PluginMetroModeInfoBarDelegate::~PluginMetroModeInfoBarDelegate() { } int PluginMetroModeInfoBarDelegate::GetIconID() const { return IDR_INFOBAR_PLUGIN_INSTALL; } base::string16 PluginMetroModeInfoBarDelegate::GetMessageText() const { return l10n_util::GetStringFUTF16((mode_ == MISSING_PLUGIN) ? IDS_METRO_MISSING_PLUGIN_PROMPT : IDS_METRO_NPAPI_PLUGIN_PROMPT, name_); } int PluginMetroModeInfoBarDelegate::GetButtons() const { return BUTTON_OK; } base::string16 PluginMetroModeInfoBarDelegate::GetButtonLabel( InfoBarButton button) const { #if defined(USE_AURA) && defined(USE_ASH) return l10n_util::GetStringUTF16(IDS_WIN8_DESKTOP_RESTART); #else return l10n_util::GetStringUTF16((mode_ == MISSING_PLUGIN) ? IDS_WIN8_DESKTOP_RESTART : IDS_WIN8_DESKTOP_OPEN); #endif } #if defined(USE_AURA) && defined(USE_ASH) void LaunchDesktopInstanceHelper(const base::string16& url) { base::FilePath exe_path; if (!PathService::Get(base::FILE_EXE, &exe_path)) return; base::FilePath shortcut_path( ShellIntegration::GetStartMenuShortcut(exe_path)); // Actually launching the process needs to happen in the metro viewer, // otherwise it won't automatically transition to desktop. So we have // to send an IPC to the viewer to do the ShellExecute. aura::RemoteRootWindowHostWin::Instance()->HandleOpenURLOnDesktop( shortcut_path, url); } #endif bool PluginMetroModeInfoBarDelegate::Accept() { chrome::AttemptRestartToDesktopMode(); return true; } base::string16 PluginMetroModeInfoBarDelegate::GetLinkText() const { return l10n_util::GetStringUTF16(IDS_LEARN_MORE); } bool PluginMetroModeInfoBarDelegate::LinkClicked( WindowOpenDisposition disposition) { // TODO(shrikant): We may need to change language a little at following // support URLs. With new approach we will just restart for both missing // and not missing mode. web_contents()->OpenURL(content::OpenURLParams( GURL((mode_ == MISSING_PLUGIN) ? "https://support.google.com/chrome/?p=ib_display_in_desktop" : "https://support.google.com/chrome/?p=ib_redirect_to_desktop"), content::Referrer(), (disposition == CURRENT_TAB) ? NEW_FOREGROUND_TAB : disposition, content::PAGE_TRANSITION_LINK, false)); return false; } #endif // defined(OS_WIN) #endif // defined(ENABLE_PLUGIN_INSTALLATION)