Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2017 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "chrome/browser/ui/views/srt_prompt_dialog.h" | |
| 6 | |
| 7 #include <vector> | |
| 8 | |
| 9 #include "base/strings/string16.h" | |
| 10 #include "chrome/app/vector_icons/vector_icons.h" | |
| 11 #include "chrome/browser/safe_browsing/srt_prompt_controller.h" | |
| 12 #include "chrome/browser/ui/browser.h" | |
| 13 #include "chrome/browser/ui/chrome_web_modal_dialog_manager_delegate.h" | |
| 14 #include "chrome/browser/ui/views/frame/browser_view.h" | |
| 15 #include "components/constrained_window/constrained_window_views.h" | |
| 16 #include "components/web_modal/web_contents_modal_dialog_host.h" | |
| 17 #include "ui/base/ui_base_types.h" | |
| 18 #include "ui/events/event.h" | |
| 19 #include "ui/gfx/geometry/size.h" | |
| 20 #include "ui/gfx/native_widget_types.h" | |
| 21 #include "ui/gfx/paint_vector_icon.h" | |
| 22 #include "ui/gfx/text_constants.h" | |
| 23 #include "ui/native_theme/native_theme.h" | |
| 24 #include "ui/views/controls/label.h" | |
| 25 #include "ui/views/controls/scroll_view.h" | |
| 26 #include "ui/views/controls/separator.h" | |
| 27 #include "ui/views/layout/box_layout.h" | |
| 28 #include "ui/views/layout/grid_layout.h" | |
| 29 #include "ui/views/layout/layout_constants.h" | |
| 30 #include "ui/views/widget/widget.h" | |
| 31 | |
| 32 namespace safe_browsing { | |
| 33 | |
| 34 // static | |
| 35 void SRTPromptController::ShowSRTPrompt(Browser* browser, | |
| 36 SRTPromptController* controller) { | |
| 37 SRTPromptDialog* dialog = new SRTPromptDialog(controller); | |
| 38 dialog->Show(browser); | |
| 39 } | |
| 40 | |
| 41 } // namespace safe_browsing | |
| 42 | |
| 43 namespace { | |
| 44 | |
| 45 using LabelInfo = safe_browsing::SRTPromptController::LabelInfo; | |
| 46 | |
| 47 constexpr int kDialogWidth = 448; | |
| 48 constexpr int kDetailsSectionMaxHeight = 150; | |
| 49 constexpr int kBulletColumnWidth = 10; | |
| 50 // Constants used for the layout of the label views. | |
| 51 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.
| |
| 52 static constexpr int kBulletColumnSetId = 1; | |
| 53 | |
| 54 // Returns a view containing |item| with insets defined by |top|, |left|, | |
| 55 // |bottom|, and |right|. | |
| 56 views::View* CreateViewWithInsets(views::View* item, | |
| 57 int top, | |
| 58 int left, | |
| 59 int bottom, | |
| 60 int right) { | |
| 61 views::View* view = new views::View(); | |
| 62 views::GridLayout* layout = new views::GridLayout(view); | |
| 63 view->SetLayoutManager(layout); | |
| 64 layout->SetInsets(top, left, bottom, right); | |
| 65 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
| |
| 66 views::GridLayout::LEADING, views::GridLayout::LEADING, | |
| 67 /*resize_percent=*/1, views::GridLayout::USE_PREF, | |
| 68 /*fixed_width=*/0, | |
| 69 /*min_width=*/0); | |
| 70 layout->StartRow(0, 0); | |
| 71 layout->AddView(item); | |
| 72 | |
| 73 return view; | |
| 74 } | |
| 75 | |
| 76 // Helper function used by |CreateLabelView()| below that adds |labels| to | |
| 77 // |label_view|. | |
| 78 void AddLabelsToLabelView(views::View* label_view, | |
| 79 views::GridLayout* layout, | |
| 80 const std::vector<LabelInfo>& labels) { | |
| 81 static constexpr base::char16 kBulletPoint[] = {0x2022, 0}; | |
| 82 | |
| 83 bool first_label = true; | |
| 84 bool last_label_was_bullet = false; | |
| 85 for (const LabelInfo& label_info : labels) { | |
| 86 const bool is_bullet = label_info.type == LabelInfo::BULLET_ITEM; | |
| 87 views::Label* label = new views::Label(label_info.text); | |
| 88 label->SetMultiLine(true); | |
| 89 label->SetHorizontalAlignment(gfx::ALIGN_LEFT); | |
| 90 | |
| 91 // Do not add a padding row if | |
| 92 // - this is the first label being added, or | |
| 93 // - a bullet item is being added and the last label was also a bullet item. | |
| 94 bool skip_padding = first_label || (is_bullet && last_label_was_bullet); | |
| 95 if (!skip_padding) | |
| 96 layout->AddPaddingRow(0, views::kRelatedControlVerticalSpacing); | |
| 97 | |
| 98 layout->StartRow(0, is_bullet ? kBulletColumnSetId : kMainColumSetId); | |
| 99 if (is_bullet) { | |
| 100 views::Label* bullet = new views::Label(base::string16(kBulletPoint)); | |
| 101 layout->AddView(bullet); | |
| 102 } | |
| 103 layout->AddView(label); | |
| 104 | |
| 105 last_label_was_bullet = is_bullet; | |
| 106 first_label = false; | |
| 107 } | |
| 108 } | |
| 109 | |
| 110 // Creates a view that displays two types of labels: multiline labels | |
| 111 // representing whole paragraphs or indented labels that are start with a bullet | |
| 112 // point. | |
| 113 // | |
| 114 // A vertical padding of size |top_vertical_space| is added before the labels. | |
| 115 views::View* CreateLabelView(int top_vertical_space, | |
| 116 const std::vector<LabelInfo>& labels) { | |
| 117 views::View* label_view = new views::View(); | |
| 118 views::GridLayout* layout = new views::GridLayout(label_view); | |
| 119 layout->SetInsets(0, views::kButtonHEdgeMarginNew, 0, | |
| 120 views::kButtonHEdgeMarginNew); | |
| 121 | |
| 122 label_view->SetLayoutManager(layout); | |
| 123 | |
| 124 views::ColumnSet* main_column_set = layout->AddColumnSet(kMainColumSetId); | |
| 125 main_column_set->AddColumn(views::GridLayout::FILL, | |
| 126 views::GridLayout::LEADING, | |
| 127 /*resize_percent=*/1, views::GridLayout::USE_PREF, | |
| 128 /*fixed_width=*/0, | |
| 129 /*min_width=*/0); | |
| 130 | |
| 131 views::ColumnSet* bullet_column_set_ = | |
| 132 layout->AddColumnSet(kBulletColumnSetId); | |
| 133 bullet_column_set_->AddPaddingColumn( | |
| 134 /*resize_percent=*/0, views::kUnrelatedControlLargeHorizontalSpacing); | |
| 135 bullet_column_set_->AddColumn( | |
| 136 views::GridLayout::FILL, views::GridLayout::LEADING, | |
| 137 /*resize_percent=*/0, views::GridLayout::USE_PREF, | |
| 138 /*fixed_width=*/0, | |
| 139 /*min_width=*/0); | |
| 140 bullet_column_set_->AddPaddingColumn(/*resize_percent=*/0, | |
| 141 kBulletColumnWidth); | |
| 142 bullet_column_set_->AddColumn( | |
| 143 views::GridLayout::FILL, views::GridLayout::LEADING, | |
| 144 /*resize_percent=*/1, views::GridLayout::USE_PREF, | |
| 145 /*fixed_width=*/0, | |
| 146 /*min_width=*/0); | |
| 147 | |
| 148 if (top_vertical_space > 0) | |
| 149 layout->AddPaddingRow(/*vertical_resize=*/0, top_vertical_space); | |
| 150 | |
| 151 AddLabelsToLabelView(label_view, layout, labels); | |
| 152 | |
| 153 layout->AddPaddingRow(/*vertical_resize=*/0, | |
| 154 views::kUnrelatedControlLargeHorizontalSpacing); | |
| 155 return label_view; | |
| 156 } | |
| 157 | |
| 158 } // namespace | |
| 159 | |
| 160 //////////////////////////////////////////////////////////////////////////////// | |
| 161 // SRTPromptDialog::ExpandableMessageView | |
| 162 // | |
| 163 // A view, whose visibilty can be toggled, and will be used for the details | |
| 164 // section the main dialog. | |
| 165 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
| |
| 166 public: | |
| 167 explicit ExpandableMessageView(const std::vector<LabelInfo>& labels); | |
| 168 ~ExpandableMessageView() override; | |
| 169 | |
| 170 void ToggleShowView(); | |
| 171 | |
| 172 // views::View overrides. | |
| 173 gfx::Size GetPreferredSize() const override; | |
| 174 int GetHeightForWidth(int width) const override; | |
| 175 | |
| 176 private: | |
| 177 DISALLOW_COPY_AND_ASSIGN(ExpandableMessageView); | |
| 178 }; | |
| 179 | |
| 180 SRTPromptDialog::ExpandableMessageView::ExpandableMessageView( | |
| 181 const std::vector<LabelInfo>& labels) { | |
| 182 views::GridLayout* layout = new views::GridLayout(this); | |
| 183 SetLayoutManager(layout); | |
| 184 | |
| 185 SetVisible(false); | |
| 186 | |
| 187 constexpr int kColumnId = 0; | |
| 188 views::ColumnSet* column_set = layout->AddColumnSet(kColumnId); | |
| 189 column_set->AddColumn(views::GridLayout::FILL, views::GridLayout::LEADING, | |
| 190 /*resize_percent=*/1, views::GridLayout::USE_PREF, | |
| 191 /*fixed_width=*/0, | |
| 192 /*min_width=*/0); | |
| 193 | |
| 194 // Add the main message view inside a scroll view. | |
| 195 views::View* label_view = | |
| 196 CreateLabelView(views::kUnrelatedControlLargeHorizontalSpacing, labels); | |
| 197 views::ScrollView* scroll_view = new views::ScrollView(); | |
| 198 scroll_view->ClipHeightTo(/*min_height=*/0, kDetailsSectionMaxHeight); | |
| 199 scroll_view->SetContents(label_view); | |
| 200 layout->StartRow(0, kColumnId); | |
| 201 layout->AddView(scroll_view); | |
| 202 } | |
| 203 | |
| 204 SRTPromptDialog::ExpandableMessageView::~ExpandableMessageView() {} | |
| 205 | |
| 206 void SRTPromptDialog::ExpandableMessageView::ToggleShowView() { | |
| 207 SetVisible(!visible()); | |
| 208 PreferredSizeChanged(); | |
| 209 } | |
| 210 | |
| 211 gfx::Size SRTPromptDialog::ExpandableMessageView::GetPreferredSize() const { | |
| 212 gfx::Size size = views::View::GetPreferredSize(); | |
| 213 return gfx::Size(size.width(), visible() ? size.height() : 0); | |
| 214 } | |
| 215 | |
| 216 int SRTPromptDialog::ExpandableMessageView::GetHeightForWidth(int width) const { | |
| 217 return GetPreferredSize().height(); | |
| 218 } | |
| 219 | |
| 220 //////////////////////////////////////////////////////////////////////////////// | |
| 221 // SRTPromptDialog | |
| 222 | |
| 223 SRTPromptDialog::SRTPromptDialog(safe_browsing::SRTPromptController* controller) | |
| 224 : browser_(nullptr), | |
| 225 controller_(controller), | |
| 226 interaction_done_(false), | |
| 227 details_view_(nullptr), | |
| 228 expand_details_button_(nullptr), | |
| 229 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
| |
| 230 ui::NativeTheme::kColorId_LinkEnabled)), | |
| 231 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.
| |
| 232 gfx::CreateVectorIcon(kCaretDownIcon, expand_details_button_color_)), | |
| 233 fold_icon_( | |
| 234 gfx::CreateVectorIcon(kCaretUpIcon, expand_details_button_color_)) { | |
| 235 DCHECK(controller_); | |
| 236 | |
| 237 SetLayoutManager(new views::BoxLayout( | |
| 238 /*orientation=*/views::BoxLayout::kVertical, | |
| 239 /*inside_border_horizontal_spacing=*/0, | |
| 240 /*inside_border_vertical_spacing=*/views::kPanelVertMargin, | |
| 241 /*between_child_spacing=*/0)); | |
| 242 | |
| 243 AddChildView(CreateLabelView(0, controller_->GetMainText())); | |
| 244 AddChildView(new views::Separator()); | |
| 245 | |
| 246 details_view_ = new ExpandableMessageView(controller_->GetDetailsText()); | |
| 247 AddChildView(details_view_); | |
| 248 | |
| 249 expand_details_button_ = | |
| 250 new views::LabelButton(this, controller_->GetShowDetailsLabel()); | |
| 251 expand_details_button_->SetEnabledTextColors(expand_details_button_color_); | |
| 252 expand_details_button_->SetImage( | |
| 253 views::Button::STATE_NORMAL, | |
| 254 details_view_->visible() ? fold_icon_ : expand_icon_); | |
| 255 AddChildView(CreateViewWithInsets( | |
| 256 expand_details_button_, views::kPanelVertMargin, | |
| 257 views::kButtonHEdgeMarginNew, views::kPanelVertMargin, | |
| 258 views::kButtonHEdgeMarginNew)); | |
| 259 AddChildView(new views::Separator()); | |
| 260 } | |
| 261 | |
| 262 SRTPromptDialog::~SRTPromptDialog() { | |
| 263 // Make sure the controller is correctly notified in case the dialog widget is | |
| 264 // closed by some other means than the dialog buttons. | |
| 265 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
| |
| 266 } | |
| 267 | |
| 268 void SRTPromptDialog::Show(Browser* browser) { | |
| 269 DCHECK(browser); | |
|
sky
2017/04/05 21:17:42
DCHECK(!browser_)?
alito
2017/04/06 00:20:22
Done.
| |
| 270 | |
| 271 browser_ = browser; | |
| 272 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.
| |
| 273 constrained_window::CreateBrowserModalDialogViews( | |
| 274 this, browser_view->GetNativeWindow()) | |
| 275 ->Show(); | |
| 276 controller_->DialogShown(); | |
| 277 } | |
| 278 | |
| 279 // DialogModel overrides. | |
| 280 | |
| 281 bool SRTPromptDialog::ShouldDefaultButtonBeBlue() const { | |
| 282 return true; | |
| 283 } | |
| 284 | |
| 285 // WidgetDelegate overrides. | |
| 286 | |
| 287 ui::ModalType SRTPromptDialog::GetModalType() const { | |
| 288 return ui::MODAL_TYPE_WINDOW; | |
| 289 } | |
| 290 | |
| 291 base::string16 SRTPromptDialog::GetWindowTitle() const { | |
| 292 return controller_->GetWindowTitle(); | |
| 293 } | |
| 294 | |
| 295 // DialogDelegate overrides. | |
| 296 | |
| 297 base::string16 SRTPromptDialog::GetDialogButtonLabel( | |
| 298 ui::DialogButton button) const { | |
| 299 DCHECK(button == ui::DIALOG_BUTTON_OK || button == ui::DIALOG_BUTTON_CANCEL); | |
| 300 | |
| 301 if (button == ui::DIALOG_BUTTON_OK) | |
| 302 return controller_->GetAcceptButtonLabel(); | |
| 303 return DialogDelegate::GetDialogButtonLabel(button); | |
| 304 } | |
| 305 | |
| 306 bool SRTPromptDialog::Accept() { | |
| 307 if (!interaction_done_) { | |
| 308 interaction_done_ = true; | |
| 309 controller_->Accept(); | |
| 310 } | |
| 311 return true; | |
| 312 } | |
| 313 | |
| 314 bool SRTPromptDialog::Cancel() { | |
| 315 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.
| |
| 316 interaction_done_ = true; | |
| 317 controller_->Cancel(); | |
| 318 } | |
| 319 return true; | |
| 320 } | |
| 321 | |
| 322 // View overrides. | |
| 323 | |
| 324 gfx::Size SRTPromptDialog::GetPreferredSize() const { | |
| 325 return gfx::Size(kDialogWidth, GetHeightForWidth(kDialogWidth)); | |
| 326 } | |
| 327 | |
| 328 // views::ButtonListener overrides. | |
| 329 | |
| 330 void SRTPromptDialog::ButtonPressed(views::Button* sender, | |
| 331 const ui::Event& event) { | |
| 332 DCHECK_EQ(sender, expand_details_button_); | |
| 333 DCHECK(browser_); | |
| 334 | |
| 335 details_view_->ToggleShowView(); | |
| 336 expand_details_button_->SetText(details_view_->visible() | |
| 337 ? controller_->GetHideDetailsLabel() | |
| 338 : controller_->GetShowDetailsLabel()); | |
| 339 expand_details_button_->SetImage( | |
| 340 views::Button::STATE_NORMAL, | |
| 341 details_view_->visible() ? fold_icon_ : expand_icon_); | |
| 342 | |
| 343 ChromeWebModalDialogManagerDelegate* manager = browser_; | |
| 344 constrained_window::UpdateWidgetModalDialogPosition( | |
| 345 GetWidget(), manager->GetWebContentsModalDialogHost()); | |
| 346 } | |
| OLD | NEW |