Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(5345)

Unified Diff: chrome/browser/ui/views/srt_prompt_dialog.cc

Issue 2795133002: Chrome Cleaner UI: Add a dialog class for the new UI (Closed)
Patch Set: Addressed Fabio's comments Created 3 years, 8 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/ui/views/srt_prompt_dialog.cc
diff --git a/chrome/browser/ui/views/srt_prompt_dialog.cc b/chrome/browser/ui/views/srt_prompt_dialog.cc
new file mode 100644
index 0000000000000000000000000000000000000000..f1887d30fdbfdcbf9fe99d42b02c176159617b71
--- /dev/null
+++ b/chrome/browser/ui/views/srt_prompt_dialog.cc
@@ -0,0 +1,346 @@
+// Copyright 2017 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/srt_prompt_dialog.h"
+
+#include <vector>
+
+#include "base/strings/string16.h"
+#include "chrome/app/vector_icons/vector_icons.h"
+#include "chrome/browser/safe_browsing/srt_prompt_controller.h"
+#include "chrome/browser/ui/browser.h"
+#include "chrome/browser/ui/chrome_web_modal_dialog_manager_delegate.h"
+#include "chrome/browser/ui/views/frame/browser_view.h"
+#include "components/constrained_window/constrained_window_views.h"
+#include "components/web_modal/web_contents_modal_dialog_host.h"
+#include "ui/base/ui_base_types.h"
+#include "ui/events/event.h"
+#include "ui/gfx/geometry/size.h"
+#include "ui/gfx/native_widget_types.h"
+#include "ui/gfx/paint_vector_icon.h"
+#include "ui/gfx/text_constants.h"
+#include "ui/native_theme/native_theme.h"
+#include "ui/views/controls/label.h"
+#include "ui/views/controls/scroll_view.h"
+#include "ui/views/controls/separator.h"
+#include "ui/views/layout/box_layout.h"
+#include "ui/views/layout/grid_layout.h"
+#include "ui/views/layout/layout_constants.h"
+#include "ui/views/widget/widget.h"
+
+namespace safe_browsing {
+
+// static
+void SRTPromptController::ShowSRTPrompt(Browser* browser,
+ SRTPromptController* controller) {
+ SRTPromptDialog* dialog = new SRTPromptDialog(controller);
+ dialog->Show(browser);
+}
+
+} // namespace safe_browsing
+
+namespace {
+
+using LabelInfo = safe_browsing::SRTPromptController::LabelInfo;
+
+constexpr int kDialogWidth = 448;
+constexpr int kDetailsSectionMaxHeight = 150;
+constexpr int kBulletColumnWidth = 10;
+// Constants used for the layout of the label views.
+static constexpr int kMainColumSetId = 0;
sky 2017/04/05 21:17:42 Be consistent. No need for the static here.
alito 2017/04/06 00:20:22 Done.
+static constexpr int kBulletColumnSetId = 1;
+
+// Returns a view containing |item| with insets defined by |top|, |left|,
+// |bottom|, and |right|.
+views::View* CreateViewWithInsets(views::View* item,
+ int top,
+ int left,
+ int bottom,
+ int right) {
+ views::View* view = new views::View();
+ views::GridLayout* layout = new views::GridLayout(view);
+ view->SetLayoutManager(layout);
+ layout->SetInsets(top, left, bottom, right);
+ layout->AddColumnSet(0)->AddColumn(
sky 2017/04/05 21:17:42 GridLayout seems like overkill here. Isn't this th
alito 2017/04/06 00:20:22 Indeed. I had tried adding an empty border to the
+ views::GridLayout::LEADING, views::GridLayout::LEADING,
+ /*resize_percent=*/1, views::GridLayout::USE_PREF,
+ /*fixed_width=*/0,
+ /*min_width=*/0);
+ layout->StartRow(0, 0);
+ layout->AddView(item);
+
+ return view;
+}
+
+// Helper function used by |CreateLabelView()| below that adds |labels| to
+// |label_view|.
+void AddLabelsToLabelView(views::View* label_view,
+ views::GridLayout* layout,
+ const std::vector<LabelInfo>& labels) {
+ static constexpr base::char16 kBulletPoint[] = {0x2022, 0};
+
+ bool first_label = true;
+ bool last_label_was_bullet = false;
+ for (const LabelInfo& label_info : labels) {
+ const bool is_bullet = label_info.type == LabelInfo::BULLET_ITEM;
+ views::Label* label = new views::Label(label_info.text);
+ label->SetMultiLine(true);
+ label->SetHorizontalAlignment(gfx::ALIGN_LEFT);
+
+ // Do not add a padding row if
+ // - this is the first label being added, or
+ // - a bullet item is being added and the last label was also a bullet item.
+ bool skip_padding = first_label || (is_bullet && last_label_was_bullet);
+ if (!skip_padding)
+ layout->AddPaddingRow(0, views::kRelatedControlVerticalSpacing);
+
+ layout->StartRow(0, is_bullet ? kBulletColumnSetId : kMainColumSetId);
+ if (is_bullet) {
+ views::Label* bullet = new views::Label(base::string16(kBulletPoint));
+ layout->AddView(bullet);
+ }
+ layout->AddView(label);
+
+ last_label_was_bullet = is_bullet;
+ first_label = false;
+ }
+}
+
+// Creates a view that displays two types of labels: multiline labels
+// representing whole paragraphs or indented labels that are start with a bullet
+// point.
+//
+// A vertical padding of size |top_vertical_space| is added before the labels.
+views::View* CreateLabelView(int top_vertical_space,
+ const std::vector<LabelInfo>& labels) {
+ views::View* label_view = new views::View();
+ views::GridLayout* layout = new views::GridLayout(label_view);
+ layout->SetInsets(0, views::kButtonHEdgeMarginNew, 0,
+ views::kButtonHEdgeMarginNew);
+
+ label_view->SetLayoutManager(layout);
+
+ views::ColumnSet* main_column_set = layout->AddColumnSet(kMainColumSetId);
+ main_column_set->AddColumn(views::GridLayout::FILL,
+ views::GridLayout::LEADING,
+ /*resize_percent=*/1, views::GridLayout::USE_PREF,
+ /*fixed_width=*/0,
+ /*min_width=*/0);
+
+ views::ColumnSet* bullet_column_set_ =
+ layout->AddColumnSet(kBulletColumnSetId);
+ bullet_column_set_->AddPaddingColumn(
+ /*resize_percent=*/0, views::kUnrelatedControlLargeHorizontalSpacing);
+ bullet_column_set_->AddColumn(
+ views::GridLayout::FILL, views::GridLayout::LEADING,
+ /*resize_percent=*/0, views::GridLayout::USE_PREF,
+ /*fixed_width=*/0,
+ /*min_width=*/0);
+ bullet_column_set_->AddPaddingColumn(/*resize_percent=*/0,
+ kBulletColumnWidth);
+ bullet_column_set_->AddColumn(
+ views::GridLayout::FILL, views::GridLayout::LEADING,
+ /*resize_percent=*/1, views::GridLayout::USE_PREF,
+ /*fixed_width=*/0,
+ /*min_width=*/0);
+
+ if (top_vertical_space > 0)
+ layout->AddPaddingRow(/*vertical_resize=*/0, top_vertical_space);
+
+ AddLabelsToLabelView(label_view, layout, labels);
+
+ layout->AddPaddingRow(/*vertical_resize=*/0,
+ views::kUnrelatedControlLargeHorizontalSpacing);
+ return label_view;
+}
+
+} // namespace
+
+////////////////////////////////////////////////////////////////////////////////
+// SRTPromptDialog::ExpandableMessageView
+//
+// A view, whose visibilty can be toggled, and will be used for the details
+// section the main dialog.
+class SRTPromptDialog::ExpandableMessageView : public views::View {
sky 2017/04/05 21:17:42 It seems like you're subclassing here for convenie
alito 2017/04/06 00:20:22 I plan to add animation to the expansion and foldi
sky 2017/04/06 02:10:02 It's easy to add it back when you get there, vs on
alito 2017/04/06 03:37:46 Fair enough. I've changed this to populate the det
+ public:
+ explicit ExpandableMessageView(const std::vector<LabelInfo>& labels);
+ ~ExpandableMessageView() override;
+
+ void ToggleShowView();
+
+ // views::View overrides.
+ gfx::Size GetPreferredSize() const override;
+ int GetHeightForWidth(int width) const override;
+
+ private:
+ DISALLOW_COPY_AND_ASSIGN(ExpandableMessageView);
+};
+
+SRTPromptDialog::ExpandableMessageView::ExpandableMessageView(
+ const std::vector<LabelInfo>& labels) {
+ views::GridLayout* layout = new views::GridLayout(this);
+ SetLayoutManager(layout);
+
+ SetVisible(false);
+
+ constexpr int kColumnId = 0;
+ views::ColumnSet* column_set = layout->AddColumnSet(kColumnId);
+ column_set->AddColumn(views::GridLayout::FILL, views::GridLayout::LEADING,
+ /*resize_percent=*/1, views::GridLayout::USE_PREF,
+ /*fixed_width=*/0,
+ /*min_width=*/0);
+
+ // Add the main message view inside a scroll view.
+ views::View* label_view =
+ CreateLabelView(views::kUnrelatedControlLargeHorizontalSpacing, labels);
+ views::ScrollView* scroll_view = new views::ScrollView();
+ scroll_view->ClipHeightTo(/*min_height=*/0, kDetailsSectionMaxHeight);
+ scroll_view->SetContents(label_view);
+ layout->StartRow(0, kColumnId);
+ layout->AddView(scroll_view);
+}
+
+SRTPromptDialog::ExpandableMessageView::~ExpandableMessageView() {}
+
+void SRTPromptDialog::ExpandableMessageView::ToggleShowView() {
+ SetVisible(!visible());
+ PreferredSizeChanged();
+}
+
+gfx::Size SRTPromptDialog::ExpandableMessageView::GetPreferredSize() const {
+ gfx::Size size = views::View::GetPreferredSize();
+ return gfx::Size(size.width(), visible() ? size.height() : 0);
+}
+
+int SRTPromptDialog::ExpandableMessageView::GetHeightForWidth(int width) const {
+ return GetPreferredSize().height();
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// SRTPromptDialog
+
+SRTPromptDialog::SRTPromptDialog(safe_browsing::SRTPromptController* controller)
+ : browser_(nullptr),
+ controller_(controller),
+ interaction_done_(false),
+ details_view_(nullptr),
+ expand_details_button_(nullptr),
+ expand_details_button_color_(GetNativeTheme()->GetSystemColor(
sky 2017/04/05 21:17:42 This variables is only used in this function. No n
alito 2017/04/06 00:20:22 A private member function now returns the color (a
+ ui::NativeTheme::kColorId_LinkEnabled)),
+ expand_icon_(
sky 2017/04/05 21:17:42 Why do you need to cache these icons? Can you look
alito 2017/04/06 00:20:22 Changed to look them up as needed.
+ gfx::CreateVectorIcon(kCaretDownIcon, expand_details_button_color_)),
+ fold_icon_(
+ gfx::CreateVectorIcon(kCaretUpIcon, expand_details_button_color_)) {
+ DCHECK(controller_);
+
+ SetLayoutManager(new views::BoxLayout(
+ /*orientation=*/views::BoxLayout::kVertical,
+ /*inside_border_horizontal_spacing=*/0,
+ /*inside_border_vertical_spacing=*/views::kPanelVertMargin,
+ /*between_child_spacing=*/0));
+
+ AddChildView(CreateLabelView(0, controller_->GetMainText()));
+ AddChildView(new views::Separator());
+
+ details_view_ = new ExpandableMessageView(controller_->GetDetailsText());
+ AddChildView(details_view_);
+
+ expand_details_button_ =
+ new views::LabelButton(this, controller_->GetShowDetailsLabel());
+ expand_details_button_->SetEnabledTextColors(expand_details_button_color_);
+ expand_details_button_->SetImage(
+ views::Button::STATE_NORMAL,
+ details_view_->visible() ? fold_icon_ : expand_icon_);
+ AddChildView(CreateViewWithInsets(
+ expand_details_button_, views::kPanelVertMargin,
+ views::kButtonHEdgeMarginNew, views::kPanelVertMargin,
+ views::kButtonHEdgeMarginNew));
+ AddChildView(new views::Separator());
+}
+
+SRTPromptDialog::~SRTPromptDialog() {
+ // Make sure the controller is correctly notified in case the dialog widget is
+ // closed by some other means than the dialog buttons.
+ Close();
sky 2017/04/05 21:17:42 This class is owned by the widget and deleted when
alito 2017/04/06 00:20:22 This call ensures that the controller is notified
sky 2017/04/06 02:10:02 It's better to be explicit rather than relying on
alito 2017/04/06 03:37:46 Changed to explicitly notify the controller if nee
+}
+
+void SRTPromptDialog::Show(Browser* browser) {
+ DCHECK(browser);
sky 2017/04/05 21:17:42 DCHECK(!browser_)?
alito 2017/04/06 00:20:22 Done.
+
+ browser_ = browser;
+ BrowserView* browser_view = BrowserView::GetBrowserViewForBrowser(browser_);
sky 2017/04/05 21:17:42 Why do you need to get the BrowserView for browser
alito 2017/04/06 00:20:22 Ah! That's a shortcut I hadn't noticed.
+ constrained_window::CreateBrowserModalDialogViews(
+ this, browser_view->GetNativeWindow())
+ ->Show();
+ controller_->DialogShown();
+}
+
+// DialogModel overrides.
+
+bool SRTPromptDialog::ShouldDefaultButtonBeBlue() const {
+ return true;
+}
+
+// WidgetDelegate overrides.
+
+ui::ModalType SRTPromptDialog::GetModalType() const {
+ return ui::MODAL_TYPE_WINDOW;
+}
+
+base::string16 SRTPromptDialog::GetWindowTitle() const {
+ return controller_->GetWindowTitle();
+}
+
+// DialogDelegate overrides.
+
+base::string16 SRTPromptDialog::GetDialogButtonLabel(
+ ui::DialogButton button) const {
+ DCHECK(button == ui::DIALOG_BUTTON_OK || button == ui::DIALOG_BUTTON_CANCEL);
+
+ if (button == ui::DIALOG_BUTTON_OK)
+ return controller_->GetAcceptButtonLabel();
+ return DialogDelegate::GetDialogButtonLabel(button);
+}
+
+bool SRTPromptDialog::Accept() {
+ if (!interaction_done_) {
+ interaction_done_ = true;
+ controller_->Accept();
+ }
+ return true;
+}
+
+bool SRTPromptDialog::Cancel() {
+ if (!interaction_done_) {
sky 2017/04/05 21:17:42 If the underlying widget is destroyed outside of c
alito 2017/04/06 00:20:22 This is already handled in the destructor, which c
sky 2017/04/06 02:10:02 Acknowledged.
+ interaction_done_ = true;
+ controller_->Cancel();
+ }
+ return true;
+}
+
+// View overrides.
+
+gfx::Size SRTPromptDialog::GetPreferredSize() const {
+ return gfx::Size(kDialogWidth, GetHeightForWidth(kDialogWidth));
+}
+
+// views::ButtonListener overrides.
+
+void SRTPromptDialog::ButtonPressed(views::Button* sender,
+ const ui::Event& event) {
+ DCHECK_EQ(sender, expand_details_button_);
+ DCHECK(browser_);
+
+ details_view_->ToggleShowView();
+ expand_details_button_->SetText(details_view_->visible()
+ ? controller_->GetHideDetailsLabel()
+ : controller_->GetShowDetailsLabel());
+ expand_details_button_->SetImage(
+ views::Button::STATE_NORMAL,
+ details_view_->visible() ? fold_icon_ : expand_icon_);
+
+ ChromeWebModalDialogManagerDelegate* manager = browser_;
+ constrained_window::UpdateWidgetModalDialogPosition(
+ GetWidget(), manager->GetWebContentsModalDialogHost());
+}

Powered by Google App Engine
This is Rietveld 408576698