1 // Copyright 2014 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/system/web_notification/ash_popup_alignment_delegate.h"
7 #include "ash/display/display_controller.h"
8 #include "ash/shelf/shelf_constants.h"
9 #include "ash/shelf/shelf_layout_manager.h"
10 #include "ash/shelf/shelf_types.h"
11 #include "ash/shelf/shelf_widget.h"
12 #include "ash/shell.h"
13 #include "base/i18n/rtl.h"
14 #include "ui/aura/window.h"
15 #include "ui/gfx/display.h"
16 #include "ui/gfx/geometry/rect.h"
17 #include "ui/gfx/screen.h"
18 #include "ui/message_center/message_center_style.h"
19 #include "ui/message_center/views/message_popup_collection.h"
25 const int kToastMarginX
= 3;
27 // If there should be no margin for the first item, this value needs to be
28 // substracted to flush the message to the shelf (the width of the border +
30 const int kNoToastMarginBorderAndShadowOffset
= 2;
34 AshPopupAlignmentDelegate::AshPopupAlignmentDelegate()
35 : display_id_(gfx::Display::kInvalidDisplayID
),
39 system_tray_height_(0) {
42 AshPopupAlignmentDelegate::~AshPopupAlignmentDelegate() {
44 screen_
->RemoveObserver(this);
45 Shell::GetInstance()->RemoveShellObserver(this);
47 shelf_
->RemoveObserver(this);
50 void AshPopupAlignmentDelegate::StartObserving(gfx::Screen
* screen
,
51 const gfx::Display
& display
) {
53 display_id_
= display
.id();
54 root_window_
= ash::Shell::GetInstance()->display_controller()->
55 GetRootWindowForDisplayId(display_id_
);
57 screen
->AddObserver(this);
58 Shell::GetInstance()->AddShellObserver(this);
59 if (system_tray_height_
> 0)
60 OnAutoHideStateChanged(shelf_
->auto_hide_state());
63 void AshPopupAlignmentDelegate::SetSystemTrayHeight(int height
) {
64 system_tray_height_
= height
;
66 // If the shelf is shown during auto-hide state, the distance from the edge
67 // should be reduced by the height of shelf's shown height.
68 if (shelf_
&& shelf_
->visibility_state() == SHELF_AUTO_HIDE
&&
69 shelf_
->auto_hide_state() == SHELF_AUTO_HIDE_SHOWN
) {
70 system_tray_height_
-= kShelfSize
- ShelfLayoutManager::kAutoHideSize
;
73 if (system_tray_height_
> 0)
74 system_tray_height_
+= message_center::kMarginBetweenItems
;
76 system_tray_height_
= 0;
84 int AshPopupAlignmentDelegate::GetToastOriginX(
85 const gfx::Rect
& toast_bounds
) const {
86 // In Ash, RTL UI language mirrors the whole ash layout, so the toast
87 // widgets should be at the bottom-left instead of bottom right.
88 if (base::i18n::IsRTL())
89 return work_area_
.x() + kToastMarginX
;
92 return work_area_
.x() + kToastMarginX
;
93 return work_area_
.right() - kToastMarginX
- toast_bounds
.width();
96 int AshPopupAlignmentDelegate::GetBaseLine() const {
98 ? work_area_
.y() + kNoToastMarginBorderAndShadowOffset
+
100 : work_area_
.bottom() - kNoToastMarginBorderAndShadowOffset
-
104 int AshPopupAlignmentDelegate::GetWorkAreaBottom() const {
105 return work_area_
.bottom() - system_tray_height_
;
108 bool AshPopupAlignmentDelegate::IsTopDown() const {
109 return GetAlignment() == SHELF_ALIGNMENT_TOP
;
112 bool AshPopupAlignmentDelegate::IsFromLeft() const {
113 return GetAlignment() == SHELF_ALIGNMENT_LEFT
;
116 void AshPopupAlignmentDelegate::RecomputeAlignment(
117 const gfx::Display
& display
) {
118 // Nothing needs to be done.
121 ShelfAlignment
AshPopupAlignmentDelegate::GetAlignment() const {
122 return shelf_
? shelf_
->GetAlignment() : SHELF_ALIGNMENT_BOTTOM
;
125 void AshPopupAlignmentDelegate::UpdateShelf() {
129 shelf_
= ShelfLayoutManager::ForShelf(root_window_
);
131 shelf_
->AddObserver(this);
134 void AshPopupAlignmentDelegate::OnDisplayWorkAreaInsetsChanged() {
137 work_area_
= Shell::GetScreen()->GetDisplayNearestWindow(
138 shelf_
->shelf_widget()->GetNativeView()).work_area();
141 void AshPopupAlignmentDelegate::OnAutoHideStateChanged(
142 ShelfAutoHideState new_state
) {
143 work_area_
= Shell::GetScreen()->GetDisplayNearestWindow(
144 shelf_
->shelf_widget()->GetNativeView()).work_area();
146 if ((shelf_
->visibility_state() == SHELF_AUTO_HIDE
) &&
147 new_state
== SHELF_AUTO_HIDE_SHOWN
) {
148 // Since the work_area is already reduced by kAutoHideSize, the inset width
149 // should be just the difference.
150 width
= kShelfSize
- ShelfLayoutManager::kAutoHideSize
;
152 work_area_
.Inset(shelf_
->SelectValueForShelfAlignment(
153 gfx::Insets(0, 0, width
, 0),
154 gfx::Insets(0, width
, 0, 0),
155 gfx::Insets(0, 0, 0, width
),
156 gfx::Insets(width
, 0, 0, 0)));
158 DoUpdateIfPossible();
161 void AshPopupAlignmentDelegate::OnDisplayAdded(
162 const gfx::Display
& new_display
) {
165 void AshPopupAlignmentDelegate::OnDisplayRemoved(
166 const gfx::Display
& old_display
) {
169 void AshPopupAlignmentDelegate::OnDisplayMetricsChanged(
170 const gfx::Display
& display
,
172 if (display
.id() == display_id_
&& shelf_
)
173 OnAutoHideStateChanged(shelf_
->auto_hide_state());