signin_promo.cc revision a3f6a49ab37290eeeb8db0f41ec0f1cb74a68be7
1// Copyright 2013 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "chrome/browser/signin/signin_promo.h"
6
7#include "base/command_line.h"
8#include "base/prefs/pref_service.h"
9#include "base/strings/string_number_conversions.h"
10#include "base/strings/string_util.h"
11#include "base/strings/stringprintf.h"
12#include "base/strings/utf_string_conversions.h"
13#include "chrome/browser/browser_process.h"
14#include "chrome/browser/first_run/first_run.h"
15#include "chrome/browser/google/google_util.h"
16#include "chrome/browser/profiles/profile.h"
17#include "chrome/browser/profiles/profile_info_cache.h"
18#include "chrome/browser/profiles/profile_manager.h"
19#include "chrome/browser/signin/signin_manager.h"
20#include "chrome/browser/signin/signin_manager_factory.h"
21#include "chrome/browser/sync/profile_sync_service.h"
22#include "chrome/browser/sync/profile_sync_service_factory.h"
23#include "chrome/browser/ui/webui/options/core_options_handler.h"
24#include "chrome/browser/ui/webui/theme_source.h"
25#include "chrome/common/chrome_switches.h"
26#include "chrome/common/net/url_util.h"
27#include "chrome/common/pref_names.h"
28#include "chrome/common/url_constants.h"
29#include "components/user_prefs/pref_registry_syncable.h"
30#include "content/public/browser/url_data_source.h"
31#include "content/public/browser/web_contents.h"
32#include "content/public/browser/web_ui.h"
33#include "content/public/browser/web_ui_data_source.h"
34#include "google_apis/gaia/gaia_urls.h"
35#include "grit/browser_resources.h"
36#include "grit/generated_resources.h"
37#include "grit/theme_resources.h"
38#include "net/base/escape.h"
39#include "net/base/network_change_notifier.h"
40#include "net/base/url_util.h"
41#include "ui/base/l10n/l10n_util.h"
42
43using content::WebContents;
44
45namespace {
46
47const char kSignInPromoQueryKeyAutoClose[] = "auto_close";
48const char kSignInPromoQueryKeyContinue[] = "continue";
49const char kSignInPromoQueryKeySource[] = "source";
50const char kSignInPromoQueryKeyConstrained[] = "constrained";
51
52// Gaia cannot support about:blank as a continue URL, so using a hosted blank
53// page instead.
54const char kSignInLandingUrlPrefix[] =
55    "https://www.google.com/intl/%s/chrome/blank.html";
56
57// The maximum number of times we want to show the sign in promo at startup.
58const int kSignInPromoShowAtStartupMaximum = 10;
59
60// Forces the web based signin flow when set.
61bool g_force_web_based_signin_flow = false;
62
63// Checks we want to show the sign in promo for the given brand.
64bool AllowPromoAtStartupForCurrentBrand() {
65  std::string brand;
66  google_util::GetBrand(&brand);
67
68  if (brand.empty())
69    return true;
70
71  if (google_util::IsInternetCafeBrandCode(brand))
72    return false;
73
74  // Enable for both organic and distribution.
75  return true;
76}
77
78// Returns true if a user has seen the sign in promo at startup previously.
79bool HasShownPromoAtStartup(Profile* profile) {
80  return profile->GetPrefs()->HasPrefPath(prefs::kSignInPromoStartupCount);
81}
82
83// Returns true if the user has previously skipped the sign in promo.
84bool HasUserSkippedPromo(Profile* profile) {
85  return profile->GetPrefs()->GetBoolean(prefs::kSignInPromoUserSkipped);
86}
87
88}  // namespace
89
90namespace signin {
91
92bool ShouldShowPromo(Profile* profile) {
93#if defined(OS_CHROMEOS)
94  // There's no need to show the sign in promo on cros since cros users are
95  // already logged in.
96  return false;
97#else
98
99  // Don't bother if we don't have any kind of network connection.
100  if (net::NetworkChangeNotifier::IsOffline())
101    return false;
102
103  // Don't show for managed profiles.
104  if (profile->IsManaged())
105    return false;
106
107  // Display the signin promo if the user is not signed in.
108  SigninManager* signin = SigninManagerFactory::GetForProfile(
109      profile->GetOriginalProfile());
110  return !signin->AuthInProgress() && signin->IsSigninAllowed() &&
111      signin->GetAuthenticatedUsername().empty();
112#endif
113}
114
115bool ShouldShowPromoAtStartup(Profile* profile, bool is_new_profile) {
116  DCHECK(profile);
117
118  // Don't show if the profile is an incognito.
119  if (profile->IsOffTheRecord())
120    return false;
121
122  if (!ShouldShowPromo(profile))
123    return false;
124
125  if (!is_new_profile) {
126    if (!HasShownPromoAtStartup(profile))
127      return false;
128  }
129
130  if (HasUserSkippedPromo(profile))
131    return false;
132
133  // For Chinese users skip the sign in promo.
134  if (g_browser_process->GetApplicationLocale() == "zh-CN")
135    return false;
136
137  PrefService* prefs = profile->GetPrefs();
138  int show_count = prefs->GetInteger(prefs::kSignInPromoStartupCount);
139  if (show_count >= kSignInPromoShowAtStartupMaximum)
140    return false;
141
142  // This pref can be set in the master preferences file to allow or disallow
143  // showing the sign in promo at startup.
144  if (prefs->HasPrefPath(prefs::kSignInPromoShowOnFirstRunAllowed))
145    return prefs->GetBoolean(prefs::kSignInPromoShowOnFirstRunAllowed);
146
147  // For now don't show the promo for some brands.
148  if (!AllowPromoAtStartupForCurrentBrand())
149    return false;
150
151  // Default to show the promo for Google Chrome builds.
152#if defined(GOOGLE_CHROME_BUILD)
153  return true;
154#else
155  return false;
156#endif
157}
158
159void DidShowPromoAtStartup(Profile* profile) {
160  int show_count = profile->GetPrefs()->GetInteger(
161      prefs::kSignInPromoStartupCount);
162  show_count++;
163  profile->GetPrefs()->SetInteger(prefs::kSignInPromoStartupCount, show_count);
164}
165
166void SetUserSkippedPromo(Profile* profile) {
167  profile->GetPrefs()->SetBoolean(prefs::kSignInPromoUserSkipped, true);
168}
169
170GURL GetLandingURL(const char* option, int value) {
171  const std::string& locale = g_browser_process->GetApplicationLocale();
172  std::string url = base::StringPrintf(kSignInLandingUrlPrefix, locale.c_str());
173  base::StringAppendF(&url, "?%s=%d", option, value);
174  return GURL(url);
175}
176
177GURL GetPromoURL(Source source, bool auto_close) {
178  return GetPromoURL(source, auto_close, false /* is_constrained */);
179}
180
181GURL GetPromoURL(Source source, bool auto_close, bool is_constrained) {
182  DCHECK_NE(SOURCE_UNKNOWN, source);
183
184  bool enable_inline = CommandLine::ForCurrentProcess()->HasSwitch(
185      switches::kEnableInlineSignin);
186  if (enable_inline) {
187    std::string url(chrome::kChromeUIChromeSigninURL);
188    base::StringAppendF(&url, "?%s=%d", kSignInPromoQueryKeySource, source);
189    if (auto_close)
190      base::StringAppendF(&url, "&%s=1", kSignInPromoQueryKeyAutoClose);
191    if (is_constrained)
192      base::StringAppendF(&url, "&%s=1", kSignInPromoQueryKeyConstrained);
193    return GURL(url);
194  }
195
196  // Build a Gaia-based URL that can be used to sign the user into chrome.
197  // There are required request parameters:
198  //
199  //  - tell Gaia which service the user is signing into.  In this case,
200  //    a chrome sign in uses the service "chromiumsync"
201  //  - provide a continue URL.  This is the URL that Gaia will redirect to
202  //    once the sign is complete.
203  //
204  // The continue URL includes a source parameter that can be extracted using
205  // the function GetSourceForSignInPromoURL() below.  This is used to know
206  // which of the chrome sign in access points was used to sign the user in.
207  // It is also parsed for the |auto_close| flag, which indicates that the tab
208  // must be closed after sync setup is successful.
209  // See OneClickSigninHelper for details.
210  std::string query_string = "?service=chromiumsync&sarp=1";
211
212  std::string continue_url = GetLandingURL(kSignInPromoQueryKeySource,
213                                           static_cast<int>(source)).spec();
214  if (auto_close)
215    base::StringAppendF(&continue_url, "&%s=1", kSignInPromoQueryKeyAutoClose);
216
217  base::StringAppendF(&query_string, "&%s=%s", kSignInPromoQueryKeyContinue,
218                      net::EscapeQueryParamValue(
219                          continue_url, false).c_str());
220
221  return GaiaUrls::GetInstance()->service_login_url().Resolve(query_string);
222}
223
224GURL GetNextPageURLForPromoURL(const GURL& url) {
225  std::string value;
226  if (net::GetValueForKeyInQuery(url, kSignInPromoQueryKeyContinue, &value))
227    return GURL(value);
228
229  return GURL();
230}
231
232Source GetSourceForPromoURL(const GURL& url) {
233  std::string value;
234  if (net::GetValueForKeyInQuery(url, kSignInPromoQueryKeySource, &value)) {
235    int source = 0;
236    if (base::StringToInt(value, &source) && source >= SOURCE_START_PAGE &&
237        source < SOURCE_UNKNOWN) {
238      return static_cast<Source>(source);
239    }
240  }
241  return SOURCE_UNKNOWN;
242}
243
244bool IsAutoCloseEnabledInURL(const GURL& url) {
245  std::string value;
246  if (net::GetValueForKeyInQuery(url, kSignInPromoQueryKeyAutoClose, &value)) {
247    int enabled = 0;
248    if (base::StringToInt(value, &enabled) && enabled == 1)
249      return true;
250  }
251  return false;
252}
253
254bool IsContinueUrlForWebBasedSigninFlow(const GURL& url) {
255  GURL::Replacements replacements;
256  replacements.ClearQuery();
257  const std::string& locale = g_browser_process->GetApplicationLocale();
258  return url.ReplaceComponents(replacements) ==
259      GURL(base::StringPrintf(kSignInLandingUrlPrefix, locale.c_str()));
260}
261
262void ForceWebBasedSigninFlowForTesting(bool force) {
263  g_force_web_based_signin_flow = force;
264}
265
266void RegisterProfilePrefs(
267    user_prefs::PrefRegistrySyncable* registry) {
268  registry->RegisterIntegerPref(
269      prefs::kSignInPromoStartupCount,
270      0,
271      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
272  registry->RegisterBooleanPref(
273      prefs::kSignInPromoUserSkipped,
274      false,
275      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
276  registry->RegisterBooleanPref(
277      prefs::kSignInPromoShowOnFirstRunAllowed,
278      true,
279      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
280  registry->RegisterBooleanPref(
281      prefs::kSignInPromoShowNTPBubble,
282      false,
283      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF);
284}
285
286}  // namespace signin
287