Index: chrome/browser/ui/views/recovery_component_bubble_view.cc |
diff --git a/chrome/browser/ui/views/recovery_component_bubble_view.cc b/chrome/browser/ui/views/recovery_component_bubble_view.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..7c191354b1c1f2274f8e9c7f51f1992b3193e877 |
--- /dev/null |
+++ b/chrome/browser/ui/views/recovery_component_bubble_view.cc |
@@ -0,0 +1,186 @@ |
+// Copyright (c) 2014 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/recovery_component_bubble_view.h" |
+ |
+#include "base/bind.h" |
+#include "chrome/browser/browser_process.h" |
+#include "chrome/browser/component_updater/recovery_component_installer.h" |
+#include "chrome/browser/ui/views/elevation_icon_setter.h" |
+#include "chrome/browser/upgrade_detector.h" |
+#include "content/public/browser/user_metrics.h" |
+#include "grit/chromium_strings.h" |
+#include "grit/generated_resources.h" |
+#include "grit/theme_resources.h" |
+#include "ui/base/l10n/l10n_util.h" |
+#include "ui/base/resource/resource_bundle.h" |
+#include "ui/views/controls/button/label_button.h" |
+#include "ui/views/controls/image_view.h" |
+#include "ui/views/controls/label.h" |
+#include "ui/views/layout/grid_layout.h" |
+#include "ui/views/layout/layout_constants.h" |
+#include "ui/views/widget/widget.h" |
+ |
+namespace { |
+ |
+// Fixed width of the column holding the description label of the bubble. |
+const int kWidthOfDescriptionText = 350; |
+ |
+// We subtract 2 to account for the natural button padding, and |
+// to bring the separation visually in line with the row separation |
+// height. |
+const int kButtonPadding = views::kRelatedButtonHSpacing - 2; |
+ |
+} // namespace |
+ |
+// RecoveryComponentBubbleView ------------------------------------------------- |
+ |
+RecoveryComponentBubbleView* |
+RecoveryComponentBubbleView::recovery_component_bubble_ = NULL; |
+ |
+// static |
+void RecoveryComponentBubbleView::ShowBubble(views::View* anchor_view) { |
+ if (IsShowing()) |
+ return; |
+ recovery_component_bubble_ = new RecoveryComponentBubbleView(anchor_view); |
+ views::BubbleDelegateView::CreateBubble(recovery_component_bubble_)->Show(); |
+ |
+ content::RecordAction( |
+ base::UserMetricsAction("RecoveryComponentBubble.Show")); |
+} |
+ |
+bool RecoveryComponentBubbleView::IsAvailable() { |
+// Right now only Windows platform is supported. |
+#if defined(OS_WIN) |
+ return true; |
+#else |
+ return false; |
+#endif |
+} |
+ |
+RecoveryComponentBubbleView::~RecoveryComponentBubbleView() { |
+ // Ensure |elevation_icon_setter_| is destroyed before |accept_button_|. |
+ elevation_icon_setter_.reset(); |
+} |
+ |
+views::View* RecoveryComponentBubbleView::GetInitiallyFocusedView() { |
+ return accept_button_; |
+} |
+ |
+void RecoveryComponentBubbleView::WindowClosing() { |
+ // Reset |recovery_component_bubble_| here, not in destructor, because |
+ // destruction is asynchronous and ShowBubble may be called before full |
+ // destruction and would attempt to show a bubble that is closing. |
+ DCHECK_EQ(recovery_component_bubble_, this); |
+ recovery_component_bubble_ = NULL; |
+} |
+ |
+void RecoveryComponentBubbleView::Init() { |
+ ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); |
+ accept_button_ = new views::LabelButton( |
+ this, l10n_util::GetStringUTF16(IDS_RUN_RECOVERY)); |
+ accept_button_->SetStyle(views::Button::STYLE_BUTTON); |
+ accept_button_->SetIsDefault(true); |
+ accept_button_->SetFontList(rb.GetFontList(ui::ResourceBundle::BoldFont)); |
+ elevation_icon_setter_.reset(new ElevationIconSetter(accept_button_)); |
+ |
+ decline_button_ = new views::LabelButton( |
+ this, l10n_util::GetStringUTF16(IDS_NO_THANKS)); |
+ decline_button_->SetStyle(views::Button::STYLE_BUTTON); |
+ |
+ views::Label* title_label = new views::Label( |
+ l10n_util::GetStringUTF16(IDS_RECOVERY_BUBBLE_TITLE)); |
+ title_label->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont)); |
+ title_label->SetHorizontalAlignment(gfx::ALIGN_LEFT); |
+ |
+ views::Label* text_label = new views::Label( |
+ l10n_util::GetStringUTF16(IDS_RECOVERY_BUBBLE_TEXT)); |
+ text_label->SetMultiLine(true); |
+ text_label->SetHorizontalAlignment(gfx::ALIGN_LEFT); |
+ |
+ views::ImageView* image_view = new views::ImageView(); |
+ image_view->SetImage(rb.GetImageSkiaNamed(IDR_UPDATE_MENU_SEVERITY_HIGH)); |
+ |
+ views::GridLayout* layout = new views::GridLayout(this); |
+ SetLayoutManager(layout); |
+ |
+ const int kIconTitleColumnSetId = 0; |
+ views::ColumnSet* cs = layout->AddColumnSet(kIconTitleColumnSetId); |
+ |
+ // Top (icon-title) row. |
+ cs->AddColumn(views::GridLayout::LEADING, views::GridLayout::CENTER, 0, |
+ views::GridLayout::USE_PREF, 0, 0); |
+ cs->AddPaddingColumn(0, views::kRelatedControlHorizontalSpacing); |
+ cs->AddColumn(views::GridLayout::FILL, views::GridLayout::CENTER, 0, |
+ views::GridLayout::USE_PREF, 0, 0); |
+ cs->AddPaddingColumn(1, views::kUnrelatedControlHorizontalSpacing); |
+ |
+ // Middle (text) row. |
+ const int kTextColumnSetId = 1; |
+ cs = layout->AddColumnSet(kTextColumnSetId); |
+ cs->AddColumn(views::GridLayout::FILL, views::GridLayout::FILL, 1, |
+ views::GridLayout::FIXED, kWidthOfDescriptionText, 0); |
+ |
+ // Bottom (buttons) row. |
+ const int kButtonsColumnSetId = 2; |
+ cs = layout->AddColumnSet(kButtonsColumnSetId); |
+ cs->AddPaddingColumn(1, views::kRelatedControlHorizontalSpacing); |
+ cs->AddColumn(views::GridLayout::LEADING, views::GridLayout::TRAILING, 0, |
+ views::GridLayout::USE_PREF, 0, 0); |
+ cs->AddPaddingColumn(0, kButtonPadding); |
+ cs->AddColumn(views::GridLayout::LEADING, views::GridLayout::TRAILING, 0, |
+ views::GridLayout::USE_PREF, 0, 0); |
+ |
+ layout->StartRow(0, kIconTitleColumnSetId); |
+ layout->AddView(image_view); |
+ layout->AddView(title_label); |
+ |
+ layout->AddPaddingRow(0, views::kRelatedControlSmallVerticalSpacing); |
+ layout->StartRow(0, kTextColumnSetId); |
+ layout->AddView(text_label); |
+ |
+ layout->AddPaddingRow(0, views::kUnrelatedControlVerticalSpacing); |
+ |
+ layout->StartRow(0, kButtonsColumnSetId); |
+ layout->AddView(accept_button_); |
+ layout->AddView(decline_button_); |
+ |
+ AddAccelerator(ui::Accelerator(ui::VKEY_RETURN, ui::EF_NONE)); |
+} |
+ |
+RecoveryComponentBubbleView::RecoveryComponentBubbleView( |
+ views::View* anchor_view) |
+ : BubbleDelegateView(anchor_view, views::BubbleBorder::TOP_RIGHT), |
+ accept_button_(NULL), |
+ decline_button_(NULL) { |
+ // Compensate for built-in vertical padding in the anchor view's image. |
+ set_anchor_view_insets(gfx::Insets(5, 0, 5, 0)); |
+} |
+ |
+void RecoveryComponentBubbleView::ButtonPressed( |
+ views::Button* sender, const ui::Event& event) { |
+ if (event.IsMouseEvent() && |
+ !(static_cast<const ui::MouseEvent*>(&event))->IsOnlyLeftMouseButton()) { |
+ return; |
+ } |
+ HandleButtonPressed(sender); |
+} |
+ |
+void RecoveryComponentBubbleView::HandleButtonPressed(views::Button* sender) { |
+ if (sender == accept_button_ || sender == decline_button_) { |
+ const bool elevation_agreed = (sender == accept_button_); |
+ component_updater::StartElevatedRecoveryProcess( |
+ g_browser_process->local_state(), |
+ elevation_agreed, |
+ base::Bind(&UpgradeDetector::CheckForUpgrade, |
+ base::Unretained(UpgradeDetector::GetInstance()))); |
+ content::RecordAction(base::UserMetricsAction( |
+ elevation_agreed ? "RecoveryComponentBubble.ElevationAgreed" |
+ : "RecoveryComponentBubble.ElevationDeclined")); |
+ } else { |
+ content::RecordAction(base::UserMetricsAction( |
+ "RecoveryComponentBubble.Ignored")); |
+ } |
+ GetWidget()->Close(); |
+} |