1 // Copyright (c) 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_layout_manager.h"
7 #include "ash/screen_util.h"
8 #include "ash/shelf/shelf.h"
9 #include "ash/shelf/shelf_constants.h"
10 #include "ash/shelf/shelf_layout_manager.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/coordinate_conversion.h"
16 #include "ash/wm/window_animations.h"
17 #include "ash/wm/window_properties.h"
18 #include "ash/wm/window_resizer.h"
19 #include "ash/wm/window_state.h"
20 #include "ash/wm/window_util.h"
21 #include "ash/wm/workspace_controller.h"
22 #include "base/auto_reset.h"
23 #include "base/command_line.h"
24 #include "base/metrics/histogram.h"
25 #include "grit/ash_resources.h"
26 #include "third_party/skia/include/core/SkColor.h"
27 #include "third_party/skia/include/core/SkPaint.h"
28 #include "ui/aura/client/focus_client.h"
29 #include "ui/aura/client/window_tree_client.h"
30 #include "ui/aura/window.h"
31 #include "ui/aura/window_delegate.h"
32 #include "ui/aura/window_event_dispatcher.h"
33 #include "ui/base/resource/resource_bundle.h"
34 #include "ui/compositor/scoped_layer_animation_settings.h"
35 #include "ui/gfx/canvas.h"
36 #include "ui/gfx/geometry/rect.h"
37 #include "ui/gfx/image/image_skia_operations.h"
38 #include "ui/views/background.h"
39 #include "ui/wm/core/window_util.h"
40 #include "ui/wm/public/activation_client.h"
44 // Minimum, maximum width of the dock area and a width of the gap
46 const int DockedWindowLayoutManager::kMaxDockWidth
= 360;
48 const int DockedWindowLayoutManager::kMinDockWidth
= 200;
50 const int DockedWindowLayoutManager::kMinDockGap
= 2;
52 const int DockedWindowLayoutManager::kIdealWidth
= 250;
53 const int kMinimumHeight
= 250;
54 const int kSlideDurationMs
= 120;
55 const int kFadeDurationMs
= 60;
56 const int kMinimizeDurationMs
= 720;
58 class DockedBackgroundWidget
: public views::Widget
,
59 public BackgroundAnimatorDelegate
{
61 explicit DockedBackgroundWidget(aura::Window
* container
)
62 : alignment_(DOCKED_ALIGNMENT_NONE
),
63 background_animator_(this, 0, kShelfBackgroundAlpha
),
65 opaque_background_(ui::LAYER_SOLID_COLOR
),
66 visible_background_type_(SHELF_BACKGROUND_DEFAULT
),
67 visible_background_change_type_(BACKGROUND_CHANGE_IMMEDIATE
) {
68 InitWidget(container
);
71 // Sets widget bounds and sizes opaque background layer to fill the widget.
72 void SetBackgroundBounds(const gfx::Rect bounds
, DockedAlignment alignment
) {
74 opaque_background_
.SetBounds(gfx::Rect(bounds
.size()));
75 alignment_
= alignment
;
78 // Sets the background type. Starts an animation to transition to
79 // |background_type| if the widget is visible. If the widget is not visible,
80 // the animation is postponed till the widget becomes visible.
81 void SetBackgroundType(ShelfBackgroundType background_type
,
82 BackgroundAnimatorChangeType change_type
) {
83 visible_background_type_
= background_type
;
84 visible_background_change_type_
= change_type
;
90 void OnNativeWidgetVisibilityChanged(bool visible
) override
{
91 views::Widget::OnNativeWidgetVisibilityChanged(visible
);
95 void OnNativeWidgetPaint(gfx::Canvas
* canvas
) override
{
96 const gfx::ImageSkia
& shelf_background(
97 alignment_
== DOCKED_ALIGNMENT_LEFT
?
98 shelf_background_left_
: shelf_background_right_
);
99 gfx::Rect rect
= gfx::Rect(GetWindowBoundsInScreen().size());
101 paint
.setAlpha(alpha_
);
102 canvas
->DrawImageInt(shelf_background
,
105 shelf_background
.width(),
106 shelf_background
.height(),
107 alignment_
== DOCKED_ALIGNMENT_LEFT
108 ? rect
.width() - shelf_background
.width()
111 shelf_background
.width(),
115 canvas
->DrawImageInt(
117 alignment_
== DOCKED_ALIGNMENT_LEFT
? 0 : shelf_background
.width() - 1,
120 shelf_background
.height(),
121 alignment_
== DOCKED_ALIGNMENT_LEFT
? 0 : shelf_background
.width(),
123 rect
.width() - shelf_background
.width(),
129 // BackgroundAnimatorDelegate:
130 void UpdateBackground(int alpha
) override
{
132 SchedulePaintInRect(gfx::Rect(GetWindowBoundsInScreen().size()));
136 void InitWidget(aura::Window
* parent
) {
137 views::Widget::InitParams params
;
138 params
.type
= views::Widget::InitParams::TYPE_POPUP
;
139 params
.opacity
= views::Widget::InitParams::TRANSLUCENT_WINDOW
;
140 params
.keep_on_top
= false;
141 params
.ownership
= views::Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET
;
142 params
.parent
= parent
;
143 params
.accept_events
= false;
144 set_focus_on_creation(false);
146 SetVisibilityChangedAnimationsEnabled(false);
147 GetNativeWindow()->SetProperty(kStayInSameRootWindowKey
, true);
148 opaque_background_
.SetColor(SK_ColorBLACK
);
149 opaque_background_
.SetBounds(gfx::Rect(GetWindowBoundsInScreen().size()));
150 opaque_background_
.SetOpacity(0.0f
);
151 GetNativeWindow()->layer()->Add(&opaque_background_
);
154 ui::ResourceBundle
& rb
= ui::ResourceBundle::GetSharedInstance();
155 gfx::ImageSkia shelf_background
=
156 *rb
.GetImageSkiaNamed(IDR_ASH_SHELF_BACKGROUND
);
157 shelf_background_left_
= gfx::ImageSkiaOperations::CreateRotatedImage(
158 shelf_background
, SkBitmapOperations::ROTATION_90_CW
);
159 shelf_background_right_
= gfx::ImageSkiaOperations::CreateRotatedImage(
160 shelf_background
, SkBitmapOperations::ROTATION_270_CW
);
163 // Transitions to |visible_background_type_| if the widget is visible and to
164 // SHELF_BACKGROUND_DEFAULT if it is not.
165 void UpdateBackground() {
166 ShelfBackgroundType background_type
= IsVisible() ?
167 visible_background_type_
: SHELF_BACKGROUND_DEFAULT
;
168 BackgroundAnimatorChangeType change_type
= IsVisible() ?
169 visible_background_change_type_
: BACKGROUND_CHANGE_IMMEDIATE
;
171 float target_opacity
=
172 (background_type
== SHELF_BACKGROUND_MAXIMIZED
) ? 1.0f
: 0.0f
;
173 scoped_ptr
<ui::ScopedLayerAnimationSettings
> opaque_background_animation
;
174 if (change_type
!= BACKGROUND_CHANGE_IMMEDIATE
) {
175 opaque_background_animation
.reset(new ui::ScopedLayerAnimationSettings(
176 opaque_background_
.GetAnimator()));
177 opaque_background_animation
->SetTransitionDuration(
178 base::TimeDelta::FromMilliseconds(kTimeToSwitchBackgroundMs
));
180 opaque_background_
.SetOpacity(target_opacity
);
182 // TODO(varkha): use ui::Layer on both opaque_background and normal
183 // background retire background_animator_ at all. It would be simpler.
184 // See also ShelfWidget::SetPaintsBackground.
185 background_animator_
.SetPaintsBackground(
186 background_type
!= SHELF_BACKGROUND_DEFAULT
,
188 SchedulePaintInRect(gfx::Rect(GetWindowBoundsInScreen().size()));
191 DockedAlignment alignment_
;
193 // The animator for the background transitions.
194 BackgroundAnimator background_animator_
;
196 // The alpha to use for drawing image assets covering the docked background.
199 // Solid black background that can be made fully opaque.
200 ui::Layer opaque_background_
;
202 // Backgrounds created from shelf background by 90 or 270 degree rotation.
203 gfx::ImageSkia shelf_background_left_
;
204 gfx::ImageSkia shelf_background_right_
;
206 // The background type to use when the widget is visible. When not visible,
207 // the widget uses SHELF_BACKGROUND_DEFAULT.
208 ShelfBackgroundType visible_background_type_
;
210 // Whether the widget should animate to |visible_background_type_|.
211 BackgroundAnimatorChangeType visible_background_change_type_
;
213 DISALLOW_COPY_AND_ASSIGN(DockedBackgroundWidget
);
218 // Returns true if a window is a popup or a transient child.
219 bool IsPopupOrTransient(const aura::Window
* window
) {
220 return (window
->type() == ui::wm::WINDOW_TYPE_POPUP
||
221 ::wm::GetTransientParent(window
));
224 // Certain windows (minimized, hidden or popups) do not matter to docking.
225 bool IsUsedByLayout(const aura::Window
* window
) {
226 return (window
->IsVisible() &&
227 !wm::GetWindowState(window
)->IsMinimized() &&
228 !IsPopupOrTransient(window
));
231 void UndockWindow(aura::Window
* window
) {
232 gfx::Rect previous_bounds
= window
->bounds();
233 aura::Window
* old_parent
= window
->parent();
234 aura::client::ParentWindowWithContext(window
, window
, gfx::Rect());
235 if (window
->parent() != old_parent
)
236 wm::ReparentTransientChildrenOfChild(window
, old_parent
, window
->parent());
237 // Start maximize or fullscreen (affecting packaged apps) animation from
238 // previous window bounds.
239 window
->layer()->SetBounds(previous_bounds
);
242 // Returns width that is as close as possible to |target_width| while being
243 // consistent with docked min and max restrictions and respects the |window|'s
244 // minimum and maximum size.
245 int GetWindowWidthCloseTo(const aura::Window
* window
, int target_width
) {
246 if (!wm::GetWindowState(window
)->CanResize()) {
247 DCHECK_LE(window
->bounds().width(),
248 DockedWindowLayoutManager::kMaxDockWidth
);
249 return window
->bounds().width();
251 int width
= std::max(DockedWindowLayoutManager::kMinDockWidth
,
252 std::min(target_width
,
253 DockedWindowLayoutManager::kMaxDockWidth
));
254 if (window
->delegate()) {
255 if (window
->delegate()->GetMinimumSize().width() != 0)
256 width
= std::max(width
, window
->delegate()->GetMinimumSize().width());
257 if (window
->delegate()->GetMaximumSize().width() != 0)
258 width
= std::min(width
, window
->delegate()->GetMaximumSize().width());
260 DCHECK_LE(width
, DockedWindowLayoutManager::kMaxDockWidth
);
264 // Returns height that is as close as possible to |target_height| while
265 // respecting the |window|'s minimum and maximum size.
266 int GetWindowHeightCloseTo(const aura::Window
* window
, int target_height
) {
267 if (!wm::GetWindowState(window
)->CanResize())
268 return window
->bounds().height();
269 int minimum_height
= kMinimumHeight
;
270 int maximum_height
= 0;
271 const aura::WindowDelegate
* delegate(window
->delegate());
273 if (delegate
->GetMinimumSize().height() != 0) {
274 minimum_height
= std::max(kMinimumHeight
,
275 delegate
->GetMinimumSize().height());
277 if (delegate
->GetMaximumSize().height() != 0)
278 maximum_height
= delegate
->GetMaximumSize().height();
281 target_height
= std::max(target_height
, minimum_height
);
283 target_height
= std::min(target_height
, maximum_height
);
284 return target_height
;
287 // A functor used to sort the windows in order of their minimum height.
288 struct CompareMinimumHeight
{
289 bool operator()(WindowWithHeight win1
, WindowWithHeight win2
) {
290 return GetWindowHeightCloseTo(win1
.window(), 0) <
291 GetWindowHeightCloseTo(win2
.window(), 0);
295 // A functor used to sort the windows in order of their center Y position.
296 // |delta| is a pre-calculated distance from the bottom of one window to the top
297 // of the next. Its value can be positive (gap) or negative (overlap).
298 // Half of |delta| is used as a transition point at which windows could ideally
300 struct CompareWindowPos
{
301 CompareWindowPos(aura::Window
* dragged_window
,
302 aura::Window
* docked_container
,
304 : dragged_window_(dragged_window
),
305 docked_container_(docked_container
),
308 bool operator()(WindowWithHeight window_with_height1
,
309 WindowWithHeight window_with_height2
) {
310 // Use target coordinates since animations may be active when windows are
312 aura::Window
* win1(window_with_height1
.window());
313 aura::Window
* win2(window_with_height2
.window());
314 gfx::Rect win1_bounds
= ScreenUtil::ConvertRectToScreen(
315 docked_container_
, win1
->GetTargetBounds());
316 gfx::Rect win2_bounds
= ScreenUtil::ConvertRectToScreen(
317 docked_container_
, win2
->GetTargetBounds());
318 win1_bounds
.set_height(window_with_height1
.height_
);
319 win2_bounds
.set_height(window_with_height2
.height_
);
320 // If one of the windows is the |dragged_window_| attempt to make an
321 // earlier swap between the windows than just based on their centers.
322 // This is possible if the dragged window is at least as tall as the other
324 if (win1
== dragged_window_
)
325 return compare_two_windows(win1_bounds
, win2_bounds
);
326 if (win2
== dragged_window_
)
327 return !compare_two_windows(win2_bounds
, win1_bounds
);
328 // Otherwise just compare the centers.
329 return win1_bounds
.CenterPoint().y() < win2_bounds
.CenterPoint().y();
332 // Based on center point tries to deduce where the drag is coming from.
333 // When dragging from below up the transition point is lower.
334 // When dragging from above down the transition point is higher.
335 bool compare_bounds(const gfx::Rect dragged
, const gfx::Rect other
) {
336 if (dragged
.CenterPoint().y() < other
.CenterPoint().y())
337 return dragged
.CenterPoint().y() < other
.y() - delta_
;
338 return dragged
.CenterPoint().y() < other
.bottom() + delta_
;
341 // Performs comparison both ways and selects stable result.
342 bool compare_two_windows(const gfx::Rect bounds1
, const gfx::Rect bounds2
) {
343 // Try comparing windows in both possible orders and see if the comparison
345 bool result1
= compare_bounds(bounds1
, bounds2
);
346 bool result2
= compare_bounds(bounds2
, bounds1
);
347 if (result1
!= result2
)
350 // Otherwise it is not possible to be sure that the windows will not bounce.
351 // In this case just compare the centers.
352 return bounds1
.CenterPoint().y() < bounds2
.CenterPoint().y();
356 aura::Window
* dragged_window_
;
357 aura::Window
* docked_container_
;
363 ////////////////////////////////////////////////////////////////////////////////
364 // A class that observes shelf for bounds changes.
365 class DockedWindowLayoutManager::ShelfWindowObserver
: public WindowObserver
{
367 explicit ShelfWindowObserver(
368 DockedWindowLayoutManager
* docked_layout_manager
)
369 : docked_layout_manager_(docked_layout_manager
) {
370 DCHECK(docked_layout_manager_
->shelf()->shelf_widget());
371 docked_layout_manager_
->shelf()->shelf_widget()->GetNativeView()
375 ~ShelfWindowObserver() override
{
376 if (docked_layout_manager_
->shelf() &&
377 docked_layout_manager_
->shelf()->shelf_widget())
378 docked_layout_manager_
->shelf()->shelf_widget()->GetNativeView()
379 ->RemoveObserver(this);
382 // aura::WindowObserver:
383 void OnWindowBoundsChanged(aura::Window
* window
,
384 const gfx::Rect
& old_bounds
,
385 const gfx::Rect
& new_bounds
) override
{
386 shelf_bounds_in_screen_
= ScreenUtil::ConvertRectToScreen(
387 window
->parent(), new_bounds
);
388 docked_layout_manager_
->OnShelfBoundsChanged();
391 const gfx::Rect
& shelf_bounds_in_screen() const {
392 return shelf_bounds_in_screen_
;
396 DockedWindowLayoutManager
* docked_layout_manager_
;
397 gfx::Rect shelf_bounds_in_screen_
;
399 DISALLOW_COPY_AND_ASSIGN(ShelfWindowObserver
);
402 ////////////////////////////////////////////////////////////////////////////////
403 // DockedWindowLayoutManager public implementation:
404 DockedWindowLayoutManager::DockedWindowLayoutManager(
405 aura::Window
* dock_container
,
406 WorkspaceController
* workspace_controller
)
407 : SnapToPixelLayoutManager(dock_container
),
408 dock_container_(dock_container
),
410 dragged_window_(NULL
),
411 is_dragged_window_docked_(false),
412 is_dragged_from_dock_(false),
414 workspace_controller_(workspace_controller
),
415 in_fullscreen_(workspace_controller_
->GetWindowState() ==
416 WORKSPACE_WINDOW_STATE_FULL_SCREEN
),
418 alignment_(DOCKED_ALIGNMENT_NONE
),
419 preferred_alignment_(DOCKED_ALIGNMENT_NONE
),
420 event_source_(DOCKED_ACTION_SOURCE_UNKNOWN
),
421 last_active_window_(NULL
),
422 last_action_time_(base::Time::Now()),
423 background_widget_(new DockedBackgroundWidget(dock_container_
)) {
424 DCHECK(dock_container
);
425 aura::client::GetActivationClient(Shell::GetPrimaryRootWindow())->
427 Shell::GetInstance()->AddShellObserver(this);
430 DockedWindowLayoutManager::~DockedWindowLayoutManager() {
434 void DockedWindowLayoutManager::Shutdown() {
435 if (shelf_
&& shelf_
->shelf_widget()) {
436 ShelfLayoutManager
* shelf_layout_manager
= ShelfLayoutManager::ForShelf(
437 shelf_
->shelf_widget()->GetNativeWindow());
438 shelf_layout_manager
->RemoveObserver(this);
439 shelf_observer_
.reset();
442 for (size_t i
= 0; i
< dock_container_
->children().size(); ++i
) {
443 aura::Window
* child
= dock_container_
->children()[i
];
444 child
->RemoveObserver(this);
445 wm::GetWindowState(child
)->RemoveObserver(this);
447 aura::client::GetActivationClient(Shell::GetPrimaryRootWindow())->
448 RemoveObserver(this);
449 Shell::GetInstance()->RemoveShellObserver(this);
452 void DockedWindowLayoutManager::AddObserver(
453 DockedWindowLayoutManagerObserver
* observer
) {
454 observer_list_
.AddObserver(observer
);
457 void DockedWindowLayoutManager::RemoveObserver(
458 DockedWindowLayoutManagerObserver
* observer
) {
459 observer_list_
.RemoveObserver(observer
);
462 void DockedWindowLayoutManager::StartDragging(aura::Window
* window
) {
463 DCHECK(!dragged_window_
);
464 dragged_window_
= window
;
465 DCHECK(!IsPopupOrTransient(window
));
466 // Start observing a window unless it is docked container's child in which
467 // case it is already observed.
468 wm::WindowState
* dragged_state
= wm::GetWindowState(dragged_window_
);
469 if (dragged_window_
->parent() != dock_container_
) {
470 dragged_window_
->AddObserver(this);
471 dragged_state
->AddObserver(this);
472 } else if (!IsAnyWindowDocked() &&
473 dragged_state
->drag_details() &&
474 !(dragged_state
->drag_details()->bounds_change
&
475 WindowResizer::kBoundsChange_Resizes
)) {
476 // If there are no other docked windows clear alignment when a docked window
477 // is moved (but not when it is resized or the window could get undocked
478 // when resized away from the edge while docked).
479 alignment_
= DOCKED_ALIGNMENT_NONE
;
481 is_dragged_from_dock_
= window
->parent() == dock_container_
;
482 DCHECK(!is_dragged_window_docked_
);
484 // Resize all windows that are flush with the dock edge together if one of
485 // them gets resized.
486 if (dragged_window_
->bounds().width() == docked_width_
&&
487 (dragged_state
->drag_details()->bounds_change
&
488 WindowResizer::kBoundsChange_Resizes
) &&
489 (dragged_state
->drag_details()->size_change_direction
&
490 WindowResizer::kBoundsChangeDirection_Horizontal
)) {
491 for (size_t i
= 0; i
< dock_container_
->children().size(); ++i
) {
492 aura::Window
* window1(dock_container_
->children()[i
]);
493 if (IsUsedByLayout(window1
) &&
494 window1
!= dragged_window_
&&
495 window1
->bounds().width() == docked_width_
) {
496 wm::GetWindowState(window1
)->set_bounds_changed_by_user(false);
502 void DockedWindowLayoutManager::DockDraggedWindow(aura::Window
* window
) {
503 DCHECK(!IsPopupOrTransient(window
));
504 OnDraggedWindowDocked(window
);
508 void DockedWindowLayoutManager::UndockDraggedWindow() {
509 DCHECK(!IsPopupOrTransient(dragged_window_
));
510 OnDraggedWindowUndocked();
512 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
513 is_dragged_from_dock_
= false;
516 void DockedWindowLayoutManager::FinishDragging(DockedAction action
,
517 DockedActionSource source
) {
518 DCHECK(dragged_window_
);
519 DCHECK(!IsPopupOrTransient(dragged_window_
));
520 if (is_dragged_window_docked_
)
521 OnDraggedWindowUndocked();
522 DCHECK (!is_dragged_window_docked_
);
523 // Stop observing a window unless it is docked container's child in which
524 // case it needs to keep being observed after the drag completes.
525 if (dragged_window_
->parent() != dock_container_
) {
526 dragged_window_
->RemoveObserver(this);
527 wm::GetWindowState(dragged_window_
)->RemoveObserver(this);
528 if (last_active_window_
== dragged_window_
)
529 last_active_window_
= NULL
;
531 // If this is the first window that got docked by a move update alignment.
532 if (alignment_
== DOCKED_ALIGNMENT_NONE
)
533 alignment_
= GetEdgeNearestWindow(dragged_window_
);
534 // A window is no longer dragged and is a child.
535 // When a window becomes a child at drag start this is
536 // the only opportunity we will have to enforce a window
537 // count limit so do it here.
538 MaybeMinimizeChildrenExcept(dragged_window_
);
540 dragged_window_
= NULL
;
541 dragged_bounds_
= gfx::Rect();
543 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
544 RecordUmaAction(action
, source
);
547 void DockedWindowLayoutManager::SetShelf(Shelf
* shelf
) {
550 if (shelf_
->shelf_widget()) {
551 ShelfLayoutManager
* shelf_layout_manager
= ShelfLayoutManager::ForShelf(
552 shelf_
->shelf_widget()->GetNativeWindow());
553 shelf_layout_manager
->AddObserver(this);
554 shelf_observer_
.reset(new ShelfWindowObserver(this));
558 DockedAlignment
DockedWindowLayoutManager::GetAlignmentOfWindow(
559 const aura::Window
* window
) const {
560 const gfx::Rect
& bounds(window
->GetBoundsInScreen());
562 // Test overlap with an existing docked area first.
563 if (docked_bounds_
.Intersects(bounds
) &&
564 alignment_
!= DOCKED_ALIGNMENT_NONE
) {
565 // A window is being added to other docked windows (on the same side).
569 const gfx::Rect container_bounds
= dock_container_
->GetBoundsInScreen();
570 if (bounds
.x() <= container_bounds
.x() &&
571 bounds
.right() > container_bounds
.x()) {
572 return DOCKED_ALIGNMENT_LEFT
;
573 } else if (bounds
.x() < container_bounds
.right() &&
574 bounds
.right() >= container_bounds
.right()) {
575 return DOCKED_ALIGNMENT_RIGHT
;
577 return DOCKED_ALIGNMENT_NONE
;
580 DockedAlignment
DockedWindowLayoutManager::CalculateAlignment() const {
581 return CalculateAlignmentExcept(dragged_window_
);
584 DockedAlignment
DockedWindowLayoutManager::CalculateAlignmentExcept(
585 const aura::Window
* window
) const {
586 // Find a child that is not the window being queried and is not a popup.
587 // If such exists the current alignment is returned - even if some of the
588 // children are hidden or minimized (so they can be restored without losing
589 // the docked state).
590 for (size_t i
= 0; i
< dock_container_
->children().size(); ++i
) {
591 aura::Window
* child(dock_container_
->children()[i
]);
592 if (window
!= child
&& !IsPopupOrTransient(child
))
595 // No docked windows remain other than possibly the window being queried.
596 // Return |NONE| to indicate that windows may get docked on either side.
597 return DOCKED_ALIGNMENT_NONE
;
600 bool DockedWindowLayoutManager::CanDockWindow(
601 aura::Window
* window
,
602 DockedAlignment desired_alignment
) {
603 // Don't allow interactive docking of windows with transient parents such as
604 // modal browser dialogs. Prevent docking of panels attached to shelf during
606 wm::WindowState
* window_state
= wm::GetWindowState(window
);
607 bool should_attach_to_shelf
= window_state
->drag_details() &&
608 window_state
->drag_details()->should_attach_to_shelf
;
609 if (IsPopupOrTransient(window
) || should_attach_to_shelf
)
611 // If a window is wide and cannot be resized down to maximum width allowed
612 // then it cannot be docked.
613 // TODO(varkha). Prevent windows from changing size programmatically while
614 // they are docked. The size will take effect only once a window is undocked.
615 // See http://crbug.com/307792.
616 if (window
->bounds().width() > kMaxDockWidth
&&
617 (!window_state
->CanResize() ||
618 (window
->delegate() &&
619 window
->delegate()->GetMinimumSize().width() != 0 &&
620 window
->delegate()->GetMinimumSize().width() > kMaxDockWidth
))) {
623 // If a window is tall and cannot be resized down to maximum height allowed
624 // then it cannot be docked.
625 const gfx::Rect work_area
=
626 Shell::GetScreen()->GetDisplayNearestWindow(dock_container_
).work_area();
627 if (GetWindowHeightCloseTo(window
, work_area
.height()) > work_area
.height())
629 // Cannot dock on the other size from an existing dock.
630 const DockedAlignment alignment
= CalculateAlignmentExcept(window
);
631 if (desired_alignment
!= DOCKED_ALIGNMENT_NONE
&&
632 alignment
!= DOCKED_ALIGNMENT_NONE
&&
633 alignment
!= desired_alignment
) {
636 // Do not allow docking on the same side as shelf.
637 return IsDockedAlignmentValid(desired_alignment
);
640 bool DockedWindowLayoutManager::IsDockedAlignmentValid(
641 DockedAlignment alignment
) const {
642 ShelfAlignment shelf_alignment
= shelf_
? shelf_
->alignment() :
643 SHELF_ALIGNMENT_BOTTOM
;
644 if ((alignment
== DOCKED_ALIGNMENT_LEFT
&&
645 shelf_alignment
== SHELF_ALIGNMENT_LEFT
) ||
646 (alignment
== DOCKED_ALIGNMENT_RIGHT
&&
647 shelf_alignment
== SHELF_ALIGNMENT_RIGHT
)) {
653 void DockedWindowLayoutManager::MaybeSetDesiredDockedAlignment(
654 DockedAlignment alignment
) {
655 // If the requested alignment is |NONE| or there are no
656 // docked windows return early as we can't change whether there is a
657 // dock or not. If the requested alignment is the same as the current
658 // alignment return early as an optimization.
659 if (alignment
== DOCKED_ALIGNMENT_NONE
||
660 alignment_
== DOCKED_ALIGNMENT_NONE
||
661 alignment_
== alignment
||
662 !IsDockedAlignmentValid(alignment
)) {
665 alignment_
= alignment
;
668 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
671 void DockedWindowLayoutManager::OnShelfBoundsChanged() {
673 UpdateDockBounds(DockedWindowLayoutManagerObserver::DISPLAY_INSETS_CHANGED
);
676 ////////////////////////////////////////////////////////////////////////////////
677 // DockedWindowLayoutManager, aura::LayoutManager implementation:
678 void DockedWindowLayoutManager::OnWindowResized() {
679 MaybeMinimizeChildrenExcept(dragged_window_
);
681 // When screen resizes update the insets even when dock width or alignment
683 UpdateDockBounds(DockedWindowLayoutManagerObserver::DISPLAY_RESIZED
);
686 void DockedWindowLayoutManager::OnWindowAddedToLayout(aura::Window
* child
) {
687 if (IsPopupOrTransient(child
))
689 // Dragged windows are already observed by StartDragging and do not change
690 // docked alignment during the drag.
691 if (child
== dragged_window_
)
693 // If this is the first window getting docked - update alignment.
694 // A window can be added without proper bounds when window is moved to another
695 // display via API or due to display configuration change, so the alignment
696 // is set based on which edge is closer in the new display.
697 if (alignment_
== DOCKED_ALIGNMENT_NONE
) {
698 alignment_
= preferred_alignment_
!= DOCKED_ALIGNMENT_NONE
?
699 preferred_alignment_
: GetEdgeNearestWindow(child
);
701 MaybeMinimizeChildrenExcept(child
);
702 child
->AddObserver(this);
703 wm::GetWindowState(child
)->AddObserver(this);
705 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
707 // Only keyboard-initiated actions are recorded here. Dragging cases
708 // are handled in FinishDragging.
709 if (event_source_
!= DOCKED_ACTION_SOURCE_UNKNOWN
)
710 RecordUmaAction(DOCKED_ACTION_DOCK
, event_source_
);
713 void DockedWindowLayoutManager::OnWindowRemovedFromLayout(aura::Window
* child
) {
714 if (IsPopupOrTransient(child
))
716 // Dragged windows are stopped being observed by FinishDragging and do not
717 // change alignment during the drag. They also cannot be set to be the
718 // |last_active_window_|.
719 if (child
== dragged_window_
)
721 // If this is the last window, set alignment and maximize the workspace.
722 if (!IsAnyWindowDocked()) {
723 alignment_
= DOCKED_ALIGNMENT_NONE
;
724 UpdateDockedWidth(0);
726 if (last_active_window_
== child
)
727 last_active_window_
= NULL
;
728 child
->RemoveObserver(this);
729 wm::GetWindowState(child
)->RemoveObserver(this);
731 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
734 void DockedWindowLayoutManager::OnChildWindowVisibilityChanged(
737 if (IsPopupOrTransient(child
))
740 wm::WindowState
* window_state
= wm::GetWindowState(child
);
741 if (visible
&& window_state
->IsMinimized())
742 window_state
->Restore();
744 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
747 void DockedWindowLayoutManager::SetChildBounds(
749 const gfx::Rect
& requested_bounds
) {
750 // The minimum constraints have to be applied first by the layout manager.
751 gfx::Rect
actual_new_bounds(requested_bounds
);
752 if (child
->delegate()) {
753 const gfx::Size
& min_size
= child
->delegate()->GetMinimumSize();
754 actual_new_bounds
.set_width(
755 std::max(min_size
.width(), actual_new_bounds
.width()));
756 actual_new_bounds
.set_height(
757 std::max(min_size
.height(), actual_new_bounds
.height()));
759 SnapToPixelLayoutManager::SetChildBounds(child
, actual_new_bounds
);
760 if (IsPopupOrTransient(child
))
762 // Whenever one of our windows is moved or resized enforce layout.
763 ShelfLayoutManager
* shelf_layout
=
764 ShelfLayoutManager::ForShelf(dock_container_
);
766 shelf_layout
->UpdateVisibilityState();
769 ////////////////////////////////////////////////////////////////////////////////
770 // DockedWindowLayoutManager, ash::ShellObserver implementation:
772 void DockedWindowLayoutManager::OnDisplayWorkAreaInsetsChanged() {
774 UpdateDockBounds(DockedWindowLayoutManagerObserver::DISPLAY_INSETS_CHANGED
);
775 MaybeMinimizeChildrenExcept(dragged_window_
);
778 void DockedWindowLayoutManager::OnFullscreenStateChanged(
779 bool is_fullscreen
, aura::Window
* root_window
) {
780 if (dock_container_
->GetRootWindow() != root_window
)
782 // Entering fullscreen mode (including immersive) hides docked windows.
783 in_fullscreen_
= workspace_controller_
->GetWindowState() ==
784 WORKSPACE_WINDOW_STATE_FULL_SCREEN
;
786 // prevent Relayout from getting called multiple times during this
787 base::AutoReset
<bool> auto_reset_in_layout(&in_layout_
, true);
788 // Use a copy of children array because a call to MinimizeDockedWindow or
789 // RestoreDockedWindow can change order.
790 aura::Window::Windows
children(dock_container_
->children());
791 for (aura::Window::Windows::const_iterator iter
= children
.begin();
792 iter
!= children
.end(); ++iter
) {
793 aura::Window
* window(*iter
);
794 if (IsPopupOrTransient(window
))
796 wm::WindowState
* window_state
= wm::GetWindowState(window
);
797 if (in_fullscreen_
) {
798 if (window
->IsVisible())
799 MinimizeDockedWindow(window_state
);
801 if (!window_state
->IsMinimized())
802 RestoreDockedWindow(window_state
);
807 UpdateDockBounds(DockedWindowLayoutManagerObserver::CHILD_CHANGED
);
810 void DockedWindowLayoutManager::OnShelfAlignmentChanged(
811 aura::Window
* root_window
) {
812 if (dock_container_
->GetRootWindow() != root_window
)
815 if (!shelf_
|| !shelf_
->shelf_widget())
818 if (alignment_
== DOCKED_ALIGNMENT_NONE
)
821 // Do not allow shelf and dock on the same side. Switch side that
822 // the dock is attached to and move all dock windows to that new side.
823 ShelfAlignment shelf_alignment
= shelf_
->shelf_widget()->GetAlignment();
824 if (alignment_
== DOCKED_ALIGNMENT_LEFT
&&
825 shelf_alignment
== SHELF_ALIGNMENT_LEFT
) {
826 alignment_
= DOCKED_ALIGNMENT_RIGHT
;
827 } else if (alignment_
== DOCKED_ALIGNMENT_RIGHT
&&
828 shelf_alignment
== SHELF_ALIGNMENT_RIGHT
) {
829 alignment_
= DOCKED_ALIGNMENT_LEFT
;
832 UpdateDockBounds(DockedWindowLayoutManagerObserver::SHELF_ALIGNMENT_CHANGED
);
835 /////////////////////////////////////////////////////////////////////////////
836 // DockedWindowLayoutManager, ShelfLayoutManagerObserver implementation:
837 void DockedWindowLayoutManager::OnBackgroundUpdated(
838 ShelfBackgroundType background_type
,
839 BackgroundAnimatorChangeType change_type
) {
840 background_widget_
->SetBackgroundType(background_type
, change_type
);
843 /////////////////////////////////////////////////////////////////////////////
844 // DockedWindowLayoutManager, WindowStateObserver implementation:
846 void DockedWindowLayoutManager::OnPreWindowStateTypeChange(
847 wm::WindowState
* window_state
,
848 wm::WindowStateType old_type
) {
849 aura::Window
* window
= window_state
->window();
850 if (IsPopupOrTransient(window
))
852 // The window property will still be set, but no actual change will occur
853 // until OnFullscreenStateChange is called when exiting fullscreen.
856 if (!window_state
->IsDocked()) {
857 if (window
!= dragged_window_
) {
858 UndockWindow(window
);
859 if (window_state
->IsMaximizedOrFullscreen())
860 RecordUmaAction(DOCKED_ACTION_MAXIMIZE
, event_source_
);
862 RecordUmaAction(DOCKED_ACTION_UNDOCK
, event_source_
);
864 } else if (window_state
->IsMinimized()) {
865 MinimizeDockedWindow(window_state
);
866 } else if (old_type
== wm::WINDOW_STATE_TYPE_DOCKED_MINIMIZED
) {
867 RestoreDockedWindow(window_state
);
868 } else if (old_type
== wm::WINDOW_STATE_TYPE_MINIMIZED
) {
869 NOTREACHED() << "Minimized window in docked layout manager";
873 /////////////////////////////////////////////////////////////////////////////
874 // DockedWindowLayoutManager, WindowObserver implementation:
876 void DockedWindowLayoutManager::OnWindowBoundsChanged(
877 aura::Window
* window
,
878 const gfx::Rect
& old_bounds
,
879 const gfx::Rect
& new_bounds
) {
880 // Only relayout if the dragged window would get docked.
881 if (window
== dragged_window_
&& is_dragged_window_docked_
)
885 void DockedWindowLayoutManager::OnWindowVisibilityChanging(
886 aura::Window
* window
, bool visible
) {
887 if (IsPopupOrTransient(window
))
889 int animation_type
= ::wm::WINDOW_VISIBILITY_ANIMATION_TYPE_DEFAULT
;
891 animation_type
= ::wm::WINDOW_VISIBILITY_ANIMATION_TYPE_DROP
;
892 ::wm::SetWindowVisibilityAnimationDuration(
893 window
, base::TimeDelta::FromMilliseconds(kFadeDurationMs
));
894 } else if (wm::GetWindowState(window
)->IsMinimized()) {
895 animation_type
= WINDOW_VISIBILITY_ANIMATION_TYPE_MINIMIZE
;
897 ::wm::SetWindowVisibilityAnimationType(window
, animation_type
);
900 void DockedWindowLayoutManager::OnWindowDestroying(aura::Window
* window
) {
901 if (dragged_window_
== window
) {
902 FinishDragging(DOCKED_ACTION_NONE
, DOCKED_ACTION_SOURCE_UNKNOWN
);
903 DCHECK(!dragged_window_
);
904 DCHECK(!is_dragged_window_docked_
);
906 if (window
== last_active_window_
)
907 last_active_window_
= NULL
;
908 RecordUmaAction(DOCKED_ACTION_CLOSE
, event_source_
);
912 ////////////////////////////////////////////////////////////////////////////////
913 // DockedWindowLayoutManager, aura::client::ActivationChangeObserver
916 void DockedWindowLayoutManager::OnWindowActivated(aura::Window
* gained_active
,
917 aura::Window
* lost_active
) {
918 if (gained_active
&& IsPopupOrTransient(gained_active
))
920 // Ignore if the window that is not managed by this was activated.
921 aura::Window
* ancestor
= NULL
;
922 for (aura::Window
* parent
= gained_active
;
923 parent
; parent
= parent
->parent()) {
924 if (parent
->parent() == dock_container_
) {
930 UpdateStacking(ancestor
);
933 ////////////////////////////////////////////////////////////////////////////////
934 // DockedWindowLayoutManager private implementation:
936 void DockedWindowLayoutManager::MaybeMinimizeChildrenExcept(
937 aura::Window
* child
) {
938 // Minimize any windows that don't fit without overlap.
939 const gfx::Rect work_area
=
940 Shell::GetScreen()->GetDisplayNearestWindow(dock_container_
).work_area();
941 int available_room
= work_area
.height();
942 bool gap_needed
= !!child
;
944 available_room
-= GetWindowHeightCloseTo(child
, 0);
945 // Use a copy of children array because a call to Minimize can change order.
946 aura::Window::Windows
children(dock_container_
->children());
947 aura::Window::Windows::const_reverse_iterator iter
= children
.rbegin();
948 while (iter
!= children
.rend()) {
949 aura::Window
* window(*iter
++);
950 if (window
== child
|| !IsUsedByLayout(window
))
952 int room_needed
= GetWindowHeightCloseTo(window
, 0) +
953 (gap_needed
? kMinDockGap
: 0);
955 if (available_room
> room_needed
) {
956 available_room
-= room_needed
;
958 // Slow down minimizing animations. Lock duration so that it is not
959 // overridden by other ScopedLayerAnimationSettings down the stack.
960 ui::ScopedLayerAnimationSettings
settings(window
->layer()->GetAnimator());
961 settings
.SetTransitionDuration(
962 base::TimeDelta::FromMilliseconds(kMinimizeDurationMs
));
963 settings
.LockTransitionDuration();
964 wm::GetWindowState(window
)->Minimize();
969 void DockedWindowLayoutManager::MinimizeDockedWindow(
970 wm::WindowState
* window_state
) {
971 DCHECK(!IsPopupOrTransient(window_state
->window()));
972 window_state
->window()->Hide();
973 if (window_state
->IsActive())
974 window_state
->Deactivate();
975 RecordUmaAction(DOCKED_ACTION_MINIMIZE
, event_source_
);
978 void DockedWindowLayoutManager::RestoreDockedWindow(
979 wm::WindowState
* window_state
) {
980 aura::Window
* window
= window_state
->window();
981 DCHECK(!IsPopupOrTransient(window
));
982 // Always place restored window at the bottom shuffling the other windows up.
983 // TODO(varkha): add a separate container for docked windows to keep track
985 gfx::Display display
= Shell::GetScreen()->GetDisplayNearestWindow(
987 const gfx::Rect work_area
= display
.work_area();
989 // Evict the window if it can no longer be docked because of its height.
990 if (!CanDockWindow(window
, DOCKED_ALIGNMENT_NONE
)) {
991 window_state
->Restore();
992 RecordUmaAction(DOCKED_ACTION_EVICT
, event_source_
);
995 gfx::Rect
bounds(window
->bounds());
996 bounds
.set_y(work_area
.bottom());
997 window
->SetBounds(bounds
);
999 MaybeMinimizeChildrenExcept(window
);
1000 RecordUmaAction(DOCKED_ACTION_RESTORE
, event_source_
);
1003 void DockedWindowLayoutManager::RecordUmaAction(DockedAction action
,
1004 DockedActionSource source
) {
1005 if (action
== DOCKED_ACTION_NONE
)
1007 UMA_HISTOGRAM_ENUMERATION("Ash.Dock.Action", action
, DOCKED_ACTION_COUNT
);
1008 UMA_HISTOGRAM_ENUMERATION("Ash.Dock.ActionSource", source
,
1009 DOCKED_ACTION_SOURCE_COUNT
);
1010 base::Time time_now
= base::Time::Now();
1011 base::TimeDelta time_between_use
= time_now
- last_action_time_
;
1012 UMA_HISTOGRAM_CUSTOM_COUNTS("Ash.Dock.TimeBetweenUse",
1013 time_between_use
.InSeconds(),
1015 base::TimeDelta::FromHours(10).InSeconds(),
1017 last_action_time_
= time_now
;
1018 int docked_all_count
= 0;
1019 int docked_visible_count
= 0;
1020 int docked_panels_count
= 0;
1021 int large_windows_count
= 0;
1022 for (size_t i
= 0; i
< dock_container_
->children().size(); ++i
) {
1023 const aura::Window
* window(dock_container_
->children()[i
]);
1024 if (IsPopupOrTransient(window
))
1027 if (!IsUsedByLayout(window
))
1029 docked_visible_count
++;
1030 if (window
->type() == ui::wm::WINDOW_TYPE_PANEL
)
1031 docked_panels_count
++;
1032 const wm::WindowState
* window_state
= wm::GetWindowState(window
);
1033 if (window_state
->HasRestoreBounds()) {
1034 const gfx::Rect restore_bounds
= window_state
->GetRestoreBoundsInScreen();
1035 if (restore_bounds
.width() > kMaxDockWidth
)
1036 large_windows_count
++;
1039 UMA_HISTOGRAM_COUNTS_100("Ash.Dock.ItemsAll", docked_all_count
);
1040 UMA_HISTOGRAM_COUNTS_100("Ash.Dock.ItemsLarge", large_windows_count
);
1041 UMA_HISTOGRAM_COUNTS_100("Ash.Dock.ItemsPanels", docked_panels_count
);
1042 UMA_HISTOGRAM_COUNTS_100("Ash.Dock.ItemsVisible", docked_visible_count
);
1045 void DockedWindowLayoutManager::UpdateDockedWidth(int width
) {
1046 if (docked_width_
== width
)
1048 docked_width_
= width
;
1049 UMA_HISTOGRAM_COUNTS_10000("Ash.Dock.Width", docked_width_
);
1052 void DockedWindowLayoutManager::OnDraggedWindowDocked(aura::Window
* window
) {
1053 DCHECK(!is_dragged_window_docked_
);
1054 is_dragged_window_docked_
= true;
1057 void DockedWindowLayoutManager::OnDraggedWindowUndocked() {
1058 DCHECK (is_dragged_window_docked_
);
1059 is_dragged_window_docked_
= false;
1062 bool DockedWindowLayoutManager::IsAnyWindowDocked() {
1063 return CalculateAlignment() != DOCKED_ALIGNMENT_NONE
;
1066 DockedAlignment
DockedWindowLayoutManager::GetEdgeNearestWindow(
1067 const aura::Window
* window
) const {
1068 const gfx::Rect
& bounds(window
->GetBoundsInScreen());
1069 const gfx::Rect container_bounds
= dock_container_
->GetBoundsInScreen();
1070 return (abs(bounds
.x() - container_bounds
.x()) <
1071 abs(bounds
.right() - container_bounds
.right())) ?
1072 DOCKED_ALIGNMENT_LEFT
: DOCKED_ALIGNMENT_RIGHT
;
1075 void DockedWindowLayoutManager::Relayout() {
1078 if (alignment_
== DOCKED_ALIGNMENT_NONE
&& !is_dragged_window_docked_
)
1080 base::AutoReset
<bool> auto_reset_in_layout(&in_layout_
, true);
1082 gfx::Rect dock_bounds
= dock_container_
->GetBoundsInScreen();
1083 aura::Window
* active_window
= NULL
;
1084 std::vector
<WindowWithHeight
> visible_windows
;
1085 for (size_t i
= 0; i
< dock_container_
->children().size(); ++i
) {
1086 aura::Window
* window(dock_container_
->children()[i
]);
1088 if (!IsUsedByLayout(window
) || window
== dragged_window_
)
1091 // If the shelf is currently hidden (full-screen mode), hide window until
1092 // full-screen mode is exited.
1093 if (in_fullscreen_
) {
1094 // The call to Hide does not set the minimize property, so the window will
1095 // be restored when the shelf becomes visible again.
1099 if (window
->HasFocus() ||
1101 aura::client::GetFocusClient(window
)->GetFocusedWindow())) {
1102 DCHECK(!active_window
);
1103 active_window
= window
;
1105 visible_windows
.push_back(WindowWithHeight(window
));
1107 // Consider docked dragged_window_ when fanning out other child windows.
1108 if (is_dragged_window_docked_
) {
1109 visible_windows
.push_back(WindowWithHeight(dragged_window_
));
1110 DCHECK(!active_window
);
1111 active_window
= dragged_window_
;
1114 // Position docked windows as well as the window being dragged.
1115 gfx::Rect work_area
=
1116 Shell::GetScreen()->GetDisplayNearestWindow(dock_container_
).work_area();
1117 if (shelf_observer_
)
1118 work_area
.Subtract(shelf_observer_
->shelf_bounds_in_screen());
1119 int available_room
= CalculateWindowHeightsAndRemainingRoom(work_area
,
1121 FanOutChildren(work_area
,
1122 CalculateIdealWidth(visible_windows
),
1126 // After the first Relayout allow the windows to change their order easier
1127 // since we know they are docked.
1128 is_dragged_from_dock_
= true;
1129 UpdateStacking(active_window
);
1132 int DockedWindowLayoutManager::CalculateWindowHeightsAndRemainingRoom(
1133 const gfx::Rect work_area
,
1134 std::vector
<WindowWithHeight
>* visible_windows
) {
1135 int available_room
= work_area
.height();
1136 int remaining_windows
= visible_windows
->size();
1137 int gap_height
= remaining_windows
> 1 ? kMinDockGap
: 0;
1139 // Sort windows by their minimum heights and calculate target heights.
1140 std::sort(visible_windows
->begin(), visible_windows
->end(),
1141 CompareMinimumHeight());
1142 // Distribute the free space among the docked windows. Since the windows are
1143 // sorted (tall windows first) we can now assume that any window which
1144 // required more space than the current window will have already been
1145 // accounted for previously in this loop, so we can safely give that window
1146 // its proportional share of the remaining space.
1147 for (std::vector
<WindowWithHeight
>::reverse_iterator iter
=
1148 visible_windows
->rbegin();
1149 iter
!= visible_windows
->rend(); ++iter
) {
1150 iter
->height_
= GetWindowHeightCloseTo(
1152 (available_room
+ gap_height
) / remaining_windows
- gap_height
);
1153 available_room
-= (iter
->height_
+ gap_height
);
1154 remaining_windows
--;
1156 return available_room
+ gap_height
;
1159 int DockedWindowLayoutManager::CalculateIdealWidth(
1160 const std::vector
<WindowWithHeight
>& visible_windows
) {
1161 int smallest_max_width
= kMaxDockWidth
;
1162 int largest_min_width
= kMinDockWidth
;
1163 // Ideal width of the docked area is as close to kIdealWidth as possible
1164 // while still respecting the minimum and maximum width restrictions on the
1165 // individual docked windows as well as the width that was possibly set by a
1166 // user (which needs to be preserved when dragging and rearranging windows).
1167 for (std::vector
<WindowWithHeight
>::const_iterator iter
=
1168 visible_windows
.begin();
1169 iter
!= visible_windows
.end(); ++iter
) {
1170 const aura::Window
* window
= iter
->window();
1171 int min_window_width
= window
->bounds().width();
1172 int max_window_width
= min_window_width
;
1173 if (!wm::GetWindowState(window
)->bounds_changed_by_user()) {
1174 min_window_width
= GetWindowWidthCloseTo(window
, kMinDockWidth
);
1175 max_window_width
= GetWindowWidthCloseTo(window
, kMaxDockWidth
);
1177 largest_min_width
= std::max(largest_min_width
, min_window_width
);
1178 smallest_max_width
= std::min(smallest_max_width
, max_window_width
);
1180 int ideal_width
= std::max(largest_min_width
,
1181 std::min(smallest_max_width
, kIdealWidth
));
1182 // Restrict docked area width regardless of window restrictions.
1183 ideal_width
= std::max(std::min(ideal_width
, kMaxDockWidth
), kMinDockWidth
);
1187 void DockedWindowLayoutManager::FanOutChildren(
1188 const gfx::Rect
& work_area
,
1189 int ideal_docked_width
,
1191 std::vector
<WindowWithHeight
>* visible_windows
) {
1192 gfx::Rect dock_bounds
= dock_container_
->GetBoundsInScreen();
1194 // Calculate initial vertical offset and the gap or overlap between windows.
1195 const int num_windows
= visible_windows
->size();
1196 const float delta
= static_cast<float>(available_room
) /
1197 ((available_room
> 0 || num_windows
<= 1) ?
1198 num_windows
+ 1 : num_windows
- 1);
1199 float y_pos
= work_area
.y() + ((delta
> 0) ? delta
: 0);
1201 // Docked area is shown only if there is at least one non-dragged visible
1203 int new_width
= ideal_docked_width
;
1204 if (visible_windows
->empty() ||
1205 (visible_windows
->size() == 1 &&
1206 (*visible_windows
)[0].window() == dragged_window_
)) {
1209 UpdateDockedWidth(new_width
);
1210 // Sort windows by their center positions and fan out overlapping
1212 std::sort(visible_windows
->begin(), visible_windows
->end(),
1213 CompareWindowPos(is_dragged_from_dock_
? dragged_window_
: NULL
,
1216 for (std::vector
<WindowWithHeight
>::iterator iter
= visible_windows
->begin();
1217 iter
!= visible_windows
->end(); ++iter
) {
1218 aura::Window
* window
= iter
->window();
1219 gfx::Rect bounds
= ScreenUtil::ConvertRectToScreen(
1220 dock_container_
, window
->GetTargetBounds());
1221 // A window is extended or shrunk to be as close as possible to the ideal
1222 // docked area width. Windows that were resized by a user are kept at their
1224 // This also enforces the min / max restrictions on the docked area width.
1225 bounds
.set_width(GetWindowWidthCloseTo(
1227 wm::GetWindowState(window
)->bounds_changed_by_user() ?
1228 bounds
.width() : ideal_docked_width
));
1229 DCHECK_LE(bounds
.width(), ideal_docked_width
);
1231 DockedAlignment alignment
= alignment_
;
1232 if (alignment
== DOCKED_ALIGNMENT_NONE
&& window
== dragged_window_
)
1233 alignment
= GetEdgeNearestWindow(window
);
1235 // Fan out windows evenly distributing the overlap or remaining free space.
1236 bounds
.set_height(iter
->height_
);
1237 bounds
.set_y(std::max(work_area
.y(),
1238 std::min(work_area
.bottom() - bounds
.height(),
1239 static_cast<int>(y_pos
+ 0.5))));
1240 y_pos
+= bounds
.height() + delta
+ kMinDockGap
;
1242 // All docked windows other than the one currently dragged remain stuck
1243 // to the screen edge (flush with the edge or centered in the dock area).
1244 switch (alignment
) {
1245 case DOCKED_ALIGNMENT_LEFT
:
1246 bounds
.set_x(dock_bounds
.x() +
1247 (ideal_docked_width
- bounds
.width()) / 2);
1249 case DOCKED_ALIGNMENT_RIGHT
:
1250 bounds
.set_x(dock_bounds
.right() -
1251 (ideal_docked_width
+ bounds
.width()) / 2);
1253 case DOCKED_ALIGNMENT_NONE
:
1256 if (window
== dragged_window_
) {
1257 dragged_bounds_
= bounds
;
1260 // If the following asserts it is probably because not all the children
1261 // have been removed when dock was closed.
1262 DCHECK_NE(alignment_
, DOCKED_ALIGNMENT_NONE
);
1263 bounds
= ScreenUtil::ConvertRectFromScreen(dock_container_
, bounds
);
1264 if (bounds
!= window
->GetTargetBounds()) {
1265 ui::Layer
* layer
= window
->layer();
1266 ui::ScopedLayerAnimationSettings
slide_settings(layer
->GetAnimator());
1267 slide_settings
.SetPreemptionStrategy(
1268 ui::LayerAnimator::IMMEDIATELY_ANIMATE_TO_NEW_TARGET
);
1269 slide_settings
.SetTransitionDuration(
1270 base::TimeDelta::FromMilliseconds(kSlideDurationMs
));
1271 SetChildBoundsDirect(window
, bounds
);
1276 void DockedWindowLayoutManager::UpdateDockBounds(
1277 DockedWindowLayoutManagerObserver::Reason reason
) {
1278 int dock_inset
= docked_width_
+ (docked_width_
> 0 ? kMinDockGap
: 0);
1279 const gfx::Rect work_area
=
1280 Shell::GetScreen()->GetDisplayNearestWindow(dock_container_
).work_area();
1281 gfx::Rect bounds
= gfx::Rect(
1282 alignment_
== DOCKED_ALIGNMENT_RIGHT
&& dock_inset
> 0 ?
1283 dock_container_
->bounds().right() - dock_inset
:
1284 dock_container_
->bounds().x(),
1285 dock_container_
->bounds().y(),
1287 work_area
.height());
1288 docked_bounds_
= bounds
+
1289 dock_container_
->GetBoundsInScreen().OffsetFromOrigin();
1291 DockedWindowLayoutManagerObserver
,
1293 OnDockBoundsChanging(bounds
, reason
));
1294 // Show or hide background for docked area.
1295 gfx::Rect
background_bounds(docked_bounds_
);
1296 if (shelf_observer_
)
1297 background_bounds
.Subtract(shelf_observer_
->shelf_bounds_in_screen());
1298 background_widget_
->SetBackgroundBounds(background_bounds
, alignment_
);
1299 if (docked_width_
> 0)
1300 background_widget_
->Show();
1302 background_widget_
->Hide();
1305 void DockedWindowLayoutManager::UpdateStacking(aura::Window
* active_window
) {
1306 if (!active_window
) {
1307 if (!last_active_window_
)
1309 active_window
= last_active_window_
;
1312 // Windows are stacked like a deck of cards:
1321 // Use the middle of each window to figure out how to stack the window.
1322 // This allows us to update the stacking when a window is being dragged around
1324 std::map
<int, aura::Window
*> window_ordering
;
1325 for (aura::Window::Windows::const_iterator it
=
1326 dock_container_
->children().begin();
1327 it
!= dock_container_
->children().end(); ++it
) {
1328 if (!IsUsedByLayout(*it
) ||
1329 ((*it
) == dragged_window_
&& !is_dragged_window_docked_
)) {
1332 gfx::Rect bounds
= (*it
)->bounds();
1333 window_ordering
.insert(std::make_pair(bounds
.y() + bounds
.height() / 2,
1336 int active_center_y
= active_window
->bounds().CenterPoint().y();
1338 aura::Window
* previous_window
= NULL
;
1339 for (std::map
<int, aura::Window
*>::const_iterator it
=
1340 window_ordering
.begin();
1341 it
!= window_ordering
.end() && it
->first
< active_center_y
; ++it
) {
1342 if (previous_window
)
1343 dock_container_
->StackChildAbove(it
->second
, previous_window
);
1344 previous_window
= it
->second
;
1346 for (std::map
<int, aura::Window
*>::const_reverse_iterator it
=
1347 window_ordering
.rbegin();
1348 it
!= window_ordering
.rend() && it
->first
> active_center_y
; ++it
) {
1349 if (previous_window
)
1350 dock_container_
->StackChildAbove(it
->second
, previous_window
);
1351 previous_window
= it
->second
;
1354 if (previous_window
&& active_window
->parent() == dock_container_
)
1355 dock_container_
->StackChildAbove(active_window
, previous_window
);
1356 if (active_window
!= dragged_window_
)
1357 last_active_window_
= active_window
;
1360 ////////////////////////////////////////////////////////////////////////////////
1361 // keyboard::KeyboardControllerObserver implementation:
1363 void DockedWindowLayoutManager::OnKeyboardBoundsChanging(
1364 const gfx::Rect
& keyboard_bounds
) {
1365 // This bounds change will have caused a change to the Shelf which does not
1366 // propagate automatically to this class, so manually recalculate bounds.
1368 UpdateDockBounds(DockedWindowLayoutManagerObserver::KEYBOARD_BOUNDS_CHANGING
);