• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 "ui/views/bubble/bubble_delegate.h"
6 
7 #include "ui/accessibility/ax_view_state.h"
8 #include "ui/base/resource/resource_bundle.h"
9 #include "ui/gfx/color_utils.h"
10 #include "ui/gfx/rect.h"
11 #include "ui/native_theme/native_theme.h"
12 #include "ui/views/bubble/bubble_frame_view.h"
13 #include "ui/views/focus/view_storage.h"
14 #include "ui/views/widget/widget.h"
15 #include "ui/views/widget/widget_observer.h"
16 
17 #if defined(OS_WIN)
18 #include "ui/base/win/shell.h"
19 #endif
20 
21 // The defaut margin between the content and the inside border, in pixels.
22 static const int kDefaultMargin = 6;
23 
24 namespace views {
25 
26 namespace {
27 
28 // Create a widget to host the bubble.
CreateBubbleWidget(BubbleDelegateView * bubble)29 Widget* CreateBubbleWidget(BubbleDelegateView* bubble) {
30   Widget* bubble_widget = new Widget();
31   Widget::InitParams bubble_params(Widget::InitParams::TYPE_BUBBLE);
32   bubble_params.delegate = bubble;
33   bubble_params.opacity = Widget::InitParams::TRANSLUCENT_WINDOW;
34   bubble_params.accept_events = bubble->accept_events();
35   if (bubble->parent_window())
36     bubble_params.parent = bubble->parent_window();
37   else if (bubble->anchor_widget())
38     bubble_params.parent = bubble->anchor_widget()->GetNativeView();
39   bubble_params.activatable = bubble->CanActivate() ?
40       Widget::InitParams::ACTIVATABLE_YES : Widget::InitParams::ACTIVATABLE_NO;
41   bubble->OnBeforeBubbleWidgetInit(&bubble_params, bubble_widget);
42   bubble_widget->Init(bubble_params);
43   return bubble_widget;
44 }
45 
46 }  // namespace
47 
BubbleDelegateView()48 BubbleDelegateView::BubbleDelegateView()
49     : close_on_esc_(true),
50       close_on_deactivate_(true),
51       anchor_view_storage_id_(ViewStorage::GetInstance()->CreateStorageID()),
52       anchor_widget_(NULL),
53       arrow_(BubbleBorder::TOP_LEFT),
54       shadow_(BubbleBorder::SMALL_SHADOW),
55       color_explicitly_set_(false),
56       margins_(kDefaultMargin, kDefaultMargin, kDefaultMargin, kDefaultMargin),
57       use_focusless_(false),
58       accept_events_(true),
59       border_accepts_events_(true),
60       adjust_if_offscreen_(true),
61       parent_window_(NULL) {
62   AddAccelerator(ui::Accelerator(ui::VKEY_ESCAPE, ui::EF_NONE));
63   UpdateColorsFromTheme(GetNativeTheme());
64 }
65 
BubbleDelegateView(View * anchor_view,BubbleBorder::Arrow arrow)66 BubbleDelegateView::BubbleDelegateView(
67     View* anchor_view,
68     BubbleBorder::Arrow arrow)
69     : close_on_esc_(true),
70       close_on_deactivate_(true),
71       anchor_view_storage_id_(ViewStorage::GetInstance()->CreateStorageID()),
72       anchor_widget_(NULL),
73       arrow_(arrow),
74       shadow_(BubbleBorder::SMALL_SHADOW),
75       color_explicitly_set_(false),
76       margins_(kDefaultMargin, kDefaultMargin, kDefaultMargin, kDefaultMargin),
77       use_focusless_(false),
78       accept_events_(true),
79       border_accepts_events_(true),
80       adjust_if_offscreen_(true),
81       parent_window_(NULL) {
82   SetAnchorView(anchor_view);
83   AddAccelerator(ui::Accelerator(ui::VKEY_ESCAPE, ui::EF_NONE));
84   UpdateColorsFromTheme(GetNativeTheme());
85 }
86 
~BubbleDelegateView()87 BubbleDelegateView::~BubbleDelegateView() {
88   if (GetWidget())
89     GetWidget()->RemoveObserver(this);
90   SetLayoutManager(NULL);
91   SetAnchorView(NULL);
92 }
93 
94 // static
CreateBubble(BubbleDelegateView * bubble_delegate)95 Widget* BubbleDelegateView::CreateBubble(BubbleDelegateView* bubble_delegate) {
96   bubble_delegate->Init();
97   // Get the latest anchor widget from the anchor view at bubble creation time.
98   bubble_delegate->SetAnchorView(bubble_delegate->GetAnchorView());
99   Widget* bubble_widget = CreateBubbleWidget(bubble_delegate);
100 
101 #if defined(OS_WIN)
102   // If glass is enabled, the bubble is allowed to extend outside the bounds of
103   // the parent frame and let DWM handle compositing.  If not, then we don't
104   // want to allow the bubble to extend the frame because it will be clipped.
105   bubble_delegate->set_adjust_if_offscreen(ui::win::IsAeroGlassEnabled());
106 #elif defined(OS_LINUX) && !defined(OS_CHROMEOS)
107   // Linux clips bubble windows that extend outside their parent window bounds.
108   bubble_delegate->set_adjust_if_offscreen(false);
109 #endif
110 
111   bubble_delegate->SizeToContents();
112   bubble_widget->AddObserver(bubble_delegate);
113   return bubble_widget;
114 }
115 
AsBubbleDelegate()116 BubbleDelegateView* BubbleDelegateView::AsBubbleDelegate() {
117   return this;
118 }
119 
CanActivate() const120 bool BubbleDelegateView::CanActivate() const {
121   return !use_focusless();
122 }
123 
ShouldShowCloseButton() const124 bool BubbleDelegateView::ShouldShowCloseButton() const {
125   return false;
126 }
127 
GetContentsView()128 View* BubbleDelegateView::GetContentsView() {
129   return this;
130 }
131 
CreateNonClientFrameView(Widget * widget)132 NonClientFrameView* BubbleDelegateView::CreateNonClientFrameView(
133     Widget* widget) {
134   BubbleFrameView* frame = new BubbleFrameView(margins());
135   // Note: In CreateBubble, the call to SizeToContents() will cause
136   // the relayout that this call requires.
137   frame->SetTitleFontList(GetTitleFontList());
138   BubbleBorder::Arrow adjusted_arrow = arrow();
139   if (base::i18n::IsRTL())
140     adjusted_arrow = BubbleBorder::horizontal_mirror(adjusted_arrow);
141   frame->SetBubbleBorder(scoped_ptr<BubbleBorder>(
142       new BubbleBorder(adjusted_arrow, shadow(), color())));
143   return frame;
144 }
145 
GetAccessibleState(ui::AXViewState * state)146 void BubbleDelegateView::GetAccessibleState(ui::AXViewState* state) {
147   state->role = ui::AX_ROLE_DIALOG;
148 }
149 
OnWidgetDestroying(Widget * widget)150 void BubbleDelegateView::OnWidgetDestroying(Widget* widget) {
151   if (anchor_widget() == widget)
152     SetAnchorView(NULL);
153 }
154 
OnWidgetVisibilityChanging(Widget * widget,bool visible)155 void BubbleDelegateView::OnWidgetVisibilityChanging(Widget* widget,
156                                                     bool visible) {
157 #if defined(OS_WIN)
158   // On Windows we need to handle this before the bubble is visible or hidden.
159   // Please see the comment on the OnWidgetVisibilityChanging function. On
160   // other platforms it is fine to handle it after the bubble is shown/hidden.
161   HandleVisibilityChanged(widget, visible);
162 #endif
163 }
164 
OnWidgetVisibilityChanged(Widget * widget,bool visible)165 void BubbleDelegateView::OnWidgetVisibilityChanged(Widget* widget,
166                                                    bool visible) {
167 #if !defined(OS_WIN)
168   HandleVisibilityChanged(widget, visible);
169 #endif
170 }
171 
OnWidgetActivationChanged(Widget * widget,bool active)172 void BubbleDelegateView::OnWidgetActivationChanged(Widget* widget,
173                                                    bool active) {
174   if (close_on_deactivate() && widget == GetWidget() && !active)
175     GetWidget()->Close();
176 }
177 
OnWidgetBoundsChanged(Widget * widget,const gfx::Rect & new_bounds)178 void BubbleDelegateView::OnWidgetBoundsChanged(Widget* widget,
179                                                const gfx::Rect& new_bounds) {
180   if (anchor_widget() == widget)
181     SizeToContents();
182 }
183 
GetAnchorView() const184 View* BubbleDelegateView::GetAnchorView() const {
185   return ViewStorage::GetInstance()->RetrieveView(anchor_view_storage_id_);
186 }
187 
GetAnchorRect() const188 gfx::Rect BubbleDelegateView::GetAnchorRect() const {
189   if (!GetAnchorView())
190     return anchor_rect_;
191 
192   anchor_rect_ = GetAnchorView()->GetBoundsInScreen();
193   anchor_rect_.Inset(anchor_view_insets_);
194   return anchor_rect_;
195 }
196 
OnBeforeBubbleWidgetInit(Widget::InitParams * params,Widget * widget) const197 void BubbleDelegateView::OnBeforeBubbleWidgetInit(Widget::InitParams* params,
198                                                   Widget* widget) const {
199 }
200 
SetAlignment(BubbleBorder::BubbleAlignment alignment)201 void BubbleDelegateView::SetAlignment(BubbleBorder::BubbleAlignment alignment) {
202   GetBubbleFrameView()->bubble_border()->set_alignment(alignment);
203   SizeToContents();
204 }
205 
SetArrowPaintType(BubbleBorder::ArrowPaintType paint_type)206 void BubbleDelegateView::SetArrowPaintType(
207     BubbleBorder::ArrowPaintType paint_type) {
208   GetBubbleFrameView()->bubble_border()->set_paint_arrow(paint_type);
209   SizeToContents();
210 }
211 
OnAnchorBoundsChanged()212 void BubbleDelegateView::OnAnchorBoundsChanged() {
213   SizeToContents();
214 }
215 
AcceleratorPressed(const ui::Accelerator & accelerator)216 bool BubbleDelegateView::AcceleratorPressed(
217     const ui::Accelerator& accelerator) {
218   if (!close_on_esc() || accelerator.key_code() != ui::VKEY_ESCAPE)
219     return false;
220   GetWidget()->Close();
221   return true;
222 }
223 
OnNativeThemeChanged(const ui::NativeTheme * theme)224 void BubbleDelegateView::OnNativeThemeChanged(const ui::NativeTheme* theme) {
225   UpdateColorsFromTheme(theme);
226 }
227 
Init()228 void BubbleDelegateView::Init() {}
229 
SetAnchorView(View * anchor_view)230 void BubbleDelegateView::SetAnchorView(View* anchor_view) {
231   // When the anchor view gets set the associated anchor widget might
232   // change as well.
233   if (!anchor_view || anchor_widget() != anchor_view->GetWidget()) {
234     if (anchor_widget()) {
235       anchor_widget_->RemoveObserver(this);
236       anchor_widget_ = NULL;
237     }
238     if (anchor_view) {
239       anchor_widget_ = anchor_view->GetWidget();
240       if (anchor_widget_)
241         anchor_widget_->AddObserver(this);
242     }
243   }
244 
245   // Remove the old storage item and set the new (if there is one).
246   ViewStorage* view_storage = ViewStorage::GetInstance();
247   if (view_storage->RetrieveView(anchor_view_storage_id_))
248     view_storage->RemoveView(anchor_view_storage_id_);
249   if (anchor_view)
250     view_storage->StoreView(anchor_view_storage_id_, anchor_view);
251 
252   // Do not update anchoring for NULL views; this could indicate that our
253   // NativeWindow is being destroyed, so it would be dangerous for us to update
254   // our anchor bounds at that point. (It's safe to skip this, since if we were
255   // to update the bounds when |anchor_view| is NULL, the bubble won't move.)
256   if (anchor_view && GetWidget())
257     OnAnchorBoundsChanged();
258 }
259 
SetAnchorRect(const gfx::Rect & rect)260 void BubbleDelegateView::SetAnchorRect(const gfx::Rect& rect) {
261   anchor_rect_ = rect;
262   if (GetWidget())
263     OnAnchorBoundsChanged();
264 }
265 
SizeToContents()266 void BubbleDelegateView::SizeToContents() {
267   GetWidget()->SetBounds(GetBubbleBounds());
268 }
269 
GetBubbleFrameView() const270 BubbleFrameView* BubbleDelegateView::GetBubbleFrameView() const {
271   const NonClientView* view =
272       GetWidget() ? GetWidget()->non_client_view() : NULL;
273   return view ? static_cast<BubbleFrameView*>(view->frame_view()) : NULL;
274 }
275 
GetBubbleBounds()276 gfx::Rect BubbleDelegateView::GetBubbleBounds() {
277   // The argument rect has its origin at the bubble's arrow anchor point;
278   // its size is the preferred size of the bubble's client view (this view).
279   bool anchor_minimized = anchor_widget() && anchor_widget()->IsMinimized();
280   return GetBubbleFrameView()->GetUpdatedWindowBounds(GetAnchorRect(),
281       GetPreferredSize(), adjust_if_offscreen_ && !anchor_minimized);
282 }
283 
GetTitleFontList() const284 const gfx::FontList& BubbleDelegateView::GetTitleFontList() const {
285   ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
286   return rb.GetFontList(ui::ResourceBundle::MediumFont);
287 }
288 
289 
UpdateColorsFromTheme(const ui::NativeTheme * theme)290 void BubbleDelegateView::UpdateColorsFromTheme(const ui::NativeTheme* theme) {
291   if (!color_explicitly_set_)
292     color_ = theme->GetSystemColor(ui::NativeTheme::kColorId_DialogBackground);
293   set_background(Background::CreateSolidBackground(color()));
294   BubbleFrameView* frame_view = GetBubbleFrameView();
295   if (frame_view)
296     frame_view->bubble_border()->set_background_color(color());
297 }
298 
HandleVisibilityChanged(Widget * widget,bool visible)299 void BubbleDelegateView::HandleVisibilityChanged(Widget* widget, bool visible) {
300   if (widget == GetWidget() && anchor_widget() &&
301       anchor_widget()->GetTopLevelWidget()) {
302     if (visible)
303       anchor_widget()->GetTopLevelWidget()->DisableInactiveRendering();
304     else
305       anchor_widget()->GetTopLevelWidget()->EnableInactiveRendering();
306   }
307 }
308 
309 }  // namespace views
310