Home | History | Annotate | Download | only in infobars
      1 // Copyright (c) 2012 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/infobars/infobar_view.h"
      6 
      7 #include <algorithm>
      8 
      9 #include "base/memory/scoped_ptr.h"
     10 #include "base/strings/utf_string_conversions.h"
     11 #include "chrome/browser/ui/views/infobars/infobar_background.h"
     12 #include "components/infobars/core/infobar_delegate.h"
     13 #include "grit/generated_resources.h"
     14 #include "grit/theme_resources.h"
     15 #include "grit/ui_resources.h"
     16 #include "third_party/skia/include/effects/SkGradientShader.h"
     17 #include "ui/accessibility/ax_view_state.h"
     18 #include "ui/base/l10n/l10n_util.h"
     19 #include "ui/base/resource/resource_bundle.h"
     20 #include "ui/gfx/canvas.h"
     21 #include "ui/gfx/image/image.h"
     22 #include "ui/views/controls/button/image_button.h"
     23 #include "ui/views/controls/button/label_button.h"
     24 #include "ui/views/controls/button/label_button_border.h"
     25 #include "ui/views/controls/button/menu_button.h"
     26 #include "ui/views/controls/button/text_button.h"
     27 #include "ui/views/controls/image_view.h"
     28 #include "ui/views/controls/label.h"
     29 #include "ui/views/controls/link.h"
     30 #include "ui/views/controls/menu/menu_runner.h"
     31 #include "ui/views/layout/layout_constants.h"
     32 #include "ui/views/widget/widget.h"
     33 #include "ui/views/window/non_client_view.h"
     34 
     35 
     36 // Helpers --------------------------------------------------------------------
     37 
     38 namespace {
     39 
     40 const int kEdgeItemPadding = views::kRelatedControlHorizontalSpacing;
     41 const int kIconToLabelSpacing = views::kRelatedControlHorizontalSpacing;
     42 const int kBeforeCloseButtonSpacing = views::kUnrelatedControlHorizontalSpacing;
     43 
     44 bool SortLabelsByDecreasingWidth(views::Label* label_1, views::Label* label_2) {
     45   return label_1->GetPreferredSize().width() >
     46       label_2->GetPreferredSize().width();
     47 }
     48 
     49 }  // namespace
     50 
     51 
     52 // InfoBar --------------------------------------------------------------------
     53 
     54 // static
     55 const int infobars::InfoBar::kSeparatorLineHeight =
     56     views::NonClientFrameView::kClientEdgeThickness;
     57 const int infobars::InfoBar::kDefaultArrowTargetHeight = 9;
     58 const int infobars::InfoBar::kMaximumArrowTargetHeight = 24;
     59 const int infobars::InfoBar::kDefaultArrowTargetHalfWidth =
     60     kDefaultArrowTargetHeight;
     61 const int infobars::InfoBar::kMaximumArrowTargetHalfWidth = 14;
     62 const int infobars::InfoBar::kDefaultBarTargetHeight = 36;
     63 
     64 // InfoBarView ----------------------------------------------------------------
     65 
     66 // static
     67 const int InfoBarView::kButtonButtonSpacing = views::kRelatedButtonHSpacing;
     68 const int InfoBarView::kEndOfLabelSpacing = views::kItemLabelSpacing;
     69 
     70 InfoBarView::InfoBarView(scoped_ptr<infobars::InfoBarDelegate> delegate)
     71     : infobars::InfoBar(delegate.Pass()),
     72       views::ExternalFocusTracker(this, NULL),
     73       icon_(NULL),
     74       close_button_(NULL) {
     75   set_owned_by_client();  // InfoBar deletes itself at the appropriate time.
     76   set_background(
     77       new InfoBarBackground(infobars::InfoBar::delegate()->GetInfoBarType()));
     78 }
     79 
     80 InfoBarView::~InfoBarView() {
     81   // We should have closed any open menus in PlatformSpecificHide(), then
     82   // subclasses' RunMenu() functions should have prevented opening any new ones
     83   // once we became unowned.
     84   DCHECK(!menu_runner_.get());
     85 }
     86 
     87 views::Label* InfoBarView::CreateLabel(const base::string16& text) const {
     88   ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
     89   views::Label* label = new views::Label(
     90       text, rb.GetFontList(ui::ResourceBundle::MediumFont));
     91   label->SizeToPreferredSize();
     92   label->SetBackgroundColor(background()->get_color());
     93   label->SetEnabledColor(SK_ColorBLACK);
     94   label->SetHorizontalAlignment(gfx::ALIGN_LEFT);
     95   return label;
     96 }
     97 
     98 views::Link* InfoBarView::CreateLink(const base::string16& text,
     99                                      views::LinkListener* listener) const {
    100   ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
    101   views::Link* link = new views::Link(text);
    102   link->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont));
    103   link->SizeToPreferredSize();
    104   link->SetHorizontalAlignment(gfx::ALIGN_LEFT);
    105   link->set_listener(listener);
    106   link->SetBackgroundColor(background()->get_color());
    107   return link;
    108 }
    109 
    110 // static
    111 views::MenuButton* InfoBarView::CreateMenuButton(
    112     const base::string16& text,
    113     views::MenuButtonListener* menu_button_listener) {
    114   scoped_ptr<views::TextButtonDefaultBorder> menu_button_border(
    115       new views::TextButtonDefaultBorder());
    116   const int kNormalImageSet[] = IMAGE_GRID(IDR_INFOBARBUTTON_NORMAL);
    117   menu_button_border->set_normal_painter(
    118       views::Painter::CreateImageGridPainter(kNormalImageSet));
    119   const int kHotImageSet[] = IMAGE_GRID(IDR_INFOBARBUTTON_HOVER);
    120   menu_button_border->set_hot_painter(
    121       views::Painter::CreateImageGridPainter(kHotImageSet));
    122   const int kPushedImageSet[] = IMAGE_GRID(IDR_INFOBARBUTTON_PRESSED);
    123   menu_button_border->set_pushed_painter(
    124       views::Painter::CreateImageGridPainter(kPushedImageSet));
    125 
    126   views::MenuButton* menu_button = new views::MenuButton(
    127       NULL, text, menu_button_listener, true);
    128   menu_button->SetBorder(menu_button_border.PassAs<views::Border>());
    129   menu_button->set_animate_on_state_change(false);
    130   ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
    131   menu_button->set_menu_marker(
    132       rb.GetImageNamed(IDR_INFOBARBUTTON_MENU_DROPARROW).ToImageSkia());
    133   menu_button->SetTextColor(views::Button::STATE_NORMAL, SK_ColorBLACK);
    134   menu_button->SetTextColor(views::Button::STATE_HOVERED, SK_ColorBLACK);
    135   menu_button->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont));
    136   menu_button->SizeToPreferredSize();
    137   menu_button->SetFocusable(true);
    138   return menu_button;
    139 }
    140 
    141 // static
    142 views::LabelButton* InfoBarView::CreateLabelButton(
    143     views::ButtonListener* listener,
    144     const base::string16& text) {
    145   scoped_ptr<views::LabelButtonBorder> label_button_border(
    146       new views::LabelButtonBorder(views::Button::STYLE_TEXTBUTTON));
    147   const int kNormalImageSet[] = IMAGE_GRID(IDR_INFOBARBUTTON_NORMAL);
    148   label_button_border->SetPainter(
    149       false, views::Button::STATE_NORMAL,
    150       views::Painter::CreateImageGridPainter(kNormalImageSet));
    151   const int kHoveredImageSet[] = IMAGE_GRID(IDR_INFOBARBUTTON_HOVER);
    152   label_button_border->SetPainter(
    153       false, views::Button::STATE_HOVERED,
    154       views::Painter::CreateImageGridPainter(kHoveredImageSet));
    155   const int kPressedImageSet[] = IMAGE_GRID(IDR_INFOBARBUTTON_PRESSED);
    156   label_button_border->SetPainter(
    157       false, views::Button::STATE_PRESSED,
    158       views::Painter::CreateImageGridPainter(kPressedImageSet));
    159 
    160   views::LabelButton* label_button = new views::LabelButton(listener, text);
    161   label_button->SetBorder(label_button_border.PassAs<views::Border>());
    162   label_button->set_animate_on_state_change(false);
    163   label_button->SetTextColor(views::Button::STATE_NORMAL, SK_ColorBLACK);
    164   label_button->SetTextColor(views::Button::STATE_HOVERED, SK_ColorBLACK);
    165   ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
    166   label_button->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont));
    167   label_button->SizeToPreferredSize();
    168   label_button->SetFocusable(true);
    169   return label_button;
    170 }
    171 
    172 // static
    173 void InfoBarView::AssignWidths(Labels* labels, int available_width) {
    174   std::sort(labels->begin(), labels->end(), SortLabelsByDecreasingWidth);
    175   AssignWidthsSorted(labels, available_width);
    176 }
    177 
    178 void InfoBarView::Layout() {
    179   // Calculate the fill and stroke paths.  We do this here, rather than in
    180   // PlatformSpecificRecalculateHeight(), because this is also reached when our
    181   // width is changed, which affects both paths.
    182   stroke_path_.rewind();
    183   fill_path_.rewind();
    184   const infobars::InfoBarContainer::Delegate* delegate = container_delegate();
    185   if (delegate) {
    186     static_cast<InfoBarBackground*>(background())->set_separator_color(
    187         delegate->GetInfoBarSeparatorColor());
    188     int arrow_x;
    189     SkScalar arrow_fill_height =
    190         SkIntToScalar(std::max(arrow_height() - kSeparatorLineHeight, 0));
    191     SkScalar arrow_fill_half_width = SkIntToScalar(arrow_half_width());
    192     SkScalar separator_height = SkIntToScalar(kSeparatorLineHeight);
    193     if (delegate->DrawInfoBarArrows(&arrow_x) && arrow_fill_height) {
    194       // Skia pixel centers are at the half-values, so the arrow is horizontally
    195       // centered at |arrow_x| + 0.5.  Vertically, the stroke path is the center
    196       // of the separator, while the fill path is a closed path that extends up
    197       // through the entire height of the separator and down to the bottom of
    198       // the arrow where it joins the bar.
    199       stroke_path_.moveTo(
    200           SkIntToScalar(arrow_x) + SK_ScalarHalf - arrow_fill_half_width,
    201           SkIntToScalar(arrow_height()) - (separator_height * SK_ScalarHalf));
    202       stroke_path_.rLineTo(arrow_fill_half_width, -arrow_fill_height);
    203       stroke_path_.rLineTo(arrow_fill_half_width, arrow_fill_height);
    204 
    205       fill_path_ = stroke_path_;
    206       // Move the top of the fill path up to the top of the separator and then
    207       // extend it down all the way through.
    208       fill_path_.offset(0, -separator_height * SK_ScalarHalf);
    209       // This 0.01 hack prevents the fill from filling more pixels on the right
    210       // edge of the arrow than on the left.
    211       const SkScalar epsilon = 0.01f;
    212       fill_path_.rLineTo(-epsilon, 0);
    213       fill_path_.rLineTo(0, separator_height);
    214       fill_path_.rLineTo(epsilon - (arrow_fill_half_width * 2), 0);
    215       fill_path_.close();
    216     }
    217   }
    218   if (bar_height()) {
    219     fill_path_.addRect(0.0, SkIntToScalar(arrow_height()),
    220         SkIntToScalar(width()), SkIntToScalar(height() - kSeparatorLineHeight));
    221   }
    222 
    223   int start_x = kEdgeItemPadding;
    224   if (icon_ != NULL) {
    225     icon_->SetPosition(gfx::Point(start_x, OffsetY(icon_)));
    226     start_x = icon_->bounds().right() + kIconToLabelSpacing;
    227   }
    228 
    229   int content_minimum_width = ContentMinimumWidth();
    230   close_button_->SetPosition(gfx::Point(
    231       std::max(
    232           start_x + content_minimum_width +
    233               ((content_minimum_width > 0) ? kBeforeCloseButtonSpacing : 0),
    234           width() - kEdgeItemPadding - close_button_->width()),
    235       OffsetY(close_button_)));
    236 }
    237 
    238 void InfoBarView::ViewHierarchyChanged(
    239     const ViewHierarchyChangedDetails& details) {
    240   View::ViewHierarchyChanged(details);
    241 
    242   if (details.is_add && (details.child == this) && (close_button_ == NULL)) {
    243     gfx::Image image = delegate()->GetIcon();
    244     if (!image.IsEmpty()) {
    245       icon_ = new views::ImageView;
    246       icon_->SetImage(image.ToImageSkia());
    247       icon_->SizeToPreferredSize();
    248       AddChildView(icon_);
    249     }
    250 
    251     close_button_ = new views::ImageButton(this);
    252     ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
    253     close_button_->SetImage(views::CustomButton::STATE_NORMAL,
    254                             rb.GetImageNamed(IDR_CLOSE_1).ToImageSkia());
    255     close_button_->SetImage(views::CustomButton::STATE_HOVERED,
    256                             rb.GetImageNamed(IDR_CLOSE_1_H).ToImageSkia());
    257     close_button_->SetImage(views::CustomButton::STATE_PRESSED,
    258                             rb.GetImageNamed(IDR_CLOSE_1_P).ToImageSkia());
    259     close_button_->SizeToPreferredSize();
    260     close_button_->SetAccessibleName(
    261         l10n_util::GetStringUTF16(IDS_ACCNAME_CLOSE));
    262     close_button_->SetFocusable(true);
    263     AddChildView(close_button_);
    264   } else if ((close_button_ != NULL) && (details.parent == this) &&
    265       (details.child != close_button_) && (close_button_->parent() == this) &&
    266       (child_at(child_count() - 1) != close_button_)) {
    267     // For accessibility, ensure the close button is the last child view.
    268     RemoveChildView(close_button_);
    269     AddChildView(close_button_);
    270   }
    271 
    272   // Ensure the infobar is tall enough to display its contents.
    273   const int kMinimumVerticalPadding = 6;
    274   int height = kDefaultBarTargetHeight;
    275   for (int i = 0; i < child_count(); ++i) {
    276     const int child_height = child_at(i)->height();
    277     height = std::max(height, child_height + kMinimumVerticalPadding);
    278   }
    279   SetBarTargetHeight(height);
    280 }
    281 
    282 void InfoBarView::PaintChildren(gfx::Canvas* canvas,
    283                                 const views::CullSet& cull_set) {
    284   canvas->Save();
    285 
    286   // TODO(scr): This really should be the |fill_path_|, but the clipPath seems
    287   // broken on non-Windows platforms (crbug.com/75154). For now, just clip to
    288   // the bar bounds.
    289   //
    290   // canvas->sk_canvas()->clipPath(fill_path_);
    291   DCHECK_EQ(total_height(), height())
    292       << "Infobar piecewise heights do not match overall height";
    293   canvas->ClipRect(gfx::Rect(0, arrow_height(), width(), bar_height()));
    294   views::View::PaintChildren(canvas, cull_set);
    295   canvas->Restore();
    296 }
    297 
    298 void InfoBarView::ButtonPressed(views::Button* sender,
    299                                 const ui::Event& event) {
    300   if (!owner())
    301     return;  // We're closing; don't call anything, it might access the owner.
    302   if (sender == close_button_) {
    303     delegate()->InfoBarDismissed();
    304     RemoveSelf();
    305   }
    306 }
    307 
    308 int InfoBarView::ContentMinimumWidth() const {
    309   return 0;
    310 }
    311 
    312 int InfoBarView::StartX() const {
    313   // Ensure we don't return a value greater than EndX(), so children can safely
    314   // set something's width to "EndX() - StartX()" without risking that being
    315   // negative.
    316   return std::min(EndX(), (icon_ != NULL) ?
    317       (icon_->bounds().right() + kIconToLabelSpacing) : kEdgeItemPadding);
    318 }
    319 
    320 int InfoBarView::EndX() const {
    321   return close_button_->x() - kBeforeCloseButtonSpacing;
    322 }
    323 
    324 int InfoBarView::OffsetY(views::View* view) const {
    325   return arrow_height() +
    326       std::max((bar_target_height() - view->height()) / 2, 0) -
    327       (bar_target_height() - bar_height());
    328 }
    329 
    330 const infobars::InfoBarContainer::Delegate* InfoBarView::container_delegate()
    331     const {
    332   const infobars::InfoBarContainer* infobar_container = container();
    333   return infobar_container ? infobar_container->delegate() : NULL;
    334 }
    335 
    336 void InfoBarView::RunMenuAt(ui::MenuModel* menu_model,
    337                             views::MenuButton* button,
    338                             views::MenuAnchorPosition anchor) {
    339   DCHECK(owner());  // We'd better not open any menus while we're closing.
    340   gfx::Point screen_point;
    341   views::View::ConvertPointToScreen(button, &screen_point);
    342   menu_runner_.reset(new views::MenuRunner(menu_model));
    343   // Ignore the result since we don't need to handle a deleted menu specially.
    344   ignore_result(menu_runner_->RunMenuAt(
    345       GetWidget(), button, gfx::Rect(screen_point, button->size()), anchor,
    346       ui::MENU_SOURCE_NONE, views::MenuRunner::HAS_MNEMONICS));
    347 }
    348 
    349 // static
    350 void InfoBarView::AssignWidthsSorted(Labels* labels, int available_width) {
    351   if (labels->empty())
    352     return;
    353   gfx::Size back_label_size(labels->back()->GetPreferredSize());
    354   back_label_size.set_width(
    355       std::min(back_label_size.width(),
    356                available_width / static_cast<int>(labels->size())));
    357   labels->back()->SetSize(back_label_size);
    358   labels->pop_back();
    359   AssignWidthsSorted(labels, available_width - back_label_size.width());
    360 }
    361 
    362 void InfoBarView::PlatformSpecificShow(bool animate) {
    363   // If we gain focus, we want to restore it to the previously-focused element
    364   // when we're hidden. So when we're in a Widget, create a focus tracker so
    365   // that if we gain focus we'll know what the previously-focused element was.
    366   SetFocusManager(GetFocusManager());
    367 
    368   NotifyAccessibilityEvent(ui::AX_EVENT_ALERT, true);
    369 }
    370 
    371 void InfoBarView::PlatformSpecificHide(bool animate) {
    372   // Cancel any menus we may have open.  It doesn't make sense to leave them
    373   // open while we're hidden, and if we're going to become unowned, we can't
    374   // allow the user to choose any options and potentially call functions that
    375   // try to access the owner.
    376   menu_runner_.reset();
    377 
    378   // It's possible to be called twice (once with |animate| true and once with it
    379   // false); in this case the second SetFocusManager() call will silently no-op.
    380   SetFocusManager(NULL);
    381 
    382   if (!animate)
    383     return;
    384 
    385   // Do not restore focus (and active state with it) if some other top-level
    386   // window became active.
    387   views::Widget* widget = GetWidget();
    388   if (!widget || widget->IsActive())
    389     FocusLastFocusedExternalView();
    390 }
    391 
    392 void InfoBarView::PlatformSpecificOnHeightsRecalculated() {
    393   // Ensure that notifying our container of our size change will result in a
    394   // re-layout.
    395   InvalidateLayout();
    396 }
    397 
    398 void InfoBarView::GetAccessibleState(ui::AXViewState* state) {
    399   state->name = l10n_util::GetStringUTF16(
    400       (delegate()->GetInfoBarType() ==
    401        infobars::InfoBarDelegate::WARNING_TYPE) ?
    402           IDS_ACCNAME_INFOBAR_WARNING : IDS_ACCNAME_INFOBAR_PAGE_ACTION);
    403   state->role = ui::AX_ROLE_ALERT;
    404   state->keyboard_shortcut = base::ASCIIToUTF16("Alt+Shift+A");
    405 }
    406 
    407 gfx::Size InfoBarView::GetPreferredSize() const {
    408   return gfx::Size(
    409       kEdgeItemPadding + (icon_ ? (icon_->width() + kIconToLabelSpacing) : 0) +
    410           ContentMinimumWidth() + kBeforeCloseButtonSpacing +
    411           close_button_->width() + kEdgeItemPadding,
    412       total_height());
    413 }
    414 
    415 void InfoBarView::OnWillChangeFocus(View* focused_before, View* focused_now) {
    416   views::ExternalFocusTracker::OnWillChangeFocus(focused_before, focused_now);
    417 
    418   // This will trigger some screen readers to read the entire contents of this
    419   // infobar.
    420   if (focused_before && focused_now && !Contains(focused_before) &&
    421       Contains(focused_now)) {
    422     NotifyAccessibilityEvent(ui::AX_EVENT_ALERT, true);
    423   }
    424 }
    425