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.
5 #include "ui/views/widget/native_widget_aura.h"
8 #include "base/strings/string_util.h"
9 #include "third_party/skia/include/core/SkRegion.h"
10 #include "ui/aura/client/aura_constants.h"
11 #include "ui/aura/client/cursor_client.h"
12 #include "ui/aura/client/focus_client.h"
13 #include "ui/aura/client/screen_position_client.h"
14 #include "ui/aura/client/window_tree_client.h"
15 #include "ui/aura/env.h"
16 #include "ui/aura/window.h"
17 #include "ui/aura/window_event_dispatcher.h"
18 #include "ui/aura/window_observer.h"
19 #include "ui/base/dragdrop/os_exchange_data.h"
20 #include "ui/base/ui_base_switches_util.h"
21 #include "ui/base/ui_base_types.h"
22 #include "ui/compositor/layer.h"
23 #include "ui/events/event.h"
24 #include "ui/gfx/canvas.h"
25 #include "ui/gfx/font_list.h"
26 #include "ui/gfx/screen.h"
27 #include "ui/native_theme/native_theme_aura.h"
28 #include "ui/views/drag_utils.h"
29 #include "ui/views/ime/input_method_bridge.h"
30 #include "ui/views/ime/null_input_method.h"
31 #include "ui/views/views_delegate.h"
32 #include "ui/views/widget/drop_helper.h"
33 #include "ui/views/widget/native_widget_delegate.h"
34 #include "ui/views/widget/root_view.h"
35 #include "ui/views/widget/tooltip_manager_aura.h"
36 #include "ui/views/widget/widget_aura_utils.h"
37 #include "ui/views/widget/widget_delegate.h"
38 #include "ui/views/widget/window_reorderer.h"
39 #include "ui/wm/core/shadow_types.h"
40 #include "ui/wm/core/window_util.h"
41 #include "ui/wm/public/activation_client.h"
42 #include "ui/wm/public/drag_drop_client.h"
43 #include "ui/wm/public/window_move_client.h"
44 #include "ui/wm/public/window_types.h"
47 #include "base/win/scoped_gdi_object.h"
48 #include "base/win/win_util.h"
49 #include "ui/base/l10n/l10n_util_win.h"
50 #include "ui/views/widget/desktop_aura/desktop_window_tree_host_win.h"
53 #if defined(USE_X11) && !defined(OS_CHROMEOS)
54 #include "ui/views/widget/desktop_aura/desktop_window_tree_host_x11.h"
57 #if !defined(OS_CHROMEOS)
58 #include "ui/views/widget/desktop_aura/desktop_native_widget_aura.h"
59 #include "ui/views/widget/desktop_aura/desktop_window_tree_host.h"
66 void SetRestoreBounds(aura::Window
* window
, const gfx::Rect
& bounds
) {
67 window
->SetProperty(aura::client::kRestoreBoundsKey
, new gfx::Rect(bounds
));
72 ////////////////////////////////////////////////////////////////////////////////
73 // NativeWidgetAura, public:
75 NativeWidgetAura::NativeWidgetAura(internal::NativeWidgetDelegate
* delegate
)
76 : delegate_(delegate
),
77 window_(new aura::Window(this)),
78 ownership_(Widget::InitParams::NATIVE_WIDGET_OWNS_WIDGET
),
80 cursor_(gfx::kNullCursor
),
81 saved_window_state_(ui::SHOW_STATE_DEFAULT
),
82 close_widget_factory_(this) {
83 aura::client::SetFocusChangeObserver(window_
, this);
84 aura::client::SetActivationChangeObserver(window_
, this);
88 void NativeWidgetAura::RegisterNativeWidgetForWindow(
89 internal::NativeWidgetPrivate
* native_widget
,
90 aura::Window
* window
) {
91 window
->set_user_data(native_widget
);
94 ////////////////////////////////////////////////////////////////////////////////
95 // NativeWidgetAura, internal::NativeWidgetPrivate implementation:
97 void NativeWidgetAura::InitNativeWidget(const Widget::InitParams
& params
) {
98 // Aura needs to know which desktop (Ash or regular) will manage this widget.
99 // See Widget::InitParams::context for details.
100 DCHECK(params
.parent
|| params
.context
);
102 ownership_
= params
.ownership
;
104 RegisterNativeWidgetForWindow(this, window_
);
105 window_
->SetType(GetAuraWindowTypeForWidgetType(params
.type
));
106 window_
->SetProperty(aura::client::kShowStateKey
, params
.show_state
);
107 if (params
.type
== Widget::InitParams::TYPE_BUBBLE
)
108 aura::client::SetHideOnDeactivate(window_
, true);
109 window_
->SetTransparent(
110 params
.opacity
== Widget::InitParams::TRANSLUCENT_WINDOW
);
111 window_
->Init(params
.layer_type
);
112 if (params
.shadow_type
== Widget::InitParams::SHADOW_TYPE_NONE
)
113 SetShadowType(window_
, wm::SHADOW_TYPE_NONE
);
114 else if (params
.shadow_type
== Widget::InitParams::SHADOW_TYPE_DROP
)
115 SetShadowType(window_
, wm::SHADOW_TYPE_RECTANGULAR
);
116 if (params
.type
== Widget::InitParams::TYPE_CONTROL
)
119 delegate_
->OnNativeWidgetCreated(false);
121 gfx::Rect window_bounds
= params
.bounds
;
122 gfx::NativeView parent
= params
.parent
;
123 gfx::NativeView context
= params
.context
;
125 // Set up the transient child before the window is added. This way the
126 // LayoutManager knows the window has a transient parent.
127 if (parent
&& parent
->type() != ui::wm::WINDOW_TYPE_UNKNOWN
) {
128 wm::AddTransientChild(parent
, window_
);
133 // SetAlwaysOnTop before SetParent so that always-on-top container is used.
134 SetAlwaysOnTop(params
.keep_on_top
);
135 // Make sure we have a real |window_bounds|.
136 if (parent
&& window_bounds
== gfx::Rect()) {
137 // If a parent is specified but no bounds are given,
138 // use the origin of the parent's display so that the widget
139 // will be added to the same display as the parent.
140 gfx::Rect bounds
= gfx::Screen::GetScreenFor(parent
)->
141 GetDisplayNearestWindow(parent
).bounds();
142 window_bounds
.set_origin(bounds
.origin());
146 // Set properties before adding to the parent so that its layout manager sees
147 // the correct values.
148 OnSizeConstraintsChanged();
151 parent
->AddChild(window_
);
153 aura::client::ParentWindowWithContext(
154 window_
, context
->GetRootWindow(), window_bounds
);
157 // Start observing property changes.
158 window_
->AddObserver(this);
160 // Wait to set the bounds until we have a parent. That way we can know our
161 // true state/bounds (the LayoutManager may enforce a particular
164 SetRestoreBounds(window_
, window_bounds
);
166 SetBounds(window_bounds
);
167 window_
->set_ignore_events(!params
.accept_events
);
168 DCHECK(GetWidget()->GetRootView());
169 if (params
.type
!= Widget::InitParams::TYPE_TOOLTIP
)
170 tooltip_manager_
.reset(new views::TooltipManagerAura(GetWidget()));
172 drop_helper_
.reset(new DropHelper(GetWidget()->GetRootView()));
173 if (params
.type
!= Widget::InitParams::TYPE_TOOLTIP
&&
174 params
.type
!= Widget::InitParams::TYPE_POPUP
) {
175 aura::client::SetDragDropDelegate(window_
, this);
178 aura::client::SetActivationDelegate(window_
, this);
180 window_reorderer_
.reset(new WindowReorderer(window_
,
181 GetWidget()->GetRootView()));
184 NonClientFrameView
* NativeWidgetAura::CreateNonClientFrameView() {
188 bool NativeWidgetAura::ShouldUseNativeFrame() const {
189 // There is only one frame type for aura.
193 bool NativeWidgetAura::ShouldWindowContentsBeTransparent() const {
197 void NativeWidgetAura::FrameTypeChanged() {
198 // This is called when the Theme has changed; forward the event to the root
200 GetWidget()->ThemeChanged();
201 GetWidget()->GetRootView()->SchedulePaint();
204 Widget
* NativeWidgetAura::GetWidget() {
205 return delegate_
->AsWidget();
208 const Widget
* NativeWidgetAura::GetWidget() const {
209 return delegate_
->AsWidget();
212 gfx::NativeView
NativeWidgetAura::GetNativeView() const {
216 gfx::NativeWindow
NativeWidgetAura::GetNativeWindow() const {
220 Widget
* NativeWidgetAura::GetTopLevelWidget() {
221 NativeWidgetPrivate
* native_widget
= GetTopLevelNativeWidget(GetNativeView());
222 return native_widget
? native_widget
->GetWidget() : NULL
;
225 const ui::Compositor
* NativeWidgetAura::GetCompositor() const {
226 return window_
? window_
->layer()->GetCompositor() : NULL
;
229 ui::Compositor
* NativeWidgetAura::GetCompositor() {
230 return window_
? window_
->layer()->GetCompositor() : NULL
;
233 ui::Layer
* NativeWidgetAura::GetLayer() {
234 return window_
? window_
->layer() : NULL
;
237 void NativeWidgetAura::ReorderNativeViews() {
238 window_reorderer_
->ReorderChildWindows();
241 void NativeWidgetAura::ViewRemoved(View
* view
) {
242 DCHECK(drop_helper_
.get() != NULL
);
243 drop_helper_
->ResetTargetViewIfEquals(view
);
246 void NativeWidgetAura::SetNativeWindowProperty(const char* name
, void* value
) {
248 window_
->SetNativeWindowProperty(name
, value
);
251 void* NativeWidgetAura::GetNativeWindowProperty(const char* name
) const {
252 return window_
? window_
->GetNativeWindowProperty(name
) : NULL
;
255 TooltipManager
* NativeWidgetAura::GetTooltipManager() const {
256 return tooltip_manager_
.get();
259 void NativeWidgetAura::SetCapture() {
261 window_
->SetCapture();
264 void NativeWidgetAura::ReleaseCapture() {
266 window_
->ReleaseCapture();
269 bool NativeWidgetAura::HasCapture() const {
270 return window_
&& window_
->HasCapture();
273 InputMethod
* NativeWidgetAura::CreateInputMethod() {
277 if (switches::IsTextInputFocusManagerEnabled())
278 return new NullInputMethod();
280 aura::Window
* root_window
= window_
->GetRootWindow();
281 ui::InputMethod
* host
=
282 root_window
->GetProperty(aura::client::kRootWindowInputMethodKey
);
283 return new InputMethodBridge(this, host
, true);
286 internal::InputMethodDelegate
* NativeWidgetAura::GetInputMethodDelegate() {
290 ui::InputMethod
* NativeWidgetAura::GetHostInputMethod() {
291 aura::Window
* root_window
= window_
->GetRootWindow();
292 return root_window
->GetProperty(aura::client::kRootWindowInputMethodKey
);
295 void NativeWidgetAura::CenterWindow(const gfx::Size
& size
) {
299 gfx::Rect
parent_bounds(window_
->parent()->GetBoundsInRootWindow());
300 // When centering window, we take the intersection of the host and
301 // the parent. We assume the root window represents the visible
302 // rect of a single screen.
303 gfx::Rect work_area
= gfx::Screen::GetScreenFor(window_
)->
304 GetDisplayNearestWindow(window_
).work_area();
306 aura::client::ScreenPositionClient
* screen_position_client
=
307 aura::client::GetScreenPositionClient(window_
->GetRootWindow());
308 if (screen_position_client
) {
309 gfx::Point origin
= work_area
.origin();
310 screen_position_client
->ConvertPointFromScreen(window_
->GetRootWindow(),
312 work_area
.set_origin(origin
);
315 parent_bounds
.Intersect(work_area
);
317 // If |window_|'s transient parent's bounds are big enough to fit it, then we
318 // center it with respect to the transient parent.
319 if (wm::GetTransientParent(window_
)) {
320 gfx::Rect transient_parent_rect
=
321 wm::GetTransientParent(window_
)->GetBoundsInRootWindow();
322 transient_parent_rect
.Intersect(work_area
);
323 if (transient_parent_rect
.height() >= size
.height() &&
324 transient_parent_rect
.width() >= size
.width())
325 parent_bounds
= transient_parent_rect
;
328 gfx::Rect
window_bounds(
329 parent_bounds
.x() + (parent_bounds
.width() - size
.width()) / 2,
330 parent_bounds
.y() + (parent_bounds
.height() - size
.height()) / 2,
333 // Don't size the window bigger than the parent, otherwise the user may not be
334 // able to close or move it.
335 window_bounds
.AdjustToFit(parent_bounds
);
337 // Convert the bounds back relative to the parent.
338 gfx::Point origin
= window_bounds
.origin();
339 aura::Window::ConvertPointToTarget(window_
->GetRootWindow(),
340 window_
->parent(), &origin
);
341 window_bounds
.set_origin(origin
);
342 window_
->SetBounds(window_bounds
);
345 void NativeWidgetAura::GetWindowPlacement(
347 ui::WindowShowState
* show_state
) const {
348 // The interface specifies returning restored bounds, not current bounds.
349 *bounds
= GetRestoredBounds();
350 *show_state
= window_
? window_
->GetProperty(aura::client::kShowStateKey
) :
351 ui::SHOW_STATE_DEFAULT
;
354 bool NativeWidgetAura::SetWindowTitle(const base::string16
& title
) {
357 if (window_
->title() == title
)
359 window_
->SetTitle(title
);
363 void NativeWidgetAura::SetWindowIcons(const gfx::ImageSkia
& window_icon
,
364 const gfx::ImageSkia
& app_icon
) {
365 // Aura doesn't have window icons.
368 void NativeWidgetAura::InitModalType(ui::ModalType modal_type
) {
369 if (modal_type
!= ui::MODAL_TYPE_NONE
)
370 window_
->SetProperty(aura::client::kModalKey
, modal_type
);
373 gfx::Rect
NativeWidgetAura::GetWindowBoundsInScreen() const {
374 return window_
? window_
->GetBoundsInScreen() : gfx::Rect();
377 gfx::Rect
NativeWidgetAura::GetClientAreaBoundsInScreen() const {
378 // View-to-screen coordinate system transformations depend on this returning
379 // the full window bounds, for example View::ConvertPointToScreen().
380 return window_
? window_
->GetBoundsInScreen() : gfx::Rect();
383 gfx::Rect
NativeWidgetAura::GetRestoredBounds() const {
387 // Restored bounds should only be relevant if the window is minimized or
388 // maximized. However, in some places the code expects GetRestoredBounds()
389 // to return the current window bounds if the window is not in either state.
390 if (IsMinimized() || IsMaximized() || IsFullscreen()) {
391 // Restore bounds are in screen coordinates, no need to convert.
392 gfx::Rect
* restore_bounds
=
393 window_
->GetProperty(aura::client::kRestoreBoundsKey
);
395 return *restore_bounds
;
397 return window_
->GetBoundsInScreen();
400 void NativeWidgetAura::SetBounds(const gfx::Rect
& bounds
) {
404 aura::Window
* root
= window_
->GetRootWindow();
406 aura::client::ScreenPositionClient
* screen_position_client
=
407 aura::client::GetScreenPositionClient(root
);
408 if (screen_position_client
) {
409 gfx::Display dst_display
=
410 gfx::Screen::GetScreenFor(window_
)->GetDisplayMatching(bounds
);
411 screen_position_client
->SetBounds(window_
, bounds
, dst_display
);
415 window_
->SetBounds(bounds
);
418 void NativeWidgetAura::SetSize(const gfx::Size
& size
) {
420 window_
->SetBounds(gfx::Rect(window_
->bounds().origin(), size
));
423 void NativeWidgetAura::StackAbove(gfx::NativeView native_view
) {
424 if (window_
&& window_
->parent() &&
425 window_
->parent() == native_view
->parent())
426 window_
->parent()->StackChildAbove(window_
, native_view
);
429 void NativeWidgetAura::StackAtTop() {
431 window_
->parent()->StackChildAtTop(window_
);
434 void NativeWidgetAura::StackBelow(gfx::NativeView native_view
) {
435 if (window_
&& window_
->parent() &&
436 window_
->parent() == native_view
->parent())
437 window_
->parent()->StackChildBelow(window_
, native_view
);
440 void NativeWidgetAura::SetShape(gfx::NativeRegion region
) {
442 window_
->layer()->SetAlphaShape(make_scoped_ptr(region
));
447 void NativeWidgetAura::Close() {
448 // |window_| may already be deleted by parent window. This can happen
449 // when this widget is child widget or has transient parent
450 // and ownership is WIDGET_OWNS_NATIVE_WIDGET.
452 ownership_
== Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET
);
454 window_
->SuppressPaint();
456 window_
->SetProperty(aura::client::kModalKey
, ui::MODAL_TYPE_NONE
);
459 if (!close_widget_factory_
.HasWeakPtrs()) {
460 base::MessageLoop::current()->PostTask(
462 base::Bind(&NativeWidgetAura::CloseNow
,
463 close_widget_factory_
.GetWeakPtr()));
467 void NativeWidgetAura::CloseNow() {
471 void NativeWidgetAura::Show() {
472 ShowWithWindowState(ui::SHOW_STATE_NORMAL
);
475 void NativeWidgetAura::Hide() {
480 void NativeWidgetAura::ShowMaximizedWithBounds(
481 const gfx::Rect
& restored_bounds
) {
482 SetRestoreBounds(window_
, restored_bounds
);
483 ShowWithWindowState(ui::SHOW_STATE_MAXIMIZED
);
486 void NativeWidgetAura::ShowWithWindowState(ui::WindowShowState state
) {
490 if (state
== ui::SHOW_STATE_MAXIMIZED
|| state
== ui::SHOW_STATE_FULLSCREEN
)
491 window_
->SetProperty(aura::client::kShowStateKey
, state
);
493 if (delegate_
->CanActivate()) {
494 if (state
!= ui::SHOW_STATE_INACTIVE
)
496 // SetInitialFocus() should be always be called, even for
497 // SHOW_STATE_INACTIVE. If the window has to stay inactive, the method will
498 // do the right thing.
499 SetInitialFocus(state
);
503 bool NativeWidgetAura::IsVisible() const {
504 return window_
&& window_
->IsVisible();
507 void NativeWidgetAura::Activate() {
511 // We don't necessarily have a root window yet. This can happen with
512 // constrained windows.
513 if (window_
->GetRootWindow()) {
514 aura::client::GetActivationClient(window_
->GetRootWindow())->ActivateWindow(
517 if (window_
->GetProperty(aura::client::kDrawAttentionKey
))
518 window_
->SetProperty(aura::client::kDrawAttentionKey
, false);
521 void NativeWidgetAura::Deactivate() {
524 aura::client::GetActivationClient(window_
->GetRootWindow())->DeactivateWindow(
528 bool NativeWidgetAura::IsActive() const {
529 return window_
&& wm::IsActiveWindow(window_
);
532 void NativeWidgetAura::SetAlwaysOnTop(bool on_top
) {
534 window_
->SetProperty(aura::client::kAlwaysOnTopKey
, on_top
);
537 bool NativeWidgetAura::IsAlwaysOnTop() const {
538 return window_
&& window_
->GetProperty(aura::client::kAlwaysOnTopKey
);
541 void NativeWidgetAura::SetVisibleOnAllWorkspaces(bool always_visible
) {
542 // Not implemented on chromeos or for child widgets.
545 void NativeWidgetAura::Maximize() {
547 window_
->SetProperty(aura::client::kShowStateKey
, ui::SHOW_STATE_MAXIMIZED
);
550 void NativeWidgetAura::Minimize() {
552 window_
->SetProperty(aura::client::kShowStateKey
, ui::SHOW_STATE_MINIMIZED
);
555 bool NativeWidgetAura::IsMaximized() const {
556 return window_
&& window_
->GetProperty(aura::client::kShowStateKey
) ==
557 ui::SHOW_STATE_MAXIMIZED
;
560 bool NativeWidgetAura::IsMinimized() const {
561 return window_
&& window_
->GetProperty(aura::client::kShowStateKey
) ==
562 ui::SHOW_STATE_MINIMIZED
;
565 void NativeWidgetAura::Restore() {
567 window_
->SetProperty(aura::client::kShowStateKey
, ui::SHOW_STATE_NORMAL
);
570 void NativeWidgetAura::SetFullscreen(bool fullscreen
) {
571 if (!window_
|| IsFullscreen() == fullscreen
)
572 return; // Nothing to do.
574 // Save window state before entering full screen so that it could restored
575 // when exiting full screen.
577 saved_window_state_
= window_
->GetProperty(aura::client::kShowStateKey
);
579 window_
->SetProperty(
580 aura::client::kShowStateKey
,
581 fullscreen
? ui::SHOW_STATE_FULLSCREEN
: saved_window_state_
);
584 bool NativeWidgetAura::IsFullscreen() const {
585 return window_
&& window_
->GetProperty(aura::client::kShowStateKey
) ==
586 ui::SHOW_STATE_FULLSCREEN
;
589 void NativeWidgetAura::SetOpacity(unsigned char opacity
) {
591 window_
->layer()->SetOpacity(opacity
/ 255.0);
594 void NativeWidgetAura::SetUseDragFrame(bool use_drag_frame
) {
598 void NativeWidgetAura::FlashFrame(bool flash
) {
600 window_
->SetProperty(aura::client::kDrawAttentionKey
, flash
);
603 void NativeWidgetAura::RunShellDrag(View
* view
,
604 const ui::OSExchangeData
& data
,
605 const gfx::Point
& location
,
607 ui::DragDropTypes::DragEventSource source
) {
609 views::RunShellDrag(window_
, data
, location
, operation
, source
);
612 void NativeWidgetAura::SchedulePaintInRect(const gfx::Rect
& rect
) {
614 window_
->SchedulePaintInRect(rect
);
617 void NativeWidgetAura::SetCursor(gfx::NativeCursor cursor
) {
619 aura::client::CursorClient
* cursor_client
=
620 aura::client::GetCursorClient(window_
->GetRootWindow());
622 cursor_client
->SetCursor(cursor
);
625 bool NativeWidgetAura::IsMouseEventsEnabled() const {
628 aura::client::CursorClient
* cursor_client
=
629 aura::client::GetCursorClient(window_
->GetRootWindow());
630 return cursor_client
? cursor_client
->IsMouseEventsEnabled() : true;
633 void NativeWidgetAura::ClearNativeFocus() {
634 aura::client::FocusClient
* client
= aura::client::GetFocusClient(window_
);
635 if (window_
&& client
&& window_
->Contains(client
->GetFocusedWindow()))
636 client
->ResetFocusWithinActiveWindow(window_
);
639 gfx::Rect
NativeWidgetAura::GetWorkAreaBoundsInScreen() const {
642 return gfx::Screen::GetScreenFor(window_
)->
643 GetDisplayNearestWindow(window_
).work_area();
646 Widget::MoveLoopResult
NativeWidgetAura::RunMoveLoop(
647 const gfx::Vector2d
& drag_offset
,
648 Widget::MoveLoopSource source
,
649 Widget::MoveLoopEscapeBehavior escape_behavior
) {
650 // |escape_behavior| is only needed on windows when running the native message
652 if (!window_
|| !window_
->GetRootWindow())
653 return Widget::MOVE_LOOP_CANCELED
;
654 aura::client::WindowMoveClient
* move_client
=
655 aura::client::GetWindowMoveClient(window_
->GetRootWindow());
657 return Widget::MOVE_LOOP_CANCELED
;
660 aura::client::WindowMoveSource window_move_source
=
661 source
== Widget::MOVE_LOOP_SOURCE_MOUSE
?
662 aura::client::WINDOW_MOVE_SOURCE_MOUSE
:
663 aura::client::WINDOW_MOVE_SOURCE_TOUCH
;
664 if (move_client
->RunMoveLoop(window_
, drag_offset
, window_move_source
) ==
665 aura::client::MOVE_SUCCESSFUL
) {
666 return Widget::MOVE_LOOP_SUCCESSFUL
;
668 return Widget::MOVE_LOOP_CANCELED
;
671 void NativeWidgetAura::EndMoveLoop() {
672 if (!window_
|| !window_
->GetRootWindow())
674 aura::client::WindowMoveClient
* move_client
=
675 aura::client::GetWindowMoveClient(window_
->GetRootWindow());
677 move_client
->EndMoveLoop();
680 void NativeWidgetAura::SetVisibilityChangedAnimationsEnabled(bool value
) {
682 window_
->SetProperty(aura::client::kAnimationsDisabledKey
, !value
);
685 ui::NativeTheme
* NativeWidgetAura::GetNativeTheme() const {
686 #if !defined(OS_CHROMEOS)
687 return DesktopWindowTreeHost::GetNativeTheme(window_
);
689 return ui::NativeThemeAura::instance();
693 void NativeWidgetAura::OnRootViewLayout() {
696 bool NativeWidgetAura::IsTranslucentWindowOpacitySupported() const {
700 void NativeWidgetAura::OnSizeConstraintsChanged() {
701 window_
->SetProperty(aura::client::kCanMaximizeKey
,
702 GetWidget()->widget_delegate()->CanMaximize());
703 window_
->SetProperty(aura::client::kCanMinimizeKey
,
704 GetWidget()->widget_delegate()->CanMinimize());
705 window_
->SetProperty(aura::client::kCanResizeKey
,
706 GetWidget()->widget_delegate()->CanResize());
709 void NativeWidgetAura::RepostNativeEvent(gfx::NativeEvent native_event
) {
710 OnEvent(native_event
);
713 ////////////////////////////////////////////////////////////////////////////////
714 // NativeWidgetAura, views::InputMethodDelegate implementation:
716 void NativeWidgetAura::DispatchKeyEventPostIME(const ui::KeyEvent
& key
) {
717 FocusManager
* focus_manager
= GetWidget()->GetFocusManager();
718 delegate_
->OnKeyEvent(const_cast<ui::KeyEvent
*>(&key
));
719 if (key
.handled() || !focus_manager
)
721 focus_manager
->OnKeyEvent(key
);
724 ////////////////////////////////////////////////////////////////////////////////
725 // NativeWidgetAura, aura::WindowDelegate implementation:
727 gfx::Size
NativeWidgetAura::GetMinimumSize() const {
728 return delegate_
->GetMinimumSize();
731 gfx::Size
NativeWidgetAura::GetMaximumSize() const {
732 // If a window have a maximum size, the window should not be
734 DCHECK(delegate_
->GetMaximumSize().IsEmpty() ||
735 !window_
->GetProperty(aura::client::kCanMaximizeKey
));
736 return delegate_
->GetMaximumSize();
739 void NativeWidgetAura::OnBoundsChanged(const gfx::Rect
& old_bounds
,
740 const gfx::Rect
& new_bounds
) {
741 // Assume that if the old bounds was completely empty a move happened. This
742 // handles the case of a maximize animation acquiring the layer (acquiring a
743 // layer results in clearing the bounds).
744 if (old_bounds
.origin() != new_bounds
.origin() ||
745 (old_bounds
== gfx::Rect(0, 0, 0, 0) && !new_bounds
.IsEmpty())) {
746 delegate_
->OnNativeWidgetMove();
748 if (old_bounds
.size() != new_bounds
.size())
749 delegate_
->OnNativeWidgetSizeChanged(new_bounds
.size());
752 gfx::NativeCursor
NativeWidgetAura::GetCursor(const gfx::Point
& point
) {
756 int NativeWidgetAura::GetNonClientComponent(const gfx::Point
& point
) const {
757 return delegate_
->GetNonClientComponent(point
);
760 bool NativeWidgetAura::ShouldDescendIntoChildForEventHandling(
762 const gfx::Point
& location
) {
763 views::WidgetDelegate
* widget_delegate
= GetWidget()->widget_delegate();
764 if (widget_delegate
&&
765 !widget_delegate
->ShouldDescendIntoChildForEventHandling(child
, location
))
768 // Don't descend into |child| if there is a view with a Layer that contains
769 // the point and is stacked above |child|s layer.
770 typedef std::vector
<ui::Layer
*> Layers
;
771 const Layers
& root_layers(delegate_
->GetRootLayers());
772 if (root_layers
.empty())
775 Layers::const_iterator
child_layer_iter(
776 std::find(window_
->layer()->children().begin(),
777 window_
->layer()->children().end(), child
->layer()));
778 if (child_layer_iter
== window_
->layer()->children().end())
781 for (std::vector
<ui::Layer
*>::const_reverse_iterator i
= root_layers
.rbegin();
782 i
!= root_layers
.rend(); ++i
) {
783 ui::Layer
* layer
= *i
;
784 if (layer
->visible() && layer
->bounds().Contains(location
)) {
785 Layers::const_iterator
root_layer_iter(
786 std::find(window_
->layer()->children().begin(),
787 window_
->layer()->children().end(), layer
));
788 if (root_layer_iter
> child_layer_iter
)
795 bool NativeWidgetAura::CanFocus() {
796 return ShouldActivate();
799 void NativeWidgetAura::OnCaptureLost() {
800 delegate_
->OnMouseCaptureLost();
803 void NativeWidgetAura::OnPaint(gfx::Canvas
* canvas
) {
804 delegate_
->OnNativeWidgetPaint(canvas
);
807 void NativeWidgetAura::OnDeviceScaleFactorChanged(float device_scale_factor
) {
808 // Repainting with new scale factor will paint the content at the right scale.
811 void NativeWidgetAura::OnWindowDestroying(aura::Window
* window
) {
812 window_
->RemoveObserver(this);
813 delegate_
->OnNativeWidgetDestroying();
815 // If the aura::Window is destroyed, we can no longer show tooltips.
816 tooltip_manager_
.reset();
819 void NativeWidgetAura::OnWindowDestroyed(aura::Window
* window
) {
821 delegate_
->OnNativeWidgetDestroyed();
822 if (ownership_
== Widget::InitParams::NATIVE_WIDGET_OWNS_WIDGET
)
826 void NativeWidgetAura::OnWindowTargetVisibilityChanged(bool visible
) {
827 delegate_
->OnNativeWidgetVisibilityChanged(visible
);
830 bool NativeWidgetAura::HasHitTestMask() const {
831 return delegate_
->HasHitTestMask();
834 void NativeWidgetAura::GetHitTestMask(gfx::Path
* mask
) const {
836 delegate_
->GetHitTestMask(mask
);
839 ////////////////////////////////////////////////////////////////////////////////
840 // NativeWidgetAura, aura::WindowObserver implementation:
842 void NativeWidgetAura::OnWindowPropertyChanged(aura::Window
* window
,
845 if (key
== aura::client::kShowStateKey
)
846 delegate_
->OnNativeWidgetWindowShowStateChanged();
849 ////////////////////////////////////////////////////////////////////////////////
850 // NativeWidgetAura, ui::EventHandler implementation:
852 void NativeWidgetAura::OnKeyEvent(ui::KeyEvent
* event
) {
854 if (event
->is_char()) {
855 // If a ui::InputMethod object is attached to the root window, character
856 // events are handled inside the object and are not passed to this function.
857 // If such object is not attached, character events might be sent (e.g. on
858 // Windows). In this case, we just skip these.
861 // Renderer may send a key event back to us if the key event wasn't handled,
862 // and the window may be invisible by that time.
863 if (!window_
->IsVisible())
865 InputMethod
* input_method
= GetWidget()->GetInputMethod();
868 input_method
->DispatchKeyEvent(*event
);
869 if (switches::IsTextInputFocusManagerEnabled()) {
870 FocusManager
* focus_manager
= GetWidget()->GetFocusManager();
871 delegate_
->OnKeyEvent(event
);
872 if (!event
->handled() && focus_manager
)
873 focus_manager
->OnKeyEvent(*event
);
878 void NativeWidgetAura::OnMouseEvent(ui::MouseEvent
* event
) {
880 DCHECK(window_
->IsVisible());
881 if (event
->type() == ui::ET_MOUSEWHEEL
) {
882 delegate_
->OnMouseEvent(event
);
883 if (event
->handled())
887 if (tooltip_manager_
.get())
888 tooltip_manager_
->UpdateTooltip();
889 TooltipManagerAura::UpdateTooltipManagerForCapture(GetWidget());
890 delegate_
->OnMouseEvent(event
);
893 void NativeWidgetAura::OnScrollEvent(ui::ScrollEvent
* event
) {
894 delegate_
->OnScrollEvent(event
);
897 void NativeWidgetAura::OnGestureEvent(ui::GestureEvent
* event
) {
899 DCHECK(window_
->IsVisible() || event
->IsEndingEvent());
900 delegate_
->OnGestureEvent(event
);
903 ////////////////////////////////////////////////////////////////////////////////
904 // NativeWidgetAura, aura::client::ActivationDelegate implementation:
906 bool NativeWidgetAura::ShouldActivate() const {
907 return delegate_
->CanActivate();
910 ////////////////////////////////////////////////////////////////////////////////
911 // NativeWidgetAura, aura::client::ActivationChangeObserver implementation:
913 void NativeWidgetAura::OnWindowActivated(aura::Window
* gained_active
,
914 aura::Window
* lost_active
) {
915 DCHECK(window_
== gained_active
|| window_
== lost_active
);
916 if (GetWidget()->GetFocusManager()) {
917 if (window_
== gained_active
)
918 GetWidget()->GetFocusManager()->RestoreFocusedView();
919 else if (window_
== lost_active
)
920 GetWidget()->GetFocusManager()->StoreFocusedView(true);
922 delegate_
->OnNativeWidgetActivationChanged(window_
== gained_active
);
925 ////////////////////////////////////////////////////////////////////////////////
926 // NativeWidgetAura, aura::client::FocusChangeObserver:
928 void NativeWidgetAura::OnWindowFocused(aura::Window
* gained_focus
,
929 aura::Window
* lost_focus
) {
930 if (window_
== gained_focus
) {
931 // In aura, it is possible for child native widgets to take input and focus,
932 // this differs from the behavior on windows.
933 if (GetWidget()->GetInputMethod()) // Null in tests.
934 GetWidget()->GetInputMethod()->OnFocus();
935 delegate_
->OnNativeFocus(lost_focus
);
936 } else if (window_
== lost_focus
) {
937 // GetInputMethod() recreates the input method if it's previously been
938 // destroyed. If we get called during destruction, the input method will be
939 // gone, and creating a new one and telling it that we lost the focus will
940 // trigger a DCHECK (the new input method doesn't think that we have the
941 // focus and doesn't expect a blur). OnBlur() shouldn't be called during
942 // destruction unless WIDGET_OWNS_NATIVE_WIDGET is set (which is just the
945 if (GetWidget()->GetInputMethod())
946 GetWidget()->GetInputMethod()->OnBlur();
948 DCHECK_EQ(ownership_
, Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET
);
951 delegate_
->OnNativeBlur(gained_focus
);
955 ////////////////////////////////////////////////////////////////////////////////
956 // NativeWidgetAura, aura::WindowDragDropDelegate implementation:
958 void NativeWidgetAura::OnDragEntered(const ui::DropTargetEvent
& event
) {
959 DCHECK(drop_helper_
.get() != NULL
);
960 last_drop_operation_
= drop_helper_
->OnDragOver(event
.data(),
961 event
.location(), event
.source_operations());
964 int NativeWidgetAura::OnDragUpdated(const ui::DropTargetEvent
& event
) {
965 DCHECK(drop_helper_
.get() != NULL
);
966 last_drop_operation_
= drop_helper_
->OnDragOver(event
.data(),
967 event
.location(), event
.source_operations());
968 return last_drop_operation_
;
971 void NativeWidgetAura::OnDragExited() {
972 DCHECK(drop_helper_
.get() != NULL
);
973 drop_helper_
->OnDragExit();
976 int NativeWidgetAura::OnPerformDrop(const ui::DropTargetEvent
& event
) {
977 DCHECK(drop_helper_
.get() != NULL
);
978 return drop_helper_
->OnDrop(event
.data(), event
.location(),
979 last_drop_operation_
);
982 ////////////////////////////////////////////////////////////////////////////////
983 // NativeWidgetAura, protected:
985 NativeWidgetAura::~NativeWidgetAura() {
987 if (ownership_
== Widget::InitParams::NATIVE_WIDGET_OWNS_WIDGET
)
993 ////////////////////////////////////////////////////////////////////////////////
994 // NativeWidgetAura, private:
996 void NativeWidgetAura::SetInitialFocus(ui::WindowShowState show_state
) {
997 // The window does not get keyboard messages unless we focus it.
998 if (!GetWidget()->SetInitialFocus(show_state
))
1002 ////////////////////////////////////////////////////////////////////////////////
1006 #if defined(OS_WIN) || (defined(USE_X11) && !defined(OS_CHROMEOS))
1007 void CloseWindow(aura::Window
* window
) {
1009 Widget
* widget
= Widget::GetWidgetForNativeView(window
);
1010 if (widget
&& widget
->is_secondary_widget())
1011 // To avoid the delay in shutdown caused by using Close which may wait
1012 // for animations, use CloseNow. Because this is only used on secondary
1013 // widgets it seems relatively safe to skip the extra processing of
1021 BOOL CALLBACK
WindowCallbackProc(HWND hwnd
, LPARAM lParam
) {
1022 aura::Window
* root_window
=
1023 DesktopWindowTreeHostWin::GetContentWindowForHWND(hwnd
);
1024 CloseWindow(root_window
);
1031 void Widget::CloseAllSecondaryWidgets() {
1033 EnumThreadWindows(GetCurrentThreadId(), WindowCallbackProc
, 0);
1036 #if defined(USE_X11) && !defined(OS_CHROMEOS)
1037 std::vector
<aura::Window
*> open_windows
=
1038 DesktopWindowTreeHostX11::GetAllOpenWindows();
1039 std::for_each(open_windows
.begin(), open_windows
.end(), CloseWindow
);
1040 DesktopWindowTreeHostX11::CleanUpWindowList();
1044 bool Widget::ConvertRect(const Widget
* source
,
1045 const Widget
* target
,
1050 namespace internal
{
1052 ////////////////////////////////////////////////////////////////////////////////
1053 // internal::NativeWidgetPrivate, public:
1056 NativeWidgetPrivate
* NativeWidgetPrivate::CreateNativeWidget(
1057 internal::NativeWidgetDelegate
* delegate
) {
1058 return new NativeWidgetAura(delegate
);
1062 NativeWidgetPrivate
* NativeWidgetPrivate::GetNativeWidgetForNativeView(
1063 gfx::NativeView native_view
) {
1064 // Cast must match type supplied to RegisterNativeWidgetForWindow().
1065 return reinterpret_cast<NativeWidgetPrivate
*>(native_view
->user_data());
1069 NativeWidgetPrivate
* NativeWidgetPrivate::GetNativeWidgetForNativeWindow(
1070 gfx::NativeWindow native_window
) {
1071 // Cast must match type supplied to RegisterNativeWidgetForWindow().
1072 return reinterpret_cast<NativeWidgetPrivate
*>(native_window
->user_data());
1076 NativeWidgetPrivate
* NativeWidgetPrivate::GetTopLevelNativeWidget(
1077 gfx::NativeView native_view
) {
1078 aura::Window
* window
= native_view
;
1079 NativeWidgetPrivate
* top_level_native_widget
= NULL
;
1081 NativeWidgetPrivate
* native_widget
= GetNativeWidgetForNativeView(window
);
1083 top_level_native_widget
= native_widget
;
1084 window
= window
->parent();
1086 return top_level_native_widget
;
1090 void NativeWidgetPrivate::GetAllChildWidgets(gfx::NativeView native_view
,
1091 Widget::Widgets
* children
) {
1093 // Code expects widget for |native_view| to be added to |children|.
1094 NativeWidgetPrivate
* native_widget
= static_cast<NativeWidgetPrivate
*>(
1095 GetNativeWidgetForNativeView(native_view
));
1096 if (native_widget
&& native_widget
->GetWidget())
1097 children
->insert(native_widget
->GetWidget());
1100 const aura::Window::Windows
& child_windows
= native_view
->children();
1101 for (aura::Window::Windows::const_iterator i
= child_windows
.begin();
1102 i
!= child_windows
.end(); ++i
) {
1103 GetAllChildWidgets((*i
), children
);
1108 void NativeWidgetPrivate::GetAllOwnedWidgets(gfx::NativeView native_view
,
1109 Widget::Widgets
* owned
) {
1110 // Add all owned widgets.
1111 for (aura::Window
* transient_child
: wm::GetTransientChildren(native_view
)) {
1112 NativeWidgetPrivate
* native_widget
= static_cast<NativeWidgetPrivate
*>(
1113 GetNativeWidgetForNativeView(transient_child
));
1114 if (native_widget
&& native_widget
->GetWidget())
1115 owned
->insert(native_widget
->GetWidget());
1116 GetAllOwnedWidgets(transient_child
, owned
);
1119 // Add all child windows.
1120 for (aura::Window
* child
: native_view
->children())
1121 GetAllChildWidgets(child
, owned
);
1125 void NativeWidgetPrivate::ReparentNativeView(gfx::NativeView native_view
,
1126 gfx::NativeView new_parent
) {
1127 DCHECK(native_view
!= new_parent
);
1129 gfx::NativeView previous_parent
= native_view
->parent();
1130 if (previous_parent
== new_parent
)
1133 Widget::Widgets widgets
;
1134 GetAllChildWidgets(native_view
, &widgets
);
1136 // First notify all the widgets that they are being disassociated
1137 // from their previous parent.
1138 for (Widget::Widgets::iterator it
= widgets
.begin();
1139 it
!= widgets
.end(); ++it
) {
1140 (*it
)->NotifyNativeViewHierarchyWillChange();
1144 new_parent
->AddChild(native_view
);
1146 // The following looks weird, but it's the equivalent of what aura has
1147 // always done. (The previous behaviour of aura::Window::SetParent() used
1148 // NULL as a special value that meant ask the WindowTreeClient where things
1151 // This probably isn't strictly correct, but its an invariant that a Window
1152 // in use will be attached to a RootWindow, so we can't just call
1153 // RemoveChild here. The only possible thing that could assign a RootWindow
1154 // in this case is the stacking client of the current RootWindow. This
1155 // matches our previous behaviour; the global stacking client would almost
1156 // always reattach the window to the same RootWindow.
1157 aura::Window
* root_window
= native_view
->GetRootWindow();
1158 aura::client::ParentWindowWithContext(
1159 native_view
, root_window
, root_window
->GetBoundsInScreen());
1162 // And now, notify them that they have a brand new parent.
1163 for (Widget::Widgets::iterator it
= widgets
.begin();
1164 it
!= widgets
.end(); ++it
) {
1165 (*it
)->NotifyNativeViewHierarchyChanged();
1170 bool NativeWidgetPrivate::IsMouseButtonDown() {
1171 return aura::Env::GetInstance()->IsMouseButtonDown();
1175 gfx::FontList
NativeWidgetPrivate::GetWindowTitleFontList() {
1177 NONCLIENTMETRICS_XP ncm
;
1178 base::win::GetNonClientMetrics(&ncm
);
1179 l10n_util::AdjustUIFont(&(ncm
.lfCaptionFont
));
1180 base::win::ScopedHFONT
caption_font(CreateFontIndirect(&(ncm
.lfCaptionFont
)));
1181 return gfx::FontList(gfx::Font(caption_font
));
1183 return gfx::FontList();
1187 } // namespace internal
1188 } // namespace views