1 // Copyright 2013 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.
5 #include "chrome/browser/ui/views/autofill/info_bubble.h"
7 #include "base/i18n/rtl.h"
8 #include "ui/gfx/geometry/point.h"
9 #include "ui/gfx/geometry/rect.h"
10 #include "ui/gfx/geometry/size.h"
11 #include "ui/gfx/text_constants.h"
12 #include "ui/views/bubble/bubble_border.h"
13 #include "ui/views/bubble/bubble_frame_view.h"
14 #include "ui/views/controls/combobox/combobox.h"
15 #include "ui/views/controls/label.h"
16 #include "ui/views/layout/fill_layout.h"
17 #include "ui/views/layout/layout_constants.h"
18 #include "ui/views/widget/widget.h"
24 // The visible width of bubble borders (differs from the actual width) in px.
25 const int kBubbleBorderVisibleWidth
= 1;
27 // The margin between the content of the error bubble and its border.
28 const int kInfoBubbleHorizontalMargin
= 14;
29 const int kInfoBubbleVerticalMargin
= 12;
33 class InfoBubbleFrame
: public views::BubbleFrameView
{
35 explicit InfoBubbleFrame(const gfx::Insets
& content_margins
)
36 : views::BubbleFrameView(content_margins
) {}
37 ~InfoBubbleFrame() override
{}
39 gfx::Rect
GetAvailableScreenBounds(const gfx::Rect
& rect
) const override
{
40 return available_bounds_
;
43 void set_available_bounds(const gfx::Rect
& available_bounds
) {
44 available_bounds_
= available_bounds
;
48 // Bounds that this frame should try to keep bubbles within (screen coords).
49 gfx::Rect available_bounds_
;
51 DISALLOW_COPY_AND_ASSIGN(InfoBubbleFrame
);
54 InfoBubble::InfoBubble(views::View
* anchor
,
55 const base::string16
& message
)
58 align_to_anchor_edge_(false),
59 preferred_width_(233),
60 show_above_anchor_(false) {
62 SetAnchorView(anchor_
);
64 set_margins(gfx::Insets(kInfoBubbleVerticalMargin
,
65 kInfoBubbleHorizontalMargin
,
66 kInfoBubbleVerticalMargin
,
67 kInfoBubbleHorizontalMargin
));
68 set_can_activate(false);
70 SetLayoutManager(new views::FillLayout
);
71 views::Label
* label
= new views::Label(message
);
72 label
->SetHorizontalAlignment(gfx::ALIGN_LEFT
);
73 label
->SetMultiLine(true);
77 InfoBubble::~InfoBubble() {}
79 void InfoBubble::Show() {
80 // TODO(dbeam): currently we assume that combobox menus always show downward
81 // (which isn't true). If the invalid combobox is low enough on the screen,
82 // its menu will actually show upward and obscure the bubble. Figure out when
83 // this might happen and adjust |show_above_anchor_| accordingly. This is not
84 // that big of deal because it rarely happens in practice.
85 if (show_above_anchor_
)
86 set_arrow(views::BubbleBorder::vertical_mirror(arrow()));
88 widget_
= views::BubbleDelegateView::CreateBubble(this);
90 if (align_to_anchor_edge_
) {
91 // The frame adjusts its arrow before the bubble's alignment can be changed.
92 // Set the created bubble border back to the original arrow and re-adjust.
93 frame_
->bubble_border()->set_arrow(arrow());
94 SetAlignment(views::BubbleBorder::ALIGN_EDGE_TO_ANCHOR_EDGE
);
100 void InfoBubble::Hide() {
101 views::Widget
* widget
= GetWidget();
102 if (widget
&& !widget
->IsClosed())
106 void InfoBubble::UpdatePosition() {
110 if (!anchor_
->GetVisibleBounds().IsEmpty()) {
112 widget_
->SetVisibilityChangedAnimationsEnabled(true);
113 widget_
->ShowInactive();
115 widget_
->SetVisibilityChangedAnimationsEnabled(false);
120 views::NonClientFrameView
* InfoBubble::CreateNonClientFrameView(
121 views::Widget
* widget
) {
123 frame_
= new InfoBubbleFrame(margins());
124 frame_
->set_available_bounds(anchor_widget()->GetWindowBoundsInScreen());
125 frame_
->SetBubbleBorder(scoped_ptr
<views::BubbleBorder
>(
126 new views::BubbleBorder(arrow(), shadow(), color())));
130 gfx::Size
InfoBubble::GetPreferredSize() const {
131 int pref_width
= preferred_width_
;
132 pref_width
-= frame_
->GetInsets().width();
133 pref_width
-= 2 * kBubbleBorderVisibleWidth
;
134 return gfx::Size(pref_width
, GetHeightForWidth(pref_width
));
137 void InfoBubble::OnWidgetDestroyed(views::Widget
* widget
) {
138 if (widget
== widget_
)
142 void InfoBubble::OnWidgetBoundsChanged(views::Widget
* widget
,
143 const gfx::Rect
& new_bounds
) {
144 views::BubbleDelegateView::OnWidgetBoundsChanged(widget
, new_bounds
);
145 if (anchor_widget() == widget
)
146 frame_
->set_available_bounds(widget
->GetWindowBoundsInScreen());
149 } // namespace autofill