| // Copyright (c) 2013 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/views/outdated_upgrade_bubble_view.h" |
| |
| #include "base/bind.h" |
| #include "base/metrics/histogram_macros.h" |
| #include "base/metrics/user_metrics.h" |
| #include "base/task/post_task.h" |
| #include "base/task/thread_pool.h" |
| #include "build/build_config.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/browser/ui/browser_dialogs.h" |
| #include "chrome/browser/ui/views/chrome_layout_provider.h" |
| #include "chrome/browser/ui/views/chrome_typography.h" |
| #include "chrome/browser/upgrade_detector/upgrade_detector.h" |
| #include "chrome/common/pref_names.h" |
| #include "chrome/grit/chromium_strings.h" |
| #include "chrome/grit/generated_resources.h" |
| #include "components/prefs/pref_service.h" |
| #include "content/public/browser/page_navigator.h" |
| #include "ui/base/l10n/l10n_util.h" |
| #include "ui/base/models/dialog_model.h" |
| #include "ui/views/bubble/bubble_dialog_model_host.h" |
| #include "ui/views/controls/label.h" |
| #include "ui/views/layout/fill_layout.h" |
| #include "ui/views/style/typography.h" |
| #include "ui/views/widget/widget.h" |
| #include "url/gurl.h" |
| |
| #if defined(OS_WIN) |
| #include "chrome/installer/util/google_update_util.h" |
| #endif |
| |
| namespace { |
| |
| // The URL to be used to re-install Chrome when auto-update failed for too long. |
| constexpr char kDownloadChromeUrl[] = |
| "https://www.google.com/chrome/?&brand=CHWL" |
| "&utm_campaign=en&utm_source=en-et-na-us-chrome-bubble&utm_medium=et"; |
| |
| // The maximum number of ignored bubble we track in the NumLaterPerReinstall |
| // histogram. |
| constexpr int kMaxIgnored = 50; |
| // The number of buckets we want the NumLaterPerReinstall histogram to use. |
| constexpr int kNumIgnoredBuckets = 5; |
| |
| bool g_upgrade_bubble_is_showing = false; |
| |
| // The number of times the user ignored the bubble before finally choosing to |
| // reinstall. |
| int g_num_ignored_bubbles = 0; |
| |
| void OnWindowClosing() { |
| g_upgrade_bubble_is_showing = false; |
| |
| // Increment the ignored bubble count (if this bubble wasn't ignored, this |
| // increment is offset by a decrement in OnDialogAccepted()). |
| if (g_num_ignored_bubbles < kMaxIgnored) |
| ++g_num_ignored_bubbles; |
| } |
| |
| void OnDialogAccepted(content::PageNavigator* navigator, |
| bool auto_update_enabled) { |
| // Offset the +1 in OnWindowClosing(). |
| --g_num_ignored_bubbles; |
| if (auto_update_enabled) { |
| DCHECK(UpgradeDetector::GetInstance()->is_outdated_install()); |
| UMA_HISTOGRAM_CUSTOM_COUNTS("OutdatedUpgradeBubble.NumLaterPerReinstall", |
| g_num_ignored_bubbles, 1, kMaxIgnored, |
| kNumIgnoredBuckets); |
| base::RecordAction( |
| base::UserMetricsAction("OutdatedUpgradeBubble.Reinstall")); |
| navigator->OpenURL( |
| content::OpenURLParams(GURL(kDownloadChromeUrl), content::Referrer(), |
| WindowOpenDisposition::NEW_FOREGROUND_TAB, |
| ui::PAGE_TRANSITION_LINK, false)); |
| #if defined(OS_WIN) |
| } else { |
| DCHECK(UpgradeDetector::GetInstance()->is_outdated_install_no_au()); |
| UMA_HISTOGRAM_CUSTOM_COUNTS("OutdatedUpgradeBubble.NumLaterPerEnableAU", |
| g_num_ignored_bubbles, 1, kMaxIgnored, |
| kNumIgnoredBuckets); |
| base::RecordAction( |
| base::UserMetricsAction("OutdatedUpgradeBubble.EnableAU")); |
| // Record that the autoupdate flavour of the dialog has been shown. |
| if (g_browser_process->local_state()) { |
| g_browser_process->local_state()->SetBoolean( |
| prefs::kAttemptedToEnableAutoupdate, true); |
| } |
| |
| // Re-enable updates by shelling out to setup.exe asynchronously. |
| base::ThreadPool::PostTask( |
| FROM_HERE, |
| {base::MayBlock(), base::TaskPriority::BEST_EFFORT, |
| base::TaskShutdownBehavior::BLOCK_SHUTDOWN}, |
| base::BindOnce(&google_update::ElevateIfNeededToReenableUpdates)); |
| #endif // defined(OS_WIN) |
| } |
| } |
| |
| } // namespace |
| |
| // OutdatedUpgradeBubbleView --------------------------------------------------- |
| |
| // static |
| void OutdatedUpgradeBubbleView::ShowBubble(views::View* anchor_view, |
| content::PageNavigator* navigator, |
| bool auto_update_enabled) { |
| if (g_upgrade_bubble_is_showing) |
| return; |
| |
| g_upgrade_bubble_is_showing = true; |
| |
| auto dialog_model = |
| ui::DialogModel::Builder() |
| .SetTitle(l10n_util::GetStringUTF16(IDS_UPGRADE_BUBBLE_TITLE)) |
| .AddOkButton( |
| base::BindOnce(&OnDialogAccepted, navigator, auto_update_enabled), |
| l10n_util::GetStringUTF16(auto_update_enabled |
| ? IDS_REINSTALL_APP |
| : IDS_REENABLE_UPDATES)) |
| .AddBodyText( |
| ui::DialogModelLabel(IDS_UPGRADE_BUBBLE_TEXT).set_is_secondary()) |
| .SetWindowClosingCallback(base::BindOnce(&OnWindowClosing)) |
| .SetCloseCallback(base::BindOnce( |
| &base::RecordAction, |
| base::UserMetricsAction("OutdatedUpgradeBubble.Later"))) |
| .Build(); |
| |
| auto bubble = std::make_unique<views::BubbleDialogModelHost>( |
| std::move(dialog_model), anchor_view, views::BubbleBorder::TOP_RIGHT); |
| views::BubbleDialogDelegateView::CreateBubble(std::move(bubble))->Show(); |
| |
| chrome::RecordDialogCreation(chrome::DialogIdentifier::OUTDATED_UPGRADE); |
| |
| base::RecordAction( |
| auto_update_enabled |
| ? base::UserMetricsAction("OutdatedUpgradeBubble.Show") |
| : base::UserMetricsAction("OutdatedUpgradeBubble.ShowNoAU")); |
| } |