- add sources.
[platform/framework/web/crosswalk.git] / src / ash / system / web_notification / web_notification_tray_unittest.cc
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "ash/system/web_notification/web_notification_tray.h"
6
7 #include <vector>
8
9 #include "ash/display/display_manager.h"
10 #include "ash/root_window_controller.h"
11 #include "ash/shelf/shelf_layout_manager.h"
12 #include "ash/shelf/shelf_widget.h"
13 #include "ash/shell.h"
14 #include "ash/system/status_area_widget.h"
15 #include "ash/system/tray/system_tray.h"
16 #include "ash/system/tray/system_tray_item.h"
17 #include "ash/test/ash_test_base.h"
18 #include "ash/test/test_system_tray_delegate.h"
19 #include "ash/wm/window_state.h"
20 #include "base/strings/stringprintf.h"
21 #include "base/strings/utf_string_conversions.h"
22 #include "ui/aura/client/aura_constants.h"
23 #include "ui/aura/test/event_generator.h"
24 #include "ui/aura/window.h"
25 #include "ui/gfx/display.h"
26 #include "ui/gfx/screen.h"
27 #include "ui/message_center/message_center_style.h"
28 #include "ui/message_center/message_center_tray.h"
29 #include "ui/message_center/message_center_util.h"
30 #include "ui/message_center/notification_list.h"
31 #include "ui/message_center/notification_types.h"
32 #include "ui/message_center/views/message_center_bubble.h"
33 #include "ui/message_center/views/message_popup_collection.h"
34 #include "ui/views/controls/label.h"
35 #include "ui/views/layout/fill_layout.h"
36 #include "ui/views/view.h"
37 #include "ui/views/widget/widget.h"
38
39 namespace ash {
40
41 namespace {
42
43 WebNotificationTray* GetTray() {
44   return Shell::GetPrimaryRootWindowController()->shelf()->
45       status_area_widget()->web_notification_tray();
46 }
47
48 WebNotificationTray* GetSecondaryTray() {
49   internal::RootWindowController* primary_controller =
50       Shell::GetPrimaryRootWindowController();
51   Shell::RootWindowControllerList controllers =
52       Shell::GetAllRootWindowControllers();
53   for (size_t i = 0; i < controllers.size(); ++i) {
54     if (controllers[i] != primary_controller) {
55       return controllers[i]->shelf()->
56           status_area_widget()->web_notification_tray();
57     }
58   }
59
60   return NULL;
61 }
62
63 message_center::MessageCenter* GetMessageCenter() {
64   return GetTray()->message_center();
65 }
66
67 SystemTray* GetSystemTray() {
68   return Shell::GetPrimaryRootWindowController()->shelf()->
69       status_area_widget()->system_tray();
70 }
71
72 // Trivial item implementation for testing PopupAndSystemTray test case.
73 class TestItem : public SystemTrayItem {
74  public:
75   TestItem() : SystemTrayItem(GetSystemTray()) {}
76
77   virtual views::View* CreateDefaultView(user::LoginStatus status) OVERRIDE {
78     views::View* default_view = new views::View;
79     default_view->SetLayoutManager(new views::FillLayout);
80     default_view->AddChildView(new views::Label(UTF8ToUTF16("Default")));
81     return default_view;
82   }
83
84   virtual views::View* CreateNotificationView(
85       user::LoginStatus status) OVERRIDE {
86     return new views::View;
87   }
88
89  private:
90   DISALLOW_COPY_AND_ASSIGN(TestItem);
91 };
92
93 }  // namespace
94
95 class WebNotificationTrayTest : public test::AshTestBase {
96  public:
97   WebNotificationTrayTest() {}
98   virtual ~WebNotificationTrayTest() {}
99
100   virtual void TearDown() OVERRIDE {
101     GetMessageCenter()->RemoveAllNotifications(false);
102     test::AshTestBase::TearDown();
103   }
104
105  protected:
106   void AddNotification(const std::string& id) {
107     scoped_ptr<message_center::Notification> notification;
108     notification.reset(new message_center::Notification(
109         message_center::NOTIFICATION_TYPE_SIMPLE,
110         id,
111         ASCIIToUTF16("Test Web Notification"),
112         ASCIIToUTF16("Notification message body."),
113         gfx::Image(),
114         ASCIIToUTF16("www.test.org"),
115         message_center::NotifierId(),
116         message_center::RichNotificationData(),
117         NULL /* delegate */));
118     GetMessageCenter()->AddNotification(notification.Pass());
119   }
120
121   void UpdateNotification(const std::string& old_id,
122                           const std::string& new_id) {
123     scoped_ptr<message_center::Notification> notification;
124     notification.reset(new message_center::Notification(
125         message_center::NOTIFICATION_TYPE_SIMPLE,
126         new_id,
127         ASCIIToUTF16("Updated Web Notification"),
128         ASCIIToUTF16("Updated message body."),
129         gfx::Image(),
130         ASCIIToUTF16("www.test.org"),
131         message_center::NotifierId(),
132         message_center::RichNotificationData(),
133         NULL /* delegate */));
134     GetMessageCenter()->UpdateNotification(old_id, notification.Pass());
135   }
136
137   void RemoveNotification(const std::string& id) {
138     GetMessageCenter()->RemoveNotification(id, false);
139   }
140
141   views::Widget* GetWidget() {
142     return GetTray()->GetWidget();
143   }
144
145   gfx::Rect GetPopupWorkArea() {
146     return GetPopupWorkAreaForTray(GetTray());
147   }
148
149   gfx::Rect GetPopupWorkAreaForTray(WebNotificationTray* tray) {
150     return tray->popup_collection_->work_area_;
151   }
152
153   bool IsPopupVisible() {
154     return GetTray()->IsPopupVisible();
155   }
156
157  private:
158   DISALLOW_COPY_AND_ASSIGN(WebNotificationTrayTest);
159 };
160
161 TEST_F(WebNotificationTrayTest, WebNotifications) {
162   // TODO(mukai): move this test case to ui/message_center.
163   ASSERT_TRUE(GetWidget());
164
165   // Add a notification.
166   AddNotification("test_id1");
167   EXPECT_EQ(1u, GetMessageCenter()->NotificationCount());
168   EXPECT_TRUE(GetMessageCenter()->HasNotification("test_id1"));
169   AddNotification("test_id2");
170   AddNotification("test_id2");
171   EXPECT_EQ(2u, GetMessageCenter()->NotificationCount());
172   EXPECT_TRUE(GetMessageCenter()->HasNotification("test_id2"));
173
174   // Ensure that updating a notification does not affect the count.
175   UpdateNotification("test_id2", "test_id3");
176   UpdateNotification("test_id3", "test_id3");
177   EXPECT_EQ(2u, GetMessageCenter()->NotificationCount());
178   EXPECT_FALSE(GetMessageCenter()->HasNotification("test_id2"));
179
180   // Ensure that Removing the first notification removes it from the tray.
181   RemoveNotification("test_id1");
182   EXPECT_FALSE(GetMessageCenter()->HasNotification("test_id1"));
183   EXPECT_EQ(1u, GetMessageCenter()->NotificationCount());
184
185   // Remove the remianing notification.
186   RemoveNotification("test_id3");
187   EXPECT_EQ(0u, GetMessageCenter()->NotificationCount());
188   EXPECT_FALSE(GetMessageCenter()->HasNotification("test_id3"));
189 }
190
191 TEST_F(WebNotificationTrayTest, WebNotificationPopupBubble) {
192   // TODO(mukai): move this test case to ui/message_center.
193   ASSERT_TRUE(GetWidget());
194
195   // Adding a notification should show the popup bubble.
196   AddNotification("test_id1");
197   EXPECT_TRUE(GetTray()->IsPopupVisible());
198
199   // Updating a notification should not hide the popup bubble.
200   AddNotification("test_id2");
201   UpdateNotification("test_id2", "test_id3");
202   EXPECT_TRUE(GetTray()->IsPopupVisible());
203
204   // Removing the first notification should not hide the popup bubble.
205   RemoveNotification("test_id1");
206   EXPECT_TRUE(GetTray()->IsPopupVisible());
207
208   // Removing the visible notification should hide the popup bubble.
209   RemoveNotification("test_id3");
210   EXPECT_FALSE(GetTray()->IsPopupVisible());
211 }
212
213 using message_center::NotificationList;
214
215
216 // Flakily fails. http://crbug.com/229791
217 TEST_F(WebNotificationTrayTest, DISABLED_ManyMessageCenterNotifications) {
218   // Add the max visible notifications +1, ensure the correct visible number.
219   size_t notifications_to_add =
220       message_center::kMaxVisibleMessageCenterNotifications + 1;
221   for (size_t i = 0; i < notifications_to_add; ++i) {
222     std::string id = base::StringPrintf("test_id%d", static_cast<int>(i));
223     AddNotification(id);
224   }
225   bool shown = GetTray()->message_center_tray_->ShowMessageCenterBubble();
226   EXPECT_TRUE(shown);
227   RunAllPendingInMessageLoop();
228   EXPECT_TRUE(GetTray()->message_center_bubble() != NULL);
229   EXPECT_EQ(notifications_to_add,
230             GetMessageCenter()->NotificationCount());
231   EXPECT_EQ(message_center::kMaxVisibleMessageCenterNotifications,
232             GetTray()->GetMessageCenterBubbleForTest()->
233                 NumMessageViewsForTest());
234 }
235
236 // Flakily times out. http://crbug.com/229792
237 TEST_F(WebNotificationTrayTest, DISABLED_ManyPopupNotifications) {
238   // Add the max visible popup notifications +1, ensure the correct num visible.
239   size_t notifications_to_add =
240       message_center::kMaxVisiblePopupNotifications + 1;
241   for (size_t i = 0; i < notifications_to_add; ++i) {
242     std::string id = base::StringPrintf("test_id%d", static_cast<int>(i));
243     AddNotification(id);
244   }
245   GetTray()->ShowPopups();
246   EXPECT_TRUE(GetTray()->IsPopupVisible());
247   EXPECT_EQ(notifications_to_add,
248             GetMessageCenter()->NotificationCount());
249   NotificationList::PopupNotifications popups =
250       GetMessageCenter()->GetPopupNotifications();
251   EXPECT_EQ(message_center::kMaxVisiblePopupNotifications, popups.size());
252 }
253
254 #if defined(OS_CHROMEOS)
255 // Display notification is ChromeOS only.
256 #define MAYBE_PopupShownOnBothDisplays PopupShownOnBothDisplays
257 #define MAYBE_PopupAndSystemTrayMultiDisplay PopupAndSystemTrayMultiDisplay
258 #else
259 #define MAYBE_PopupShownOnBothDisplays DISABLED_PopupShownOnBothDisplays
260 #define MAYBE_PopupAndSystemTrayMultiDisplay \
261   DISABLED_PopupAndSystemTrayMultiDisplay
262 #endif
263
264 // Verifies if the notification appears on both displays when extended mode.
265 TEST_F(WebNotificationTrayTest, MAYBE_PopupShownOnBothDisplays) {
266   if (!SupportsMultipleDisplays())
267     return;
268
269   // Enables to appear the notification for display changes.
270   test::TestSystemTrayDelegate* tray_delegate =
271       static_cast<test::TestSystemTrayDelegate*>(
272           Shell::GetInstance()->system_tray_delegate());
273   tray_delegate->set_should_show_display_notification(true);
274
275   UpdateDisplay("400x400,200x200");
276   // UpdateDisplay() creates the display notifications, so popup is visible.
277   EXPECT_TRUE(GetTray()->IsPopupVisible());
278   WebNotificationTray* secondary_tray = GetSecondaryTray();
279   ASSERT_TRUE(secondary_tray);
280   EXPECT_TRUE(secondary_tray->IsPopupVisible());
281
282   // Transition to mirroring and then back to extended display, which recreates
283   // root window controller and shelf with having notifications. This code
284   // verifies it doesn't cause crash and popups are still visible. See
285   // http://crbug.com/263664
286   internal::DisplayManager* display_manager =
287       Shell::GetInstance()->display_manager();
288
289   display_manager->SetSecondDisplayMode(internal::DisplayManager::MIRRORING);
290   UpdateDisplay("400x400,200x200");
291   EXPECT_TRUE(GetTray()->IsPopupVisible());
292   EXPECT_FALSE(GetSecondaryTray());
293
294   display_manager->SetSecondDisplayMode(internal::DisplayManager::EXTENDED);
295   UpdateDisplay("400x400,200x200");
296   EXPECT_TRUE(GetTray()->IsPopupVisible());
297   secondary_tray = GetSecondaryTray();
298   ASSERT_TRUE(secondary_tray);
299   EXPECT_TRUE(secondary_tray->IsPopupVisible());
300 }
301
302 #if defined(OS_CHROMEOS)
303 // PopupAndSystemTray may fail in platforms other than ChromeOS because the
304 // RootWindow's bound can be bigger than gfx::Display's work area so that
305 // openingsystem tray doesn't affect at all the work area of popups.
306 #define MAYBE_PopupAndSystemTray PopupAndSystemTray
307 #define MAYBE_PopupAndAutoHideShelf PopupAndAutoHideShelf
308 #define MAYBE_PopupAndFullscreen PopupAndFullscreen
309 #else
310 #define MAYBE_PopupAndSystemTray DISABLED_PopupAndSystemTray
311 #define MAYBE_PopupAndAutoHideShelf DISABLED_PopupAndAutoHideShelf
312 #define MAYBE_PopupAndFullscreen DISABLED_PopupAndFullscreen
313 #endif
314
315 TEST_F(WebNotificationTrayTest, MAYBE_PopupAndSystemTray) {
316   TestItem* test_item = new TestItem;
317   GetSystemTray()->AddTrayItem(test_item);
318
319   AddNotification("test_id");
320   EXPECT_TRUE(GetTray()->IsPopupVisible());
321   gfx::Rect work_area = GetPopupWorkArea();
322
323   // System tray is created, the popup's work area should be narrowed but still
324   // visible.
325   GetSystemTray()->ShowDefaultView(BUBBLE_CREATE_NEW);
326   EXPECT_TRUE(GetTray()->IsPopupVisible());
327   gfx::Rect work_area_with_tray = GetPopupWorkArea();
328   EXPECT_GT(work_area.size().GetArea(), work_area_with_tray.size().GetArea());
329
330   // System tray notification is also created, the popup's work area is narrowed
331   // even more, but still visible.
332   GetSystemTray()->ShowNotificationView(test_item);
333   EXPECT_TRUE(GetTray()->IsPopupVisible());
334   gfx::Rect work_area_with_tray_notificaiton = GetPopupWorkArea();
335   EXPECT_GT(work_area.size().GetArea(),
336             work_area_with_tray_notificaiton.size().GetArea());
337   EXPECT_GT(work_area_with_tray.size().GetArea(),
338             work_area_with_tray_notificaiton.size().GetArea());
339
340   // Close system tray, only system tray notifications.
341   GetSystemTray()->ClickedOutsideBubble();
342   EXPECT_TRUE(GetTray()->IsPopupVisible());
343   gfx::Rect work_area_with_notification = GetPopupWorkArea();
344   EXPECT_GT(work_area.size().GetArea(),
345             work_area_with_notification.size().GetArea());
346   EXPECT_LT(work_area_with_tray_notificaiton.size().GetArea(),
347             work_area_with_notification.size().GetArea());
348
349   // Close the system tray notifications.
350   GetSystemTray()->HideNotificationView(test_item);
351   EXPECT_TRUE(GetTray()->IsPopupVisible());
352   EXPECT_EQ(work_area.ToString(), GetPopupWorkArea().ToString());
353 }
354
355 TEST_F(WebNotificationTrayTest, MAYBE_PopupAndAutoHideShelf) {
356   AddNotification("test_id");
357   EXPECT_TRUE(GetTray()->IsPopupVisible());
358   gfx::Rect work_area = GetPopupWorkArea();
359
360   // Shelf's auto-hide state won't be HIDDEN unless window exists.
361   scoped_ptr<aura::Window> window(
362       CreateTestWindowInShellWithBounds(gfx::Rect(1, 2, 3, 4)));
363   internal::ShelfLayoutManager* shelf =
364       Shell::GetPrimaryRootWindowController()->GetShelfLayoutManager();
365   shelf->SetAutoHideBehavior(SHELF_AUTO_HIDE_BEHAVIOR_ALWAYS);
366
367   EXPECT_EQ(SHELF_AUTO_HIDE_HIDDEN, shelf->auto_hide_state());
368   gfx::Rect work_area_auto_hidden = GetPopupWorkArea();
369   EXPECT_LT(work_area.size().GetArea(), work_area_auto_hidden.size().GetArea());
370
371   // Close the window, which shows the shelf.
372   window.reset();
373   EXPECT_EQ(SHELF_AUTO_HIDE_SHOWN, shelf->auto_hide_state());
374   gfx::Rect work_area_auto_shown = GetPopupWorkArea();
375   EXPECT_EQ(work_area.ToString(), work_area_auto_shown.ToString());
376
377   // Create the system tray during auto-hide.
378   window.reset(CreateTestWindowInShellWithBounds(gfx::Rect(1, 2, 3, 4)));
379   TestItem* test_item = new TestItem;
380   GetSystemTray()->AddTrayItem(test_item);
381   GetSystemTray()->ShowDefaultView(BUBBLE_CREATE_NEW);
382
383   EXPECT_EQ(SHELF_AUTO_HIDE_SHOWN, shelf->auto_hide_state());
384   EXPECT_TRUE(GetTray()->IsPopupVisible());
385   gfx::Rect work_area_with_tray = GetPopupWorkArea();
386   EXPECT_GT(work_area_auto_shown.size().GetArea(),
387             work_area_with_tray.size().GetArea());
388
389   // Create tray notification.
390   GetSystemTray()->ShowNotificationView(test_item);
391   EXPECT_EQ(SHELF_AUTO_HIDE_SHOWN, shelf->auto_hide_state());
392   gfx::Rect work_area_with_tray_notification = GetPopupWorkArea();
393   EXPECT_GT(work_area_with_tray.size().GetArea(),
394             work_area_with_tray_notification.size().GetArea());
395
396   // Close the system tray.
397   GetSystemTray()->ClickedOutsideBubble();
398   shelf->UpdateAutoHideState();
399   RunAllPendingInMessageLoop();
400   EXPECT_EQ(SHELF_AUTO_HIDE_HIDDEN, shelf->auto_hide_state());
401   gfx::Rect work_area_hidden_with_tray_notification = GetPopupWorkArea();
402   EXPECT_LT(work_area_with_tray_notification.size().GetArea(),
403             work_area_hidden_with_tray_notification.size().GetArea());
404   EXPECT_GT(work_area_auto_hidden.size().GetArea(),
405             work_area_hidden_with_tray_notification.size().GetArea());
406
407   // Close the window again, which shows the shelf.
408   window.reset();
409   EXPECT_EQ(SHELF_AUTO_HIDE_SHOWN, shelf->auto_hide_state());
410   gfx::Rect work_area_shown_with_tray_notification = GetPopupWorkArea();
411   EXPECT_GT(work_area_hidden_with_tray_notification.size().GetArea(),
412             work_area_shown_with_tray_notification.size().GetArea());
413   EXPECT_GT(work_area_auto_shown.size().GetArea(),
414             work_area_shown_with_tray_notification.size().GetArea());
415 }
416
417 TEST_F(WebNotificationTrayTest, MAYBE_PopupAndFullscreen) {
418   AddNotification("test_id");
419   EXPECT_TRUE(IsPopupVisible());
420   gfx::Rect work_area = GetPopupWorkArea();
421
422   // Checks the work area for normal auto-hidden state.
423   scoped_ptr<aura::Window> window(
424       CreateTestWindowInShellWithBounds(gfx::Rect(1, 2, 3, 4)));
425   internal::ShelfLayoutManager* shelf =
426       Shell::GetPrimaryRootWindowController()->GetShelfLayoutManager();
427   shelf->SetAutoHideBehavior(SHELF_AUTO_HIDE_BEHAVIOR_ALWAYS);
428   EXPECT_EQ(SHELF_AUTO_HIDE_HIDDEN, shelf->auto_hide_state());
429   gfx::Rect work_area_auto_hidden = GetPopupWorkArea();
430   shelf->SetAutoHideBehavior(SHELF_AUTO_HIDE_BEHAVIOR_NEVER);
431
432   // Put |window| into fullscreen without forcing the shelf to hide. Currently,
433   // this is used by immersive fullscreen and forces the shelf to be auto
434   // hidden.
435   wm::GetWindowState(window.get())->set_hide_shelf_when_fullscreen(false);
436   window->SetProperty(aura::client::kShowStateKey, ui::SHOW_STATE_FULLSCREEN);
437   RunAllPendingInMessageLoop();
438
439   // The work area for auto-hidden status of fullscreen is a bit larger
440   // since it doesn't even have the 3-pixel width.
441   EXPECT_EQ(SHELF_AUTO_HIDE_HIDDEN, shelf->auto_hide_state());
442   gfx::Rect work_area_fullscreen_hidden = GetPopupWorkArea();
443   EXPECT_EQ(work_area_auto_hidden.ToString(),
444             work_area_fullscreen_hidden.ToString());
445
446   // Move the mouse cursor at the bottom, which shows the shelf.
447   aura::test::EventGenerator generator(Shell::GetPrimaryRootWindow());
448   gfx::Point bottom_right =
449       Shell::GetScreen()->GetPrimaryDisplay().bounds().bottom_right();
450   bottom_right.Offset(-1, -1);
451   generator.MoveMouseTo(bottom_right);
452   shelf->UpdateAutoHideStateNow();
453   EXPECT_EQ(SHELF_AUTO_HIDE_SHOWN, shelf->auto_hide_state());
454   EXPECT_EQ(work_area.ToString(), GetPopupWorkArea().ToString());
455
456   generator.MoveMouseTo(work_area.CenterPoint());
457   shelf->UpdateAutoHideStateNow();
458   EXPECT_EQ(SHELF_AUTO_HIDE_HIDDEN, shelf->auto_hide_state());
459   EXPECT_EQ(work_area_auto_hidden.ToString(), GetPopupWorkArea().ToString());
460 }
461
462 TEST_F(WebNotificationTrayTest, MAYBE_PopupAndSystemTrayMultiDisplay) {
463   UpdateDisplay("800x600,600x400");
464
465   AddNotification("test_id");
466   gfx::Rect work_area = GetPopupWorkArea();
467   gfx::Rect work_area_second = GetPopupWorkAreaForTray(GetSecondaryTray());
468
469   // System tray is created on the primary display. The popups in the secondary
470   // tray aren't affected.
471   GetSystemTray()->ShowDefaultView(BUBBLE_CREATE_NEW);
472   EXPECT_GT(work_area.size().GetArea(), GetPopupWorkArea().size().GetArea());
473   EXPECT_EQ(work_area_second.ToString(),
474             GetPopupWorkAreaForTray(GetSecondaryTray()).ToString());
475 }
476
477 }  // namespace ash