| // Copyright 2019 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "chrome/browser/ui/intent_picker_tab_helper.h" |
| |
| #include <optional> |
| #include <utility> |
| #include <vector> |
| |
| #include "base/functional/bind.h" |
| #include "base/task/sequenced_task_runner.h" |
| #include "build/build_config.h" |
| #include "chrome/browser/apps/app_service/app_service_proxy.h" |
| #include "chrome/browser/apps/app_service/app_service_proxy_factory.h" |
| #include "chrome/browser/apps/intent_helper/intent_chip_display_prefs.h" |
| #include "chrome/browser/apps/link_capturing/apps_intent_picker_delegate.h" |
| #include "chrome/browser/apps/link_capturing/intent_picker_info.h" |
| #include "chrome/browser/content_settings/host_content_settings_map_factory.h" |
| #include "chrome/browser/preloading/prefetch/no_state_prefetch/chrome_no_state_prefetch_contents_delegate.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/ui/browser.h" |
| #include "chrome/browser/ui/browser_finder.h" |
| #include "chrome/browser/ui/browser_window.h" |
| #include "chrome/browser/ui/layout_constants.h" |
| #include "chrome/browser/ui/page_action/page_action_icon_type.h" |
| #include "chrome/browser/ui/tabs/public/tab_features.h" |
| #include "chrome/browser/ui/ui_features.h" |
| #include "chrome/browser/ui/views/intent_picker/intent_picker_view_page_action_controller.h" |
| #include "chrome/browser/ui/web_applications/web_app_launch_utils.h" |
| #include "chrome/browser/web_applications/link_capturing_features.h" |
| #include "chrome/browser/web_applications/proto/web_app_install_state.pb.h" |
| #include "chrome/browser/web_applications/web_app_provider.h" |
| #include "chrome/browser/web_applications/web_app_utils.h" |
| #include "components/password_manager/content/common/web_ui_constants.h" |
| #include "components/services/app_service/public/cpp/app_types.h" |
| #include "components/services/app_service/public/cpp/icon_types.h" |
| #include "content/public/browser/navigation_handle.h" |
| #include "content/public/common/url_constants.h" |
| #include "ui/base/models/image_model.h" |
| #include "ui/gfx/favicon_size.h" |
| #include "ui/gfx/image/image.h" |
| #include "url/origin.h" |
| |
| #if BUILDFLAG(IS_CHROMEOS) |
| #include "chrome/browser/apps/link_capturing/chromeos_apps_intent_picker_delegate.h" |
| #include "chrome/browser/apps/link_capturing/metrics/intent_handling_metrics.h" |
| #else |
| #include "chrome/browser/apps/link_capturing/web_apps_intent_picker_delegate.h" |
| #endif // BUILDFLAG(IS_CHROMEOS) |
| |
| namespace { |
| |
| web_app::WebAppRegistrar* MaybeGetWebAppRegistrar( |
| content::WebContents* web_contents) { |
| // Profile for web contents might not contain a web app provider. eg. kiosk |
| // profile in Chrome OS. |
| auto* provider = web_app::WebAppProvider::GetForWebContents(web_contents); |
| return provider ? &provider->registrar_unsafe() : nullptr; |
| } |
| |
| web_app::WebAppInstallManager* MaybeGetWebAppInstallManager( |
| content::WebContents* web_contents) { |
| // Profile for web contents might not contain a web app provider. eg. kiosk |
| // profile in Chrome OS. |
| auto* provider = web_app::WebAppProvider::GetForWebContents(web_contents); |
| return provider ? &provider->install_manager() : nullptr; |
| } |
| |
| bool IsNavigatingToNewSite(content::NavigationHandle* navigation_handle) { |
| return navigation_handle->IsInPrimaryMainFrame() && |
| navigation_handle->HasCommitted() && |
| (!navigation_handle->IsSameDocument() || |
| navigation_handle->GetURL() != |
| navigation_handle->GetPreviousPrimaryMainFrameURL()); |
| } |
| |
| bool IsValidWebContentsForIntentPicker(content::WebContents* web_contents) { |
| bool is_prerender = |
| prerender::ChromeNoStatePrefetchContentsDelegate::FromWebContents( |
| web_contents) != nullptr; |
| if (is_prerender) { |
| return false; |
| } |
| |
| Browser* browser = chrome::FindBrowserWithTab(web_contents); |
| if (browser && (browser->is_type_app() || browser->is_type_app_popup())) { |
| return false; |
| } |
| return true; |
| } |
| |
| bool IsValidIntentPickerUrl(const GURL& url, bool is_error_page) { |
| if (url.SchemeIsHTTPOrHTTPS() && !is_error_page) { |
| return true; |
| } |
| |
| // chrome://password-manager is a valid PWA, so it should be considered when |
| // evaluating whether to show the intent picker. |
| if (url.SchemeIs(content::kChromeUIScheme) && |
| url.host() == password_manager::kChromeUIPasswordManagerHost) { |
| return true; |
| } |
| |
| return false; |
| } |
| |
| void ShowIntentPickerBubbleForApps( |
| content::WebContents* web_contents, |
| bool show_stay_in_chrome, |
| bool show_remember_selection, |
| IntentPickerResponse callback, |
| std::vector<apps::IntentPickerAppInfo> apps) { |
| if (apps.empty()) { |
| return; |
| } |
| |
| Browser* browser = chrome::FindBrowserWithTab(web_contents); |
| if (!browser) { |
| return; |
| } |
| |
| browser->window()->ShowIntentPickerBubble( |
| std::move(apps), show_stay_in_chrome, show_remember_selection, |
| apps::IntentPickerBubbleType::kLinkCapturing, std::nullopt, |
| std::move(callback)); |
| } |
| |
| bool IsShuttingDown(content::WebContents* web_contents) { |
| return !web_contents || web_contents->IsBeingDestroyed() || |
| web_contents->GetBrowserContext()->ShutdownStarted(); |
| } |
| |
| } // namespace |
| |
| IntentPickerTabHelper::~IntentPickerTabHelper() = default; |
| |
| void IntentPickerTabHelper::MaybeShowIntentPickerIcon() { |
| // Setting icon_resolved_ to false ensures testing callbacks can accurately |
| // wait for the entire async process to finish. |
| icon_resolved_ = false; |
| CHECK(web_contents()); |
| if (!intent_picker_delegate_->ShouldShowIntentPickerWithApps() || |
| !IsValidWebContentsForIntentPicker(web_contents())) { |
| MaybeShowIconForApps({}); |
| return; |
| } |
| |
| intent_picker_delegate_->FindAllAppsForUrl( |
| web_contents()->GetLastCommittedURL(), |
| base::BindOnce(&IntentPickerTabHelper::MaybeShowIconForApps, |
| per_navigation_weak_factory_.GetWeakPtr())); |
| } |
| |
| void IntentPickerTabHelper::ShowIntentPickerBubbleOrLaunchApp( |
| const GURL& url, |
| bool always_show, |
| ShowIntentPickerBubbleCallback callback) { |
| CHECK(web_contents()); |
| if (!intent_picker_delegate_->ShouldShowIntentPickerWithApps() || |
| !IsValidWebContentsForIntentPicker(web_contents())) { |
| base::SequencedTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, base::BindOnce(std::move(callback), /*launched=*/false)); |
| return; |
| } |
| |
| intent_picker_delegate_->FindAllAppsForUrl( |
| url, |
| base::BindOnce(&IntentPickerTabHelper::ShowIntentPickerOrLaunchAppImpl, |
| per_navigation_weak_factory_.GetWeakPtr(), url, |
| always_show, std::move(callback))); |
| } |
| |
| // static |
| void IntentPickerTabHelper::ShowOrHideIcon(content::WebContents* web_contents, |
| bool should_show_icon) { |
| IntentPickerTabHelper* tab_helper = FromWebContents(web_contents); |
| if (!tab_helper) { |
| return; |
| } |
| |
| if (apps::features::ShouldShowLinkCapturingUX()) { |
| tab_helper->current_app_icon_ = ui::ImageModel(); |
| tab_helper->show_expanded_chip_from_usage_ = false; |
| tab_helper->current_app_id_ = std::string(); |
| tab_helper->current_app_is_preferred_ = false; |
| tab_helper->last_shown_origin_ = url::Origin(); |
| } |
| |
| tab_helper->ShowOrHideIconInternal(should_show_icon); |
| } |
| |
| // static |
| int IntentPickerTabHelper::GetIntentPickerBubbleIconSize() { |
| const int kIntentPickerUiUpdateIconSize = 40; |
| return apps::features::ShouldShowLinkCapturingUX() |
| ? kIntentPickerUiUpdateIconSize |
| : gfx::kFaviconSize; |
| } |
| |
| void IntentPickerTabHelper::MaybeShowIconForApps( |
| std::vector<apps::IntentPickerAppInfo> apps) { |
| // We enter this block when we have apps available and there weren't any |
| // previously. |
| if (!should_show_icon_ && !apps.empty()) { |
| // This point doesn't exactly match when the icon is shown in the UI (e.g. |
| // if the tab is not active), but recording here corresponds more closely to |
| // navigations which cause the icon to appear. |
| intent_picker_delegate_->RecordIntentPickerIconEvent( |
| apps::IntentPickerIconEvent::kIconShown); |
| |
| #if BUILDFLAG(IS_CHROMEOS) |
| apps::IntentHandlingMetrics::RecordLinkCapturingEntryPointShown(apps); |
| #endif // BUILDFLAG(IS_CHROMEOS) |
| } |
| |
| if (apps::features::ShouldShowLinkCapturingUX()) { |
| if (apps.size() == 1 && apps[0].launch_name != current_app_id_) { |
| current_app_id_ = apps[0].launch_name; |
| |
| // If this app is the preferred app to handle this URL, the icon will |
| // always be shown as expanded, regardless of the usage-based decision |
| // calculated in UpdateExpandedState(). |
| current_app_is_preferred_ = |
| intent_picker_delegate_->IsPreferredAppForSupportedLinks( |
| current_app_id_); |
| |
| intent_picker_delegate_->LoadSingleAppIcon( |
| apps[0].type, current_app_id_, |
| GetLayoutConstant(LOCATION_BAR_ICON_SIZE), |
| base::BindOnce(&IntentPickerTabHelper::OnAppIconLoadedForChip, |
| per_navigation_weak_factory_.GetWeakPtr(), |
| current_app_id_)); |
| return; |
| } else if (apps.size() != 1) { |
| current_app_icon_ = ui::ImageModel(); |
| current_app_id_ = std::string(); |
| current_app_is_preferred_ = false; |
| } |
| } |
| |
| ShowIconForLinkIntent(!apps.empty()); |
| } |
| |
| IntentPickerTabHelper::IntentPickerTabHelper(content::WebContents* web_contents) |
| : content::WebContentsObserver(web_contents), |
| content::WebContentsUserData<IntentPickerTabHelper>(*web_contents), |
| registrar_(MaybeGetWebAppRegistrar(web_contents)), |
| install_manager_(MaybeGetWebAppInstallManager(web_contents)) { |
| if (install_manager_) { |
| install_manager_observation_.Observe(install_manager_.get()); |
| } |
| |
| Profile* profile = |
| Profile::FromBrowserContext(web_contents->GetBrowserContext()); |
| #if BUILDFLAG(IS_CHROMEOS) |
| intent_picker_delegate_ = |
| std::make_unique<apps::ChromeOsAppsIntentPickerDelegate>(profile); |
| #else |
| intent_picker_delegate_ = std::make_unique<apps::WebAppsIntentPickerDelegate>( |
| profile, std::vector<int>{GetLayoutConstant(LOCATION_BAR_ICON_SIZE), |
| GetIntentPickerBubbleIconSize()}); |
| #endif // BUILDFLAG(IS_CHROMEOS) |
| } |
| |
| void IntentPickerTabHelper::OnAppIconLoaded( |
| std::vector<apps::IntentPickerAppInfo> apps, |
| IntentPickerIconLoaderCallback callback, |
| size_t index, |
| ui::ImageModel app_icon) { |
| apps[index].icon_model = app_icon; |
| |
| if (index == apps.size() - 1) { |
| std::move(callback).Run(std::move(apps)); |
| } else { |
| LoadAppIcon(std::move(apps), index + 1, std::move(callback)); |
| } |
| } |
| |
| void IntentPickerTabHelper::LoadAppIcon( |
| std::vector<apps::IntentPickerAppInfo> apps, |
| size_t index, |
| IntentPickerIconLoaderCallback callback) { |
| if (index >= apps.size()) { |
| std::move(callback).Run(std::move(apps)); |
| return; |
| } |
| |
| const std::string& app_id = apps[index].launch_name; |
| auto app_type = apps[index].type; |
| |
| intent_picker_delegate_->LoadSingleAppIcon( |
| app_type, app_id, GetIntentPickerBubbleIconSize(), |
| base::BindOnce(&IntentPickerTabHelper::OnAppIconLoaded, |
| per_navigation_weak_factory_.GetWeakPtr(), std::move(apps), |
| std::move(callback), index)); |
| } |
| |
| void IntentPickerTabHelper::UpdateExpandedState(bool should_show_icon) { |
| GURL url = web_contents()->GetLastCommittedURL(); |
| |
| if (!should_show_icon || url.is_empty()) { |
| show_expanded_chip_from_usage_ = false; |
| last_shown_origin_ = url::Origin(); |
| return; |
| } |
| |
| url::Origin origin = url::Origin::Create(url); |
| |
| // Determine whether to show the Chip as expanded/collapsed whenever the |
| // origin changes. |
| // TODO(b/305075981): Move IntentChipDisplayPrefs to c/b/apps/link_capturing. |
| if (!origin.IsSameOriginWith(last_shown_origin_)) { |
| last_shown_origin_ = origin; |
| Profile* profile = |
| Profile::FromBrowserContext(web_contents()->GetBrowserContext()); |
| auto chip_state = |
| IntentChipDisplayPrefs::GetChipStateAndIncrementCounter(profile, url); |
| show_expanded_chip_from_usage_ = |
| chip_state == IntentChipDisplayPrefs::ChipState::kExpanded; |
| } |
| } |
| |
| void IntentPickerTabHelper::OnAppIconLoadedForChip(const std::string& app_id, |
| ui::ImageModel app_icon) { |
| if (app_id != current_app_id_) { |
| return; |
| } |
| |
| if (!app_icon.IsEmpty()) { |
| current_app_icon_ = app_icon; |
| } else { |
| current_app_id_ = std::string(); |
| current_app_icon_ = ui::ImageModel(); |
| } |
| |
| ShowIconForLinkIntent(true); |
| } |
| |
| void IntentPickerTabHelper::ShowIconForLinkIntent(bool should_show_icon) { |
| if (apps::features::ShouldShowLinkCapturingUX()) { |
| UpdateExpandedState(should_show_icon); |
| } |
| |
| ShowOrHideIconInternal(should_show_icon); |
| } |
| |
| void IntentPickerTabHelper::ShowOrHideIconInternal(bool should_show_icon) { |
| should_show_icon_ = should_show_icon; |
| |
| Browser* browser = chrome::FindBrowserWithTab(web_contents()); |
| if (!browser) { |
| return; |
| } |
| |
| if (IsPageActionMigrated(PageActionIconType::kIntentPicker)) { |
| tabs::TabInterface* tab_interface = |
| tabs::TabInterface::GetFromContents(&GetWebContents()); |
| UpdatePageAction(tab_interface, should_show_icon); |
| } else { |
| browser->window()->UpdatePageActionIcon(PageActionIconType::kIntentPicker); |
| } |
| |
| icon_resolved_ = true; |
| if (icon_update_closure_for_testing_) { |
| std::move(icon_update_closure_for_testing_).Run(); |
| } |
| } |
| |
| void IntentPickerTabHelper::ShowIntentPickerOrLaunchAppImpl( |
| const GURL& url, |
| bool always_show, |
| ShowIntentPickerBubbleCallback callback, |
| std::vector<apps::IntentPickerAppInfo> apps) { |
| if (apps.empty() || IsShuttingDown(web_contents())) { |
| base::SequencedTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, base::BindOnce(std::move(callback), /*launched=*/false)); |
| return; |
| } |
| |
| // TODO(crbug.com/421950209): Add and record new enum for when the intent |
| // picker is shown via the Web Install API. |
| intent_picker_delegate_->RecordIntentPickerIconEvent( |
| apps::IntentPickerIconEvent::kIconClicked); |
| |
| if (apps.size() == 1 && !always_show && |
| intent_picker_delegate_->ShouldLaunchAppDirectly(url, apps[0].launch_name, |
| apps[0].type)) { |
| // TODO(b/305075981): Move IntentChipDisplayPrefs to |
| // c/b/apps/link_capturing. |
| if (apps::features::ShouldShowLinkCapturingUX()) { |
| Profile* profile = |
| Profile::FromBrowserContext(web_contents()->GetBrowserContext()); |
| IntentChipDisplayPrefs::ResetIntentChipCounter(profile, url); |
| } |
| intent_picker_delegate_->LaunchApp( |
| web_contents(), url, apps[0].launch_name, apps[0].type, |
| base::BindOnce(std::move(callback), /*launched=*/true)); |
| return; |
| } |
| |
| bool show_stay_in_chrome; |
| bool show_remember_selection; |
| #if BUILDFLAG(IS_CHROMEOS) |
| show_stay_in_chrome = true; |
| show_remember_selection = true; |
| #else |
| show_stay_in_chrome = false; |
| show_remember_selection = false; |
| #endif // BUILDFLAG(IS_CHROMEOS) |
| |
| auto show_intent_picker_bubble = base::BindOnce( |
| &ShowIntentPickerBubbleForApps, web_contents(), show_stay_in_chrome, |
| show_remember_selection, |
| base::BindOnce(&IntentPickerTabHelper::OnIntentPickerClosedMaybeLaunch, |
| per_navigation_weak_factory_.GetWeakPtr(), url, |
| std::move(callback))); |
| |
| LoadAppIcon(std::move(apps), |
| /*index=*/0, std::move(show_intent_picker_bubble)); |
| } |
| |
| void IntentPickerTabHelper::OnIntentPickerClosedMaybeLaunch( |
| const GURL& url, |
| ShowIntentPickerBubbleCallback callback, |
| const std::string& launch_name, |
| apps::PickerEntryType entry_type, |
| apps::IntentPickerCloseReason close_reason, |
| bool should_persist) { |
| if (IsShuttingDown(web_contents())) { |
| base::SequencedTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, base::BindOnce(std::move(callback), /*launched=*/false)); |
| return; |
| } |
| |
| bool should_launch_app = |
| (close_reason == apps::IntentPickerCloseReason::OPEN_APP); |
| |
| intent_picker_delegate_->RecordOutputMetrics( |
| entry_type, close_reason, should_persist, should_launch_app); |
| if (should_persist) { |
| intent_picker_delegate_->PersistIntentPreferencesForApp(entry_type, |
| launch_name); |
| } |
| |
| if (!should_launch_app) { |
| base::SequencedTaskRunner::GetCurrentDefault()->PostTask( |
| FROM_HERE, base::BindOnce(std::move(callback), /*launched=*/false)); |
| return; |
| } |
| // TODO(crbug.com/305075981): Move IntentChipDisplayPrefs to |
| // c/b/apps/link_capturing. |
| if (apps::features::ShouldShowLinkCapturingUX()) { |
| Profile* profile = |
| Profile::FromBrowserContext(web_contents()->GetBrowserContext()); |
| IntentChipDisplayPrefs::ResetIntentChipCounter(profile, url); |
| } |
| intent_picker_delegate_->LaunchApp( |
| web_contents(), url, launch_name, entry_type, |
| base::BindOnce(std::move(callback), /*launched=*/true)); |
| } |
| |
| void IntentPickerTabHelper::SetIconUpdateCallbackForTesting( |
| base::OnceClosure callback, |
| bool include_latest_navigation) { |
| if (icon_resolved_ && include_latest_navigation) { |
| std::move(callback).Run(); |
| return; |
| } |
| icon_update_closure_for_testing_ = std::move(callback); |
| } |
| |
| void IntentPickerTabHelper::DidStartNavigation( |
| content::NavigationHandle* navigation_handle) { |
| if (IsNavigatingToNewSite(navigation_handle)) { |
| icon_resolved_ = false; |
| } |
| } |
| |
| void IntentPickerTabHelper::DidFinishNavigation( |
| content::NavigationHandle* navigation_handle) { |
| // For a http/https scheme URL navigation, we will check if the |
| // url can be handled by some apps, and show intent picker icon |
| // or bubble if there are some apps available. We only want to check this if |
| // the navigation happens in the primary main frame, and the navigation is not |
| // the same document with same URL. |
| if (!web_contents()) { |
| return; |
| } |
| |
| if (IsNavigatingToNewSite(navigation_handle)) { |
| per_navigation_weak_factory_.InvalidateWeakPtrs(); |
| |
| if (IsValidIntentPickerUrl(navigation_handle->GetURL(), |
| navigation_handle->IsErrorPage())) { |
| MaybeShowIntentPickerIcon(); |
| } else { |
| ShowOrHideIcon(web_contents(), /*should_show_icon=*/false); |
| } |
| } |
| } |
| |
| void IntentPickerTabHelper::OnWebAppWillBeUninstalled( |
| const webapps::AppId& app_id) { |
| // WebAppTabHelper has an app_id but it is reset during |
| // OnWebAppWillBeUninstalled so using FindAppWithUrlInScope. |
| std::optional<webapps::AppId> local_app_id = |
| registrar_->FindBestAppWithUrlInScope( |
| web_contents()->GetLastCommittedURL(), |
| web_app::WebAppFilter::InstalledInChrome()); |
| if (app_id == local_app_id) { |
| ShowOrHideIcon(web_contents(), /*should_show_icon=*/false); |
| } |
| } |
| |
| void IntentPickerTabHelper::OnWebAppInstallManagerDestroyed() { |
| install_manager_observation_.Reset(); |
| } |
| |
| void IntentPickerTabHelper::UpdatePageAction(tabs::TabInterface* tab_interface, |
| bool show_icon) { |
| if (auto* const tab_features = tab_interface->GetTabFeatures()) { |
| if (auto* controller = |
| tab_features->intent_picker_view_page_action_controller()) { |
| controller->UpdatePageActionVisibility(show_icon, app_icon()); |
| } |
| } |
| } |
| |
| WEB_CONTENTS_USER_DATA_KEY_IMPL(IntentPickerTabHelper); |