| // Copyright 2016 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/startup_tab_provider.h" |
| |
| #include "base/metrics/histogram_macros.h" |
| #include "build/branding_buildflags.h" |
| #include "build/chromeos_buildflags.h" |
| #include "chrome/browser/first_run/first_run.h" |
| #include "chrome/browser/profile_resetter/triggered_profile_resetter.h" |
| #include "chrome/browser/profile_resetter/triggered_profile_resetter_factory.h" |
| #include "chrome/browser/signin/identity_manager_factory.h" |
| #include "chrome/browser/signin/signin_util.h" |
| #include "chrome/browser/sync/profile_sync_service_factory.h" |
| #include "chrome/browser/ui/browser.h" |
| #include "chrome/browser/ui/browser_list.h" |
| #include "chrome/browser/ui/chrome_pages.h" |
| #include "chrome/browser/ui/tabs/pinned_tab_codec.h" |
| #include "chrome/browser/ui/tabs/tab_strip_model.h" |
| #include "chrome/common/pref_names.h" |
| #include "chrome/common/url_constants.h" |
| #include "components/prefs/pref_service.h" |
| #include "components/signin/public/identity_manager/identity_manager.h" |
| #include "components/signin/public/identity_manager/primary_account_mutator.h" |
| #include "net/base/url_util.h" |
| |
| #if defined(OS_WIN) |
| #include "base/win/windows_version.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/browser/shell_integration.h" |
| #endif // defined(OS_WIN) |
| |
| namespace { |
| |
| // Attempts to find an existing, non-empty tabbed browser for this profile. |
| bool ProfileHasOtherTabbedBrowser(Profile* profile) { |
| BrowserList* browser_list = BrowserList::GetInstance(); |
| auto other_tabbed_browser = std::find_if( |
| browser_list->begin(), browser_list->end(), [profile](Browser* browser) { |
| return browser->profile() == profile && browser->is_type_normal() && |
| !browser->tab_strip_model()->empty(); |
| }); |
| return other_tabbed_browser != browser_list->end(); |
| } |
| |
| } // namespace |
| |
| StartupTabs StartupTabProviderImpl::GetOnboardingTabs(Profile* profile) const { |
| // Chrome OS has its own welcome flow provided by OOBE. |
| #if BUILDFLAG(IS_CHROMEOS_ASH) |
| return StartupTabs(); |
| #else |
| if (!profile) |
| return StartupTabs(); |
| |
| StandardOnboardingTabsParams standard_params; |
| standard_params.is_first_run = first_run::IsChromeFirstRun(); |
| PrefService* prefs = profile->GetPrefs(); |
| standard_params.has_seen_welcome_page = |
| prefs && prefs->GetBoolean(prefs::kHasSeenWelcomePage); |
| standard_params.is_signin_allowed = |
| ProfileSyncServiceFactory::IsSyncAllowed(profile); |
| if (auto* identity_manager = IdentityManagerFactory::GetForProfile(profile)) { |
| standard_params.is_signed_in = |
| identity_manager->HasPrimaryAccount(signin::ConsentLevel::kSync); |
| } |
| standard_params.is_supervised_user = profile->IsSupervised(); |
| standard_params.is_force_signin_enabled = signin_util::IsForceSigninEnabled(); |
| |
| return GetStandardOnboardingTabsForState(standard_params); |
| #endif // BUILDFLAG(IS_CHROMEOS_ASH) |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetWelcomeBackTabs( |
| Profile* profile, |
| StartupBrowserCreator* browser_creator, |
| bool process_startup) const { |
| StartupTabs tabs; |
| if (!process_startup || !browser_creator) |
| return tabs; |
| if (browser_creator->welcome_back_page() && |
| CanShowWelcome(ProfileSyncServiceFactory::IsSyncAllowed(profile), |
| profile->IsSupervised(), |
| signin_util::IsForceSigninEnabled())) { |
| tabs.emplace_back(GetWelcomePageUrl(false), false); |
| } |
| return tabs; |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetDistributionFirstRunTabs( |
| StartupBrowserCreator* browser_creator) const { |
| if (!browser_creator) |
| return StartupTabs(); |
| StartupTabs tabs = GetInitialPrefsTabsForState( |
| first_run::IsChromeFirstRun(), browser_creator->first_run_tabs_); |
| browser_creator->first_run_tabs_.clear(); |
| return tabs; |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetResetTriggerTabs( |
| Profile* profile) const { |
| auto* triggered_profile_resetter = |
| TriggeredProfileResetterFactory::GetForBrowserContext(profile); |
| bool has_reset_trigger = triggered_profile_resetter && |
| triggered_profile_resetter->HasResetTrigger(); |
| return GetResetTriggerTabsForState(has_reset_trigger); |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetPinnedTabs( |
| const base::CommandLine& command_line, |
| Profile* profile) const { |
| return GetPinnedTabsForState( |
| StartupBrowserCreator::GetSessionStartupPref(command_line, profile), |
| PinnedTabCodec::ReadPinnedTabs(profile), |
| ProfileHasOtherTabbedBrowser(profile)); |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetPreferencesTabs( |
| const base::CommandLine& command_line, |
| Profile* profile) const { |
| return GetPreferencesTabsForState( |
| StartupBrowserCreator::GetSessionStartupPref(command_line, profile), |
| ProfileHasOtherTabbedBrowser(profile)); |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetNewTabPageTabs( |
| const base::CommandLine& command_line, |
| Profile* profile) const { |
| return GetNewTabPageTabsForState( |
| StartupBrowserCreator::GetSessionStartupPref(command_line, profile)); |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetPostCrashTabs( |
| bool has_incompatible_applications) const { |
| return GetPostCrashTabsForState(has_incompatible_applications); |
| } |
| |
| StartupTabs StartupTabProviderImpl::GetExtensionCheckupTabs( |
| bool serve_extensions_page) const { |
| return GetExtensionCheckupTabsForState(serve_extensions_page); |
| } |
| |
| // static |
| bool StartupTabProviderImpl::CanShowWelcome(bool is_signin_allowed, |
| bool is_supervised_user, |
| bool is_force_signin_enabled) { |
| return is_signin_allowed && !is_supervised_user && !is_force_signin_enabled; |
| } |
| |
| // static |
| bool StartupTabProviderImpl::ShouldShowWelcomeForOnboarding( |
| bool has_seen_welcome_page, |
| bool is_signed_in) { |
| return !has_seen_welcome_page && !is_signed_in; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetStandardOnboardingTabsForState( |
| const StandardOnboardingTabsParams& params) { |
| StartupTabs tabs; |
| if (CanShowWelcome(params.is_signin_allowed, params.is_supervised_user, |
| params.is_force_signin_enabled) && |
| ShouldShowWelcomeForOnboarding(params.has_seen_welcome_page, |
| params.is_signed_in)) { |
| tabs.emplace_back(GetWelcomePageUrl(!params.is_first_run), false); |
| } |
| return tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetInitialPrefsTabsForState( |
| bool is_first_run, |
| const std::vector<GURL>& first_run_tabs) { |
| // Constants: Magic words used by initial preferences files in place of a URL |
| // host to indicate that internal pages should appear on first run. |
| static constexpr char kNewTabUrlHost[] = "new_tab_page"; |
| static constexpr char kWelcomePageUrlHost[] = "welcome_page"; |
| |
| StartupTabs tabs; |
| if (is_first_run) { |
| tabs.reserve(first_run_tabs.size()); |
| for (GURL url : first_run_tabs) { |
| if (url.host_piece() == kNewTabUrlHost) |
| url = GURL(chrome::kChromeUINewTabURL); |
| else if (url.host_piece() == kWelcomePageUrlHost) |
| url = GetWelcomePageUrl(false); |
| tabs.emplace_back(url, false); |
| } |
| } |
| return tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetResetTriggerTabsForState( |
| bool profile_has_trigger) { |
| StartupTabs tabs; |
| if (profile_has_trigger) |
| tabs.emplace_back(GetTriggeredResetSettingsUrl(), false); |
| return tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetPinnedTabsForState( |
| const SessionStartupPref& pref, |
| const StartupTabs& pinned_tabs, |
| bool profile_has_other_tabbed_browser) { |
| if (pref.type == SessionStartupPref::Type::LAST || |
| profile_has_other_tabbed_browser) |
| return StartupTabs(); |
| return pinned_tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetPreferencesTabsForState( |
| const SessionStartupPref& pref, |
| bool profile_has_other_tabbed_browser) { |
| StartupTabs tabs; |
| if (pref.type == SessionStartupPref::Type::URLS && !pref.urls.empty() && |
| !profile_has_other_tabbed_browser) { |
| for (const auto& url : pref.urls) |
| tabs.push_back(StartupTab(url, false)); |
| } |
| return tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetNewTabPageTabsForState( |
| const SessionStartupPref& pref) { |
| StartupTabs tabs; |
| if (pref.type != SessionStartupPref::Type::LAST) |
| tabs.emplace_back(GURL(chrome::kChromeUINewTabURL), false); |
| return tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetPostCrashTabsForState( |
| bool has_incompatible_applications) { |
| StartupTabs tabs; |
| if (has_incompatible_applications) |
| AddIncompatibleApplicationsUrl(&tabs); |
| return tabs; |
| } |
| |
| // static |
| StartupTabs StartupTabProviderImpl::GetExtensionCheckupTabsForState( |
| bool serve_extensions_page) { |
| StartupTabs tabs; |
| if (serve_extensions_page) { |
| tabs.emplace_back( |
| net::AppendQueryParameter(GURL(chrome::kChromeUIExtensionsURL), |
| "checkup", "shown"), |
| false); |
| } |
| return tabs; |
| } |
| |
| // static |
| GURL StartupTabProviderImpl::GetWelcomePageUrl(bool use_later_run_variant) { |
| GURL url(chrome::kChromeUIWelcomeURL); |
| return use_later_run_variant |
| ? net::AppendQueryParameter(url, "variant", "everywhere") |
| : url; |
| } |
| |
| // static |
| void StartupTabProviderImpl::AddIncompatibleApplicationsUrl(StartupTabs* tabs) { |
| #if defined(OS_WIN) && BUILDFLAG(GOOGLE_CHROME_BRANDING) |
| UMA_HISTOGRAM_BOOLEAN("IncompatibleApplicationsPage.AddedPostCrash", true); |
| GURL url(chrome::kChromeUISettingsURL); |
| tabs->emplace_back(url.Resolve("incompatibleApplications"), false); |
| #endif // defined(OS_WIN) && BUILDFLAG(GOOGLE_CHROME_BRANDING) |
| } |
| |
| // static |
| GURL StartupTabProviderImpl::GetTriggeredResetSettingsUrl() { |
| return GURL( |
| chrome::GetSettingsUrl(chrome::kTriggeredResetProfileSettingsSubPage)); |
| } |