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 "ash/wm/dock/docked_window_resizer.h"
7 #include "ash/display/display_controller.h"
8 #include "ash/root_window_controller.h"
9 #include "ash/screen_util.h"
10 #include "ash/shelf/shelf.h"
11 #include "ash/shelf/shelf_types.h"
12 #include "ash/shelf/shelf_widget.h"
13 #include "ash/shell.h"
14 #include "ash/shell_window_ids.h"
15 #include "ash/wm/dock/docked_window_layout_manager.h"
16 #include "ash/wm/window_state.h"
17 #include "ash/wm/window_util.h"
18 #include "ash/wm/wm_event.h"
19 #include "ash/wm/workspace/magnetism_matcher.h"
20 #include "ash/wm/workspace/workspace_window_resizer.h"
21 #include "base/command_line.h"
22 #include "base/memory/weak_ptr.h"
23 #include "ui/aura/client/aura_constants.h"
24 #include "ui/aura/client/window_tree_client.h"
25 #include "ui/aura/env.h"
26 #include "ui/aura/window.h"
27 #include "ui/aura/window_delegate.h"
28 #include "ui/aura/window_event_dispatcher.h"
29 #include "ui/base/hit_test.h"
30 #include "ui/base/ui_base_types.h"
31 #include "ui/gfx/screen.h"
32 #include "ui/views/widget/widget.h"
33 #include "ui/wm/core/coordinate_conversion.h"
38 DockedWindowLayoutManager
* GetDockedLayoutManagerAtPoint(
39 const gfx::Point
& point
) {
40 gfx::Display display
= ScreenUtil::FindDisplayContainingPoint(point
);
41 if (!display
.is_valid())
43 aura::Window
* root
= Shell::GetInstance()->display_controller()->
44 GetRootWindowForDisplayId(display
.id());
45 aura::Window
* dock_container
= Shell::GetContainer(
46 root
, kShellWindowId_DockedContainer
);
47 return static_cast<DockedWindowLayoutManager
*>(
48 dock_container
->layout_manager());
53 DockedWindowResizer::~DockedWindowResizer() {
58 DockedWindowResizer::Create(WindowResizer
* next_window_resizer
,
59 wm::WindowState
* window_state
) {
60 return new DockedWindowResizer(next_window_resizer
, window_state
);
63 void DockedWindowResizer::Drag(const gfx::Point
& location
, int event_flags
) {
64 last_location_
= location
;
65 ::wm::ConvertPointToScreen(GetTarget()->parent(), &last_location_
);
66 if (!did_move_or_resize_
) {
67 did_move_or_resize_
= true;
71 gfx::Rect
bounds(CalculateBoundsForDrag(location
));
72 MaybeSnapToEdge(bounds
, &offset
);
73 gfx::Point
modified_location(location
);
74 modified_location
.Offset(offset
.x(), offset
.y());
76 base::WeakPtr
<DockedWindowResizer
> resizer(weak_ptr_factory_
.GetWeakPtr());
77 next_window_resizer_
->Drag(modified_location
, event_flags
);
81 DockedWindowLayoutManager
* new_dock_layout
=
82 GetDockedLayoutManagerAtPoint(last_location_
);
83 if (new_dock_layout
&& new_dock_layout
!= dock_layout_
) {
84 // The window is being dragged to a new display. If the previous
85 // container is the current parent of the window it will be informed of
86 // the end of drag when the window is reparented, otherwise let the
87 // previous container know the drag is complete. If we told the
88 // window's parent that the drag was complete it would begin
89 // positioning the window.
90 if (is_docked_
&& dock_layout_
->is_dragged_window_docked())
91 dock_layout_
->UndockDraggedWindow();
92 if (dock_layout_
!= initial_dock_layout_
)
93 dock_layout_
->FinishDragging(
95 details().source
== aura::client::WINDOW_MOVE_SOURCE_MOUSE
?
96 DOCKED_ACTION_SOURCE_MOUSE
: DOCKED_ACTION_SOURCE_TOUCH
);
98 dock_layout_
= new_dock_layout
;
99 // The window's initial layout manager already knows that the drag is
100 // in progress for this window.
101 if (new_dock_layout
!= initial_dock_layout_
)
102 new_dock_layout
->StartDragging(GetTarget());
104 // Window could get docked by the WorkspaceWindowResizer, update the state.
105 is_docked_
= dock_layout_
->is_dragged_window_docked();
106 // Whenever a window is dragged out of the dock it will be auto-sized
107 // in the dock if it gets docked again.
109 was_bounds_changed_by_user_
= false;
112 void DockedWindowResizer::CompleteDrag() {
113 // The root window can change when dragging into a different screen.
114 next_window_resizer_
->CompleteDrag();
115 FinishedDragging(aura::client::MOVE_SUCCESSFUL
);
118 void DockedWindowResizer::RevertDrag() {
119 next_window_resizer_
->RevertDrag();
120 // Restore docked state to what it was before the drag if necessary.
121 if (is_docked_
!= was_docked_
) {
122 is_docked_
= was_docked_
;
124 dock_layout_
->DockDraggedWindow(GetTarget());
126 dock_layout_
->UndockDraggedWindow();
128 FinishedDragging(aura::client::MOVE_CANCELED
);
131 DockedWindowResizer::DockedWindowResizer(WindowResizer
* next_window_resizer
,
132 wm::WindowState
* window_state
)
133 : WindowResizer(window_state
),
134 next_window_resizer_(next_window_resizer
),
136 initial_dock_layout_(NULL
),
137 did_move_or_resize_(false),
140 was_bounds_changed_by_user_(window_state
->bounds_changed_by_user()),
141 weak_ptr_factory_(this) {
142 DCHECK(details().is_resizable
);
143 aura::Window
* dock_container
= Shell::GetContainer(
144 GetTarget()->GetRootWindow(),
145 kShellWindowId_DockedContainer
);
146 dock_layout_
= static_cast<DockedWindowLayoutManager
*>(
147 dock_container
->layout_manager());
148 initial_dock_layout_
= dock_layout_
;
149 was_docked_
= GetTarget()->parent() == dock_container
;
150 is_docked_
= was_docked_
;
153 void DockedWindowResizer::MaybeSnapToEdge(const gfx::Rect
& bounds
,
154 gfx::Point
* offset
) {
155 // Windows only snap magnetically when they were previously docked.
158 DockedAlignment dock_alignment
= dock_layout_
->CalculateAlignment();
159 gfx::Rect dock_bounds
= ScreenUtil::ConvertRectFromScreen(
160 GetTarget()->parent(),
161 dock_layout_
->dock_container()->GetBoundsInScreen());
163 // Short-range magnetism when retaining docked state. Same constant as in
164 // MagnetismMatcher is used for consistency.
165 const int kSnapToDockDistance
= MagnetismMatcher::kMagneticDistance
;
167 if (dock_alignment
== DOCKED_ALIGNMENT_LEFT
||
168 dock_alignment
== DOCKED_ALIGNMENT_NONE
) {
169 const int distance
= bounds
.x() - dock_bounds
.x();
170 if (distance
< kSnapToDockDistance
&& distance
> 0) {
171 offset
->set_x(-distance
);
175 if (dock_alignment
== DOCKED_ALIGNMENT_RIGHT
||
176 dock_alignment
== DOCKED_ALIGNMENT_NONE
) {
177 const int distance
= dock_bounds
.right() - bounds
.right();
178 if (distance
< kSnapToDockDistance
&& distance
> 0)
179 offset
->set_x(distance
);
183 void DockedWindowResizer::StartedDragging() {
184 // During resizing the window width is preserved by DockedwindowLayoutManager.
186 (details().bounds_change
& WindowResizer::kBoundsChange_Resizes
)) {
187 window_state_
->set_bounds_changed_by_user(true);
190 // Tell the dock layout manager that we are dragging this window.
191 // At this point we are not yet animating the window as it may not be
192 // inside the docked area.
193 dock_layout_
->StartDragging(GetTarget());
194 // Reparent workspace windows during the drag to elevate them above workspace.
195 // Other windows for which the DockedWindowResizer is instantiated include
196 // panels and windows that are already docked. Those do not need reparenting.
197 if (GetTarget()->type() != ui::wm::WINDOW_TYPE_PANEL
&&
198 GetTarget()->parent()->id() == kShellWindowId_DefaultContainer
) {
199 // Reparent the window into the docked windows container in order to get it
200 // on top of other docked windows.
201 aura::Window
* docked_container
= Shell::GetContainer(
202 GetTarget()->GetRootWindow(),
203 kShellWindowId_DockedContainer
);
204 wm::ReparentChildWithTransientChildren(GetTarget(),
205 GetTarget()->parent(),
209 dock_layout_
->DockDraggedWindow(GetTarget());
212 void DockedWindowResizer::FinishedDragging(
213 aura::client::WindowMoveResult move_result
) {
214 if (!did_move_or_resize_
)
216 did_move_or_resize_
= false;
217 aura::Window
* window
= GetTarget();
218 const bool is_attached_panel
= window
->type() == ui::wm::WINDOW_TYPE_PANEL
&&
219 window_state_
->panel_attached();
220 const bool is_resized
=
221 (details().bounds_change
& WindowResizer::kBoundsChange_Resizes
) != 0;
223 // Undock the window if it is not in the normal, docked or minimized state
224 // type. This happens if a user snaps or maximizes a window using a
225 // keyboard shortcut while it is being dragged.
226 if (!window_state_
->IsMinimized() && !window_state_
->IsDocked() &&
227 !window_state_
->IsNormalStateType())
230 // When drag is completed the dragged docked window is resized to the bounds
231 // calculated by the layout manager that conform to other docked windows.
232 if (!is_attached_panel
&& is_docked_
&& !is_resized
) {
233 gfx::Rect bounds
= ScreenUtil::ConvertRectFromScreen(
234 window
->parent(), dock_layout_
->dragged_bounds());
235 if (!bounds
.IsEmpty() && bounds
.width() != window
->bounds().width()) {
236 window
->SetBounds(bounds
);
239 // If a window has restore bounds, update the restore origin but not the size.
240 // The size gets restored when a window is undocked.
241 if (is_resized
&& is_docked_
&& window_state_
->HasRestoreBounds()) {
242 gfx::Rect restore_bounds
= window
->GetBoundsInScreen();
243 restore_bounds
.set_size(window_state_
->GetRestoreBoundsInScreen().size());
244 window_state_
->SetRestoreBoundsInScreen(restore_bounds
);
247 // Check if the window needs to be docked or returned to workspace.
248 DockedAction action
= MaybeReparentWindowOnDragCompletion(is_resized
,
250 dock_layout_
->FinishDragging(
251 move_result
== aura::client::MOVE_CANCELED
? DOCKED_ACTION_NONE
: action
,
252 details().source
== aura::client::WINDOW_MOVE_SOURCE_MOUSE
?
253 DOCKED_ACTION_SOURCE_MOUSE
: DOCKED_ACTION_SOURCE_TOUCH
);
255 // If we started the drag in one root window and moved into another root
256 // but then canceled the drag we may need to inform the original layout
257 // manager that the drag is finished.
258 if (initial_dock_layout_
!= dock_layout_
)
259 initial_dock_layout_
->FinishDragging(
261 details().source
== aura::client::WINDOW_MOVE_SOURCE_MOUSE
?
262 DOCKED_ACTION_SOURCE_MOUSE
: DOCKED_ACTION_SOURCE_TOUCH
);
266 DockedAction
DockedWindowResizer::MaybeReparentWindowOnDragCompletion(
267 bool is_resized
, bool is_attached_panel
) {
268 aura::Window
* window
= GetTarget();
270 // Check if the window needs to be docked or returned to workspace.
271 DockedAction action
= DOCKED_ACTION_NONE
;
272 aura::Window
* dock_container
= Shell::GetContainer(
273 window
->GetRootWindow(),
274 kShellWindowId_DockedContainer
);
275 if ((is_resized
|| !is_attached_panel
) &&
276 is_docked_
!= (window
->parent() == dock_container
)) {
278 wm::ReparentChildWithTransientChildren(window
,
281 action
= DOCKED_ACTION_DOCK
;
282 } else if (window
->parent()->id() == kShellWindowId_DockedContainer
) {
283 // Reparent the window back to workspace.
284 // We need to be careful to give ParentWindowWithContext a location in
285 // the right root window (matching the logic in DragWindowResizer) based
286 // on which root window a mouse pointer is in. We want to undock into the
287 // right screen near the edge of a multiscreen setup (based on where the
289 gfx::Rect
near_last_location(last_location_
, gfx::Size());
290 // Reparenting will cause Relayout and possible dock shrinking.
291 aura::Window
* previous_parent
= window
->parent();
292 aura::client::ParentWindowWithContext(window
, window
, near_last_location
);
293 if (window
->parent() != previous_parent
) {
294 wm::ReparentTransientChildrenOfChild(window
,
298 action
= was_docked_
? DOCKED_ACTION_UNDOCK
: DOCKED_ACTION_NONE
;
301 // |action| is recorded in UMA and used to maintain |window_state_|.
302 if (is_resized
&& is_docked_
&& was_docked_
)
303 action
= DOCKED_ACTION_RESIZE
;
304 else if (is_docked_
&& was_docked_
)
305 action
= DOCKED_ACTION_REORDER
;
306 else if (is_docked_
&& !was_docked_
)
307 action
= DOCKED_ACTION_DOCK
;
308 else if (!is_docked_
&& was_docked_
)
309 action
= DOCKED_ACTION_UNDOCK
;
311 action
= DOCKED_ACTION_NONE
;
313 // When a window is newly docked it is auto-sized by docked layout adjusting
314 // to other windows. If it is just dragged (but not resized) while being
315 // docked it is auto-sized unless it has been resized while being docked
318 wm::GetWindowState(window
)->set_bounds_changed_by_user(
319 was_docked_
&& (is_resized
|| was_bounds_changed_by_user_
));
322 if (action
== DOCKED_ACTION_DOCK
) {
323 const wm::WMEvent
event(wm::WM_EVENT_DOCK
);
324 window_state_
->OnWMEvent(&event
);
325 } else if (wm::GetWindowState(window
)->IsDocked() &&
326 action
== DOCKED_ACTION_UNDOCK
) {
327 const wm::WMEvent
event(wm::WM_EVENT_NORMAL
);
328 window_state_
->OnWMEvent(&event
);