// 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/startup/autolaunch_prompt.h" #include "base/command_line.h" #include "base/message_loop.h" #include "base/utf_string_conversions.h" #include "chrome/browser/api/infobars/confirm_infobar_delegate.h" #include "chrome/browser/api/infobars/infobar_service.h" #include "chrome/browser/auto_launch_trial.h" #include "chrome/browser/first_run/first_run.h" #include "chrome/browser/prefs/pref_service.h" #include "chrome/browser/profiles/profile.h" #include "chrome/browser/ui/browser.h" #include "chrome/browser/ui/tabs/tab_strip_model.h" #include "chrome/common/chrome_constants.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/pref_names.h" #include "chrome/installer/util/auto_launch_util.h" #include "content/public/browser/browser_thread.h" #include "content/public/browser/navigation_details.h" #include "content/public/browser/web_contents.h" #include "grit/chromium_strings.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" using content::BrowserThread; namespace { const int kMaxInfobarShown = 5; // The delegate for the infobar shown when Chrome was auto-launched. class AutolaunchInfoBarDelegate : public ConfirmInfoBarDelegate { public: // Creates an autolaunch delegate and adds it to |infobar_service|. static void Create(InfoBarService* infobar_service, PrefService* prefs, Profile* profile); private: AutolaunchInfoBarDelegate(InfoBarService* infobar_service, PrefService* prefs, Profile* profile); virtual ~AutolaunchInfoBarDelegate(); void AllowExpiry() { should_expire_ = true; } // ConfirmInfoBarDelegate: virtual gfx::Image* GetIcon() const OVERRIDE; virtual string16 GetMessageText() const OVERRIDE; virtual string16 GetButtonLabel(InfoBarButton button) const OVERRIDE; virtual bool Accept() OVERRIDE; virtual bool Cancel() OVERRIDE; virtual bool ShouldExpireInternal( const content::LoadCommittedDetails& details) const OVERRIDE; // The prefs to use. PrefService* prefs_; // Whether the user clicked one of the buttons. bool action_taken_; // Whether the info-bar should be dismissed on the next navigation. bool should_expire_; // Weak pointer to the profile, not owned by us. Profile* profile_; // Used to delay the expiration of the info-bar. base::WeakPtrFactory weak_factory_; DISALLOW_COPY_AND_ASSIGN(AutolaunchInfoBarDelegate); }; // static void AutolaunchInfoBarDelegate::Create(InfoBarService* infobar_service, PrefService* prefs, Profile* profile) { infobar_service->AddInfoBar(scoped_ptr( new AutolaunchInfoBarDelegate(infobar_service, prefs, profile))); } AutolaunchInfoBarDelegate::AutolaunchInfoBarDelegate( InfoBarService* infobar_service, PrefService* prefs, Profile* profile) : ConfirmInfoBarDelegate(infobar_service), prefs_(prefs), action_taken_(false), should_expire_(false), profile_(profile), ALLOW_THIS_IN_INITIALIZER_LIST(weak_factory_(this)) { auto_launch_trial::UpdateInfobarShownMetric(); int count = prefs_->GetInteger(prefs::kShownAutoLaunchInfobar); prefs_->SetInteger(prefs::kShownAutoLaunchInfobar, count + 1); // We want the info-bar to stick-around for a few seconds and then be hidden // on the next navigation after that. MessageLoop::current()->PostDelayedTask( FROM_HERE, base::Bind(&AutolaunchInfoBarDelegate::AllowExpiry, weak_factory_.GetWeakPtr()), base::TimeDelta::FromSeconds(8)); } AutolaunchInfoBarDelegate::~AutolaunchInfoBarDelegate() { if (!action_taken_) { auto_launch_trial::UpdateInfobarResponseMetric( auto_launch_trial::INFOBAR_IGNORE); } } gfx::Image* AutolaunchInfoBarDelegate::GetIcon() const { return &ResourceBundle::GetSharedInstance().GetNativeImageNamed( IDR_PRODUCT_LOGO_32); } string16 AutolaunchInfoBarDelegate::GetMessageText() const { return l10n_util::GetStringUTF16(IDS_AUTO_LAUNCH_INFOBAR_TEXT); } string16 AutolaunchInfoBarDelegate::GetButtonLabel( InfoBarButton button) const { return l10n_util::GetStringUTF16((button == BUTTON_OK) ? IDS_AUTO_LAUNCH_OK : IDS_AUTO_LAUNCH_REVERT); } bool AutolaunchInfoBarDelegate::Accept() { action_taken_ = true; auto_launch_trial::UpdateInfobarResponseMetric( auto_launch_trial::INFOBAR_OK); return true; } bool AutolaunchInfoBarDelegate::Cancel() { action_taken_ = true; // Track infobar reponse. auto_launch_trial::UpdateInfobarResponseMetric( auto_launch_trial::INFOBAR_CUT_IT_OUT); // Also make sure we keep track of how many disable and how many enable. auto_launch_trial::UpdateToggleAutoLaunchMetric(false); content::BrowserThread::PostTask( content::BrowserThread::FILE, FROM_HERE, base::Bind(&auto_launch_util::DisableForegroundStartAtLogin, profile_->GetPath().BaseName().value())); return true; } bool AutolaunchInfoBarDelegate::ShouldExpireInternal( const content::LoadCommittedDetails& details) const { return should_expire_; } } // namespace namespace chrome { bool ShowAutolaunchPrompt(Browser* browser) { if (!auto_launch_trial::IsInAutoLaunchGroup()) return false; // Only supported on the main profile for now. Profile* profile = browser->profile(); if (profile->GetPath().BaseName() != FilePath(ASCIIToUTF16(chrome::kInitialProfile))) { return false; } int infobar_shown = profile->GetPrefs()->GetInteger(prefs::kShownAutoLaunchInfobar); if (infobar_shown >= kMaxInfobarShown) return false; const CommandLine& command_line = *CommandLine::ForCurrentProcess(); if (command_line.HasSwitch(switches::kChromeFrame)) return false; if (!command_line.HasSwitch(switches::kAutoLaunchAtStartup) && !first_run::IsChromeFirstRun()) { return false; } content::WebContents* web_contents = browser->tab_strip_model()->GetActiveWebContents(); profile = Profile::FromBrowserContext(web_contents->GetBrowserContext()); AutolaunchInfoBarDelegate::Create( InfoBarService::FromWebContents(web_contents), profile->GetPrefs(), profile); return true; } void RegisterAutolaunchUserPrefs(PrefServiceSyncable* prefs) { prefs->RegisterIntegerPref( prefs::kShownAutoLaunchInfobar, 0, PrefServiceSyncable::UNSYNCABLE_PREF); } } // namespace chrome