blob: 548ecde12e2b5ecd17868112c59fa5b18e877d8b [file] [log] [blame]
// 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/web_applications/web_app_dialog_utils.h"
#include <memory>
#include <utility>
#include "base/feature_list.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/functional/callback_helpers.h"
#include "base/metrics/user_metrics.h"
#include "base/no_destructor.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/ui/browser.h"
#include "chrome/browser/ui/browser_window.h"
#include "chrome/browser/ui/browser_window/public/browser_window_interface.h"
#include "chrome/browser/ui/tabs/public/tab_features.h"
#include "chrome/browser/ui/tabs/tab_strip_model.h"
#include "chrome/browser/ui/user_education/browser_user_education_interface.h"
#include "chrome/browser/ui/web_applications/pwa_install_page_action.h"
#include "chrome/browser/ui/web_applications/web_app_dialogs.h"
#include "chrome/browser/web_applications/mojom/user_display_mode.mojom.h"
#include "chrome/browser/web_applications/web_app_command_manager.h"
#include "chrome/browser/web_applications/web_app_command_scheduler.h"
#include "chrome/browser/web_applications/web_app_constants.h"
#include "chrome/browser/web_applications/web_app_helpers.h"
#include "chrome/browser/web_applications/web_app_install_info.h"
#include "chrome/browser/web_applications/web_app_install_manager.h"
#include "chrome/browser/web_applications/web_app_install_params.h"
#include "chrome/browser/web_applications/web_app_install_utils.h"
#include "chrome/browser/web_applications/web_app_provider.h"
#include "chrome/browser/web_applications/web_app_screenshot_fetcher.h"
#include "chrome/browser/web_applications/web_app_utils.h"
#include "chrome/common/chrome_features.h"
#include "components/feature_engagement/public/feature_constants.h"
#include "components/webapps/browser/banners/app_banner_manager.h"
#include "components/webapps/browser/banners/web_app_banner_data.h"
#include "components/webapps/browser/features.h"
#include "components/webapps/browser/installable/installable_data.h"
#include "components/webapps/browser/installable/installable_metrics.h"
#include "components/webapps/browser/installable/ml_install_operation_tracker.h"
#include "components/webapps/browser/installable/ml_installability_promoter.h"
#include "content/public/browser/navigation_entry.h"
#if BUILDFLAG(IS_CHROMEOS)
// TODO(crbug.com/40147906): Enable gn check once it handles conditional
// includes
#include "components/metrics/structured/structured_events.h" // nogncheck
#include "components/metrics/structured/structured_metrics_client.h" // nogncheck
#endif
namespace web_app {
namespace {
#if BUILDFLAG(IS_CHROMEOS)
namespace cros_events = metrics::structured::events::v2::cr_os_events;
#endif
void OnWebAppInstallShowInstallDialog(
WebAppInstallFlow flow,
webapps::WebappInstallSource install_source,
PwaInProductHelpState iph_state,
std::unique_ptr<webapps::MlInstallOperationTracker> install_tracker,
bool show_initiating_origin,
base::WeakPtr<WebAppScreenshotFetcher> screenshot_fetcher,
content::WebContents* initiator_web_contents,
std::unique_ptr<WebAppInstallInfo> web_app_info,
WebAppInstallationAcceptanceCallback web_app_acceptance_callback) {
DCHECK(web_app_info);
switch (flow) {
case WebAppInstallFlow::kInstallSite:
web_app_info->user_display_mode = mojom::UserDisplayMode::kStandalone;
#if BUILDFLAG(IS_CHROMEOS)
if (install_source == webapps::WebappInstallSource::MENU_BROWSER_TAB) {
webapps::AppId app_id =
web_app::GenerateAppIdFromManifestId(web_app_info->manifest_id());
metrics::structured::StructuredMetricsClient::Record(
cros_events::AppDiscovery_Browser_ClickInstallAppFromMenu()
.SetAppId(app_id));
}
#endif
if (screenshot_fetcher) {
ShowWebAppDetailedInstallDialog(
initiator_web_contents, std::move(web_app_info),
std::move(install_tracker), std::move(web_app_acceptance_callback),
screenshot_fetcher, iph_state);
return;
} else if (web_app_info->is_diy_app) {
ShowDiyAppInstallDialog(initiator_web_contents, std::move(web_app_info),
std::move(install_tracker),
std::move(web_app_acceptance_callback),
iph_state);
return;
} else {
ShowSimpleInstallDialogForWebApps(
initiator_web_contents, std::move(web_app_info),
std::move(install_tracker), std::move(web_app_acceptance_callback),
iph_state, show_initiating_origin);
return;
}
#if BUILDFLAG(IS_CHROMEOS)
case WebAppInstallFlow::kCreateShortcut: {
webapps::AppId app_id =
web_app::GenerateAppIdFromManifestId(web_app_info->manifest_id());
metrics::structured::StructuredMetricsClient::Record(
cros_events::AppDiscovery_Browser_CreateShortcut().SetAppId(app_id));
}
ShowCreateShortcutDialog(initiator_web_contents, std::move(web_app_info),
std::move(install_tracker),
std::move(web_app_acceptance_callback));
return;
#endif
case WebAppInstallFlow::kUnknown:
NOTREACHED();
}
NOTREACHED();
}
WebAppInstalledCallback& GetInstalledCallbackForTesting() {
static base::NoDestructor<WebAppInstalledCallback> instance;
return *instance;
}
void OnWebAppInstalled(WebAppInstalledCallback callback,
const webapps::AppId& installed_app_id,
webapps::InstallResultCode code) {
if (GetInstalledCallbackForTesting()) {
std::move(GetInstalledCallbackForTesting()).Run(installed_app_id, code);
}
std::move(callback).Run(installed_app_id, code);
}
} // namespace
bool CanCreateWebApp(const Browser* browser) {
// Check whether user is allowed to install web app.
if (!WebAppProvider::GetForWebApps(browser->profile()) ||
!AreWebAppsUserInstallable(browser->profile())) {
return false;
}
// Check whether we're able to install the current page as an app.
content::WebContents* web_contents =
browser->tab_strip_model()->GetActiveWebContents();
if (!IsValidWebAppUrl(web_contents->GetLastCommittedURL()) ||
web_contents->IsCrashed()) {
return false;
}
content::NavigationEntry* entry =
web_contents->GetController().GetLastCommittedEntry();
if (entry && entry->GetPageType() == content::PAGE_TYPE_ERROR) {
return false;
}
return true;
}
bool CanPopOutWebApp(Profile* profile) {
return AreWebAppsEnabled(profile) && !profile->IsGuestSession() &&
!profile->IsOffTheRecord();
}
void CreateWebAppFromCurrentWebContents(Browser* browser,
WebAppInstallFlow flow) {
DCHECK(CanCreateWebApp(browser));
content::WebContents* web_contents =
browser->tab_strip_model()->GetActiveWebContents();
auto* provider = WebAppProvider::GetForWebContents(web_contents);
DCHECK(provider);
webapps::MLInstallabilityPromoter* promoter =
webapps::MLInstallabilityPromoter::FromWebContents(web_contents);
CHECK(promoter);
if (promoter->HasCurrentInstall()) {
return;
}
if (provider->command_manager().IsInstallingForWebContents(web_contents)) {
return;
}
webapps::AppBannerManager* app_banner_manager =
webapps::AppBannerManager::FromWebContents(web_contents);
if (!app_banner_manager) {
return;
}
std::optional<webapps::WebAppBannerData> data =
app_banner_manager->GetCurrentWebAppBannerData();
webapps::WebappInstallSource install_source =
webapps::InstallableMetrics::GetInstallSource(
web_contents,
#if BUILDFLAG(IS_CHROMEOS)
flow == WebAppInstallFlow::kCreateShortcut
? webapps::InstallTrigger::CREATE_SHORTCUT
:
#endif
webapps::InstallTrigger::MENU);
std::unique_ptr<webapps::MlInstallOperationTracker> install_tracker =
promoter->RegisterCurrentInstallForWebContents(install_source);
WebAppInstalledCallback callback = base::DoNothing();
// Appropriately set the fallback behavior to distinguish installation of DIY
// apps with the create shortcut flow.
FallbackBehavior fallback_behavior =
#if BUILDFLAG(IS_CHROMEOS)
flow == WebAppInstallFlow::kCreateShortcut
? FallbackBehavior::kAllowFallbackDataAlways
:
#endif
FallbackBehavior::kUseFallbackInfoWhenNotInstallable;
provider->scheduler().FetchManifestAndInstall(
install_source, web_contents->GetWeakPtr(),
base::BindOnce(OnWebAppInstallShowInstallDialog, flow, install_source,
PwaInProductHelpState::kNotShown,
std::move(install_tracker),
/*show_initiating_origin=*/false),
base::BindOnce(OnWebAppInstalled, std::move(callback)),
fallback_behavior);
}
bool CreateWebAppFromManifest(content::WebContents* web_contents,
webapps::WebappInstallSource install_source,
WebAppInstalledCallback installed_callback,
PwaInProductHelpState iph_state) {
auto* provider = WebAppProvider::GetForWebContents(web_contents);
if (!provider) {
return false;
}
webapps::MLInstallabilityPromoter* promoter =
webapps::MLInstallabilityPromoter::FromWebContents(web_contents);
if (promoter->HasCurrentInstall()) {
return false;
}
if (provider->command_manager().IsInstallingForWebContents(web_contents)) {
return false;
}
webapps::AppBannerManager* app_banner_manager =
webapps::AppBannerManager::FromWebContents(web_contents);
if (!app_banner_manager) {
return false;
}
std::optional<webapps::WebAppBannerData> data =
app_banner_manager->GetCurrentWebAppBannerData();
std::unique_ptr<webapps::MlInstallOperationTracker> install_tracker =
promoter->RegisterCurrentInstallForWebContents(install_source);
// If the source is from ML, there may not be a manifest, so allow the command
// to use the metadata from the page too.
FallbackBehavior fallback_behavior =
install_source == webapps::WebappInstallSource::ML_PROMOTION
? FallbackBehavior::kUseFallbackInfoWhenNotInstallable
: FallbackBehavior::kCraftedManifestOnly;
provider->scheduler().FetchManifestAndInstall(
install_source, web_contents->GetWeakPtr(),
base::BindOnce(OnWebAppInstallShowInstallDialog,
WebAppInstallFlow::kInstallSite, install_source, iph_state,
std::move(install_tracker),
/*show_initiating_origin=*/false),
base::BindOnce(OnWebAppInstalled, std::move(installed_callback)),
fallback_behavior);
return true;
}
void CreateWebAppForBackgroundInstall(
content::WebContents* initiating_web_contents,
std::unique_ptr<webapps::MlInstallOperationTracker> tracker,
const GURL& install_url,
const std::optional<GURL>& manifest_id,
WebAppInstalledCallback installed_callback) {
auto* provider = WebAppProvider::GetForWebContents(initiating_web_contents);
CHECK(provider);
provider->scheduler().InstallAppFromUrl(
install_url, manifest_id, initiating_web_contents->GetWeakPtr(),
base::BindOnce(&OnWebAppInstallShowInstallDialog,
WebAppInstallFlow::kInstallSite,
webapps::WebappInstallSource::WEB_INSTALL,
PwaInProductHelpState::kNotShown, std::move(tracker),
/*show_initiating_origin=*/true),
std::move(installed_callback));
}
void ShowPwaInstallDialog(BrowserWindowInterface* bwi) {
CHECK(bwi);
base::RecordAction(base::UserMetricsAction("PWAInstallIcon"));
content::WebContents* const web_contents =
bwi->GetTabStripModel()->GetActiveWebContents();
CHECK(web_contents);
PwaInstallPageActionController* const pwa_install_controller =
bwi->GetActiveTabInterface()
->GetTabFeatures()
->pwa_install_page_action_controller();
pwa_install_controller->SetIsExecuting(true);
// Close PWA install IPH if it is showing.
PwaInProductHelpState iph_state = PwaInProductHelpState::kNotShown;
const bool install_icon_clicked_after_iph_shown =
BrowserUserEducationInterface::From(bwi)->NotifyFeaturePromoFeatureUsed(
feature_engagement::kIPHDesktopPwaInstallFeature,
FeaturePromoFeatureUsedAction::kClosePromoIfPresent);
if (install_icon_clicked_after_iph_shown) {
iph_state = PwaInProductHelpState::kShown;
}
#if BUILDFLAG(IS_CHROMEOS)
metrics::structured::StructuredMetricsClient::Record(
metrics::structured::events::v2::cr_os_events::
AppDiscovery_Browser_OmniboxInstallIconClicked()
.SetIPHShown(install_icon_clicked_after_iph_shown));
#endif
CreateWebAppFromManifest(web_contents,
webapps::WebappInstallSource::OMNIBOX_INSTALL_ICON,
base::DoNothing(), iph_state);
pwa_install_controller->SetIsExecuting(false);
}
void SetInstalledCallbackForTesting(WebAppInstalledCallback callback) {
GetInstalledCallbackForTesting() = std::move(callback);
}
} // namespace web_app