OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include <algorithm> | 5 #include <algorithm> |
6 #include <set> | 6 #include <set> |
7 | 7 |
8 #include "base/basictypes.h" | 8 #include "base/basictypes.h" |
9 #include "base/bind.h" | 9 #include "base/bind.h" |
10 #include "base/memory/scoped_ptr.h" | 10 #include "base/memory/scoped_ptr.h" |
11 #include "base/message_loop/message_loop.h" | 11 #include "base/message_loop/message_loop.h" |
12 #include "base/run_loop.h" | 12 #include "base/run_loop.h" |
13 #include "base/strings/utf_string_conversions.h" | 13 #include "base/strings/utf_string_conversions.h" |
14 #include "testing/gtest/include/gtest/gtest.h" | 14 #include "testing/gtest/include/gtest/gtest.h" |
| 15 #include "ui/aura/client/aura_constants.h" |
| 16 #include "ui/aura/client/window_tree_client.h" |
| 17 #include "ui/aura/root_window.h" |
| 18 #include "ui/aura/test/event_generator.h" |
| 19 #include "ui/aura/test/test_window_delegate.h" |
| 20 #include "ui/aura/window.h" |
15 #include "ui/base/hit_test.h" | 21 #include "ui/base/hit_test.h" |
16 #include "ui/events/event_utils.h" | 22 #include "ui/events/event_utils.h" |
17 #include "ui/gfx/native_widget_types.h" | 23 #include "ui/gfx/native_widget_types.h" |
18 #include "ui/gfx/point.h" | 24 #include "ui/gfx/point.h" |
19 #include "ui/views/bubble/bubble_delegate.h" | 25 #include "ui/views/bubble/bubble_delegate.h" |
20 #include "ui/views/controls/textfield/textfield.h" | 26 #include "ui/views/controls/textfield/textfield.h" |
21 #include "ui/views/test/test_views_delegate.h" | 27 #include "ui/views/test/test_views_delegate.h" |
22 #include "ui/views/test/widget_test.h" | 28 #include "ui/views/test/widget_test.h" |
23 #include "ui/views/views_delegate.h" | 29 #include "ui/views/views_delegate.h" |
| 30 #include "ui/views/widget/native_widget_aura.h" |
24 #include "ui/views/widget/native_widget_delegate.h" | 31 #include "ui/views/widget/native_widget_delegate.h" |
25 #include "ui/views/widget/root_view.h" | 32 #include "ui/views/widget/root_view.h" |
| 33 #include "ui/views/widget/widget_deletion_observer.h" |
26 #include "ui/views/window/dialog_delegate.h" | 34 #include "ui/views/window/dialog_delegate.h" |
27 #include "ui/views/window/native_frame_view.h" | 35 #include "ui/views/window/native_frame_view.h" |
28 | 36 |
29 #if defined(USE_AURA) | |
30 #include "ui/aura/client/aura_constants.h" | |
31 #include "ui/aura/client/window_tree_client.h" | |
32 #include "ui/aura/root_window.h" | |
33 #include "ui/aura/test/test_window_delegate.h" | |
34 #include "ui/aura/window.h" | |
35 #include "ui/views/widget/native_widget_aura.h" | |
36 #if !defined(OS_CHROMEOS) | 37 #if !defined(OS_CHROMEOS) |
37 #include "ui/views/widget/desktop_aura/desktop_native_widget_aura.h" | 38 #include "ui/views/widget/desktop_aura/desktop_native_widget_aura.h" |
38 #endif | 39 #endif |
39 #elif defined(OS_WIN) | |
40 #include "ui/views/widget/native_widget_win.h" | |
41 #endif | |
42 | 40 |
43 #if defined(OS_WIN) | 41 #if defined(OS_WIN) |
44 #include "ui/views/win/hwnd_util.h" | 42 #include "ui/views/win/hwnd_util.h" |
45 #endif | 43 #endif |
46 | 44 |
47 namespace views { | 45 namespace views { |
48 namespace test { | 46 namespace test { |
49 | 47 |
50 // A view that keeps track of the events it receives, but consumes no events. | 48 // A view that keeps track of the events it receives, but consumes no events. |
51 class EventCountView : public View { | 49 class EventCountView : public View { |
(...skipping 103 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
155 private: | 153 private: |
156 void RecordEvent(const ui::Event& event) { | 154 void RecordEvent(const ui::Event& event) { |
157 ++event_count_[event.type()]; | 155 ++event_count_[event.type()]; |
158 } | 156 } |
159 | 157 |
160 std::map<ui::EventType, int> event_count_; | 158 std::map<ui::EventType, int> event_count_; |
161 | 159 |
162 DISALLOW_COPY_AND_ASSIGN(EventCountHandler); | 160 DISALLOW_COPY_AND_ASSIGN(EventCountHandler); |
163 }; | 161 }; |
164 | 162 |
| 163 // Class that closes the widget (which ends up deleting it immediately) when the |
| 164 // appropriate event is received. |
| 165 class CloseWidgetView : public View { |
| 166 public: |
| 167 explicit CloseWidgetView(ui::EventType event_type) |
| 168 : event_type_(event_type) { |
| 169 } |
| 170 |
| 171 // ui::EventHandler override: |
| 172 virtual void OnEvent(ui::Event* event) OVERRIDE { |
| 173 if (event->type() == event_type_) { |
| 174 // Go through NativeWidgetPrivate to simulate what happens if the OS |
| 175 // deletes the NativeWindow out from under us. |
| 176 GetWidget()->native_widget_private()->CloseNow(); |
| 177 } else { |
| 178 View::OnEvent(event); |
| 179 if (!event->IsTouchEvent()) |
| 180 event->SetHandled(); |
| 181 } |
| 182 } |
| 183 |
| 184 private: |
| 185 const ui::EventType event_type_; |
| 186 |
| 187 DISALLOW_COPY_AND_ASSIGN(CloseWidgetView); |
| 188 }; |
| 189 |
165 ui::WindowShowState GetWidgetShowState(const Widget* widget) { | 190 ui::WindowShowState GetWidgetShowState(const Widget* widget) { |
166 // Use IsMaximized/IsMinimized/IsFullScreen instead of GetWindowPlacement | 191 // Use IsMaximized/IsMinimized/IsFullScreen instead of GetWindowPlacement |
167 // because the former is implemented on all platforms but the latter is not. | 192 // because the former is implemented on all platforms but the latter is not. |
168 return widget->IsFullscreen() ? ui::SHOW_STATE_FULLSCREEN : | 193 return widget->IsFullscreen() ? ui::SHOW_STATE_FULLSCREEN : |
169 widget->IsMaximized() ? ui::SHOW_STATE_MAXIMIZED : | 194 widget->IsMaximized() ? ui::SHOW_STATE_MAXIMIZED : |
170 widget->IsMinimized() ? ui::SHOW_STATE_MINIMIZED : | 195 widget->IsMinimized() ? ui::SHOW_STATE_MINIMIZED : |
171 widget->IsActive() ? ui::SHOW_STATE_NORMAL : | 196 widget->IsActive() ? ui::SHOW_STATE_NORMAL : |
172 ui::SHOW_STATE_INACTIVE; | 197 ui::SHOW_STATE_INACTIVE; |
173 } | 198 } |
174 | 199 |
(...skipping 423 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
598 widget->IsMouseEventsEnabled(); | 623 widget->IsMouseEventsEnabled(); |
599 widget->SetNativeWindowProperty("xx", widget); | 624 widget->SetNativeWindowProperty("xx", widget); |
600 widget->GetNativeWindowProperty("xx"); | 625 widget->GetNativeWindowProperty("xx"); |
601 widget->GetFocusTraversable(); | 626 widget->GetFocusTraversable(); |
602 widget->GetLayer(); | 627 widget->GetLayer(); |
603 widget->ReorderNativeViews(); | 628 widget->ReorderNativeViews(); |
604 widget->SetCapture(widget->GetRootView()); | 629 widget->SetCapture(widget->GetRootView()); |
605 widget->ReleaseCapture(); | 630 widget->ReleaseCapture(); |
606 widget->HasCapture(); | 631 widget->HasCapture(); |
607 widget->GetWorkAreaBoundsInScreen(); | 632 widget->GetWorkAreaBoundsInScreen(); |
608 // These three crash with NativeWidgetWin, so I'm assuming we don't need | |
609 // them to work for the other NativeWidget impls. | |
610 // widget->CenterWindow(gfx::Size(50, 60)); | |
611 // widget->GetRestoredBounds(); | |
612 // widget->ShowInactive(); | |
613 // widget->Show(); | |
614 } | 633 } |
615 | 634 |
616 private: | 635 private: |
617 DISALLOW_COPY_AND_ASSIGN(WidgetWithDestroyedNativeViewTest); | 636 DISALLOW_COPY_AND_ASSIGN(WidgetWithDestroyedNativeViewTest); |
618 }; | 637 }; |
619 | 638 |
620 TEST_F(WidgetWithDestroyedNativeViewTest, Test) { | 639 TEST_F(WidgetWithDestroyedNativeViewTest, Test) { |
621 { | 640 { |
622 Widget widget; | 641 Widget widget; |
623 Widget::InitParams params = CreateParams(Widget::InitParams::TYPE_POPUP); | 642 Widget::InitParams params = CreateParams(Widget::InitParams::TYPE_POPUP); |
(...skipping 638 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1262 EXPECT_FALSE(widget.GetNativeView()->IsVisible()); | 1281 EXPECT_FALSE(widget.GetNativeView()->IsVisible()); |
1263 widget.Show(); | 1282 widget.Show(); |
1264 EXPECT_TRUE(widget.GetNativeView()->IsVisible()); | 1283 EXPECT_TRUE(widget.GetNativeView()->IsVisible()); |
1265 } | 1284 } |
1266 | 1285 |
1267 // The following code verifies we can correctly destroy a Widget from a mouse | 1286 // The following code verifies we can correctly destroy a Widget from a mouse |
1268 // enter/exit. We could test move/drag/enter/exit but in general we don't run | 1287 // enter/exit. We could test move/drag/enter/exit but in general we don't run |
1269 // nested message loops from such events, nor has the code ever really dealt | 1288 // nested message loops from such events, nor has the code ever really dealt |
1270 // with this situation. | 1289 // with this situation. |
1271 | 1290 |
1272 // Class that closes the widget (which ends up deleting it immediately) when the | |
1273 // appropriate event is received. | |
1274 class CloseWidgetView : public View { | |
1275 public: | |
1276 explicit CloseWidgetView(ui::EventType event_type) | |
1277 : event_type_(event_type) { | |
1278 } | |
1279 | |
1280 // View overrides: | |
1281 virtual bool OnMousePressed(const ui::MouseEvent& event) OVERRIDE { | |
1282 if (!CloseWidget(event)) | |
1283 View::OnMousePressed(event); | |
1284 return true; | |
1285 } | |
1286 virtual bool OnMouseDragged(const ui::MouseEvent& event) OVERRIDE { | |
1287 if (!CloseWidget(event)) | |
1288 View::OnMouseDragged(event); | |
1289 return true; | |
1290 } | |
1291 virtual void OnMouseReleased(const ui::MouseEvent& event) OVERRIDE { | |
1292 if (!CloseWidget(event)) | |
1293 View::OnMouseReleased(event); | |
1294 } | |
1295 virtual void OnMouseMoved(const ui::MouseEvent& event) OVERRIDE { | |
1296 if (!CloseWidget(event)) | |
1297 View::OnMouseMoved(event); | |
1298 } | |
1299 virtual void OnMouseEntered(const ui::MouseEvent& event) OVERRIDE { | |
1300 if (!CloseWidget(event)) | |
1301 View::OnMouseEntered(event); | |
1302 } | |
1303 | |
1304 private: | |
1305 bool CloseWidget(const ui::LocatedEvent& event) { | |
1306 if (event.type() == event_type_) { | |
1307 // Go through NativeWidgetPrivate to simulate what happens if the OS | |
1308 // deletes the NativeWindow out from under us. | |
1309 GetWidget()->native_widget_private()->CloseNow(); | |
1310 return true; | |
1311 } | |
1312 return false; | |
1313 } | |
1314 | |
1315 const ui::EventType event_type_; | |
1316 | |
1317 DISALLOW_COPY_AND_ASSIGN(CloseWidgetView); | |
1318 }; | |
1319 | |
1320 // Generates two moves (first generates enter, second real move), a press, drag | 1291 // Generates two moves (first generates enter, second real move), a press, drag |
1321 // and release stopping at |last_event_type|. | 1292 // and release stopping at |last_event_type|. |
1322 void GenerateMouseEvents(Widget* widget, ui::EventType last_event_type) { | 1293 void GenerateMouseEvents(Widget* widget, ui::EventType last_event_type) { |
1323 const gfx::Rect screen_bounds(widget->GetWindowBoundsInScreen()); | 1294 const gfx::Rect screen_bounds(widget->GetWindowBoundsInScreen()); |
1324 ui::MouseEvent move_event(ui::ET_MOUSE_MOVED, screen_bounds.CenterPoint(), | 1295 ui::MouseEvent move_event(ui::ET_MOUSE_MOVED, screen_bounds.CenterPoint(), |
1325 screen_bounds.CenterPoint(), 0, 0); | 1296 screen_bounds.CenterPoint(), 0, 0); |
1326 aura::WindowEventDispatcher* dispatcher = | 1297 aura::WindowEventDispatcher* dispatcher = |
1327 widget->GetNativeWindow()->GetDispatcher(); | 1298 widget->GetNativeWindow()->GetDispatcher(); |
1328 ui::EventDispatchDetails details = dispatcher->OnEventFromSource(&move_event); | 1299 ui::EventDispatchDetails details = dispatcher->OnEventFromSource(&move_event); |
1329 if (last_event_type == ui::ET_MOUSE_ENTERED || details.dispatcher_destroyed) | 1300 if (last_event_type == ui::ET_MOUSE_ENTERED || details.dispatcher_destroyed) |
(...skipping 378 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1708 set_views_delegate(delegate); // ViewsTestBase takes ownership. | 1679 set_views_delegate(delegate); // ViewsTestBase takes ownership. |
1709 scoped_ptr<Widget> widget(new Widget); | 1680 scoped_ptr<Widget> widget(new Widget); |
1710 Widget::InitParams params = | 1681 Widget::InitParams params = |
1711 CreateParams(views::Widget::InitParams::TYPE_POPUP); | 1682 CreateParams(views::Widget::InitParams::TYPE_POPUP); |
1712 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; | 1683 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; |
1713 widget->Init(params); | 1684 widget->Init(params); |
1714 EXPECT_TRUE(delegate->on_before_init_called()); | 1685 EXPECT_TRUE(delegate->on_before_init_called()); |
1715 EXPECT_TRUE(delegate->is_top_level()); | 1686 EXPECT_TRUE(delegate->is_top_level()); |
1716 } | 1687 } |
1717 | 1688 |
1718 // A scumbag View that deletes its owning widget OnMousePressed. | 1689 TEST_F(WidgetTest, WidgetDeleted_InOnMousePressed) { |
1719 class WidgetDeleterView : public View { | |
1720 public: | |
1721 WidgetDeleterView() : View() {} | |
1722 | |
1723 // Overridden from View. | |
1724 virtual bool OnMousePressed(const ui::MouseEvent& event) OVERRIDE { | |
1725 delete GetWidget(); | |
1726 return true; | |
1727 } | |
1728 | |
1729 private: | |
1730 DISALLOW_COPY_AND_ASSIGN(WidgetDeleterView); | |
1731 }; | |
1732 | |
1733 TEST_F(WidgetTest, TestWidgetDeletedInOnMousePressed) { | |
1734 Widget* widget = new Widget; | 1690 Widget* widget = new Widget; |
1735 Widget::InitParams params = | 1691 Widget::InitParams params = |
1736 CreateParams(views::Widget::InitParams::TYPE_POPUP); | 1692 CreateParams(views::Widget::InitParams::TYPE_POPUP); |
1737 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; | |
1738 widget->Init(params); | 1693 widget->Init(params); |
1739 | 1694 |
1740 widget->SetContentsView(new WidgetDeleterView); | 1695 widget->SetContentsView(new CloseWidgetView(ui::ET_MOUSE_PRESSED)); |
1741 | 1696 |
1742 widget->SetSize(gfx::Size(100, 100)); | 1697 widget->SetSize(gfx::Size(100, 100)); |
1743 widget->Show(); | 1698 widget->Show(); |
1744 | 1699 |
1745 gfx::Point click_location(45, 15); | 1700 aura::test::EventGenerator generator(GetContext(), widget->GetNativeWindow()); |
1746 ui::MouseEvent press(ui::ET_MOUSE_PRESSED, click_location, click_location, | 1701 |
1747 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); | 1702 WidgetDeletionObserver deletion_observer(widget); |
1748 widget->OnMouseEvent(&press); | 1703 generator.ClickLeftButton(); |
| 1704 EXPECT_FALSE(deletion_observer.IsWidgetAlive()); |
1749 | 1705 |
1750 // Yay we did not crash! | 1706 // Yay we did not crash! |
1751 } | 1707 } |
| 1708 |
| 1709 TEST_F(WidgetTest, WidgetDeleted_InDispatchGestureEvent) { |
| 1710 Widget* widget = new Widget; |
| 1711 Widget::InitParams params = |
| 1712 CreateParams(views::Widget::InitParams::TYPE_POPUP); |
| 1713 widget->Init(params); |
| 1714 |
| 1715 widget->SetContentsView(new CloseWidgetView(ui::ET_GESTURE_TAP_DOWN)); |
| 1716 |
| 1717 widget->SetSize(gfx::Size(100, 100)); |
| 1718 widget->Show(); |
| 1719 |
| 1720 aura::test::EventGenerator generator(GetContext()); |
| 1721 |
| 1722 WidgetDeletionObserver deletion_observer(widget); |
| 1723 generator.GestureTapAt(widget->GetWindowBoundsInScreen().CenterPoint()); |
| 1724 EXPECT_FALSE(deletion_observer.IsWidgetAlive()); |
| 1725 |
| 1726 // Yay we did not crash! |
| 1727 } |
1752 | 1728 |
1753 // See description of RunGetNativeThemeFromDestructor() for details. | 1729 // See description of RunGetNativeThemeFromDestructor() for details. |
1754 class GetNativeThemeFromDestructorView : public WidgetDelegateView { | 1730 class GetNativeThemeFromDestructorView : public WidgetDelegateView { |
1755 public: | 1731 public: |
1756 GetNativeThemeFromDestructorView() {} | 1732 GetNativeThemeFromDestructorView() {} |
1757 virtual ~GetNativeThemeFromDestructorView() { | 1733 virtual ~GetNativeThemeFromDestructorView() { |
1758 VerifyNativeTheme(); | 1734 VerifyNativeTheme(); |
1759 } | 1735 } |
1760 | 1736 |
1761 virtual View* GetContentsView() OVERRIDE { | 1737 virtual View* GetContentsView() OVERRIDE { |
(...skipping 664 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2426 child_widget.Init(child_params); | 2402 child_widget.Init(child_params); |
2427 child_widget.AddObserver(&observer); | 2403 child_widget.AddObserver(&observer); |
2428 child_widget.Show(); | 2404 child_widget.Show(); |
2429 | 2405 |
2430 parent_widget.CloseNow(); | 2406 parent_widget.CloseNow(); |
2431 } | 2407 } |
2432 #endif | 2408 #endif |
2433 | 2409 |
2434 } // namespace test | 2410 } // namespace test |
2435 } // namespace views | 2411 } // namespace views |
OLD | NEW |