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" | |
21 #include "ui/base/hit_test.h" | 15 #include "ui/base/hit_test.h" |
22 #include "ui/events/event_utils.h" | 16 #include "ui/events/event_utils.h" |
23 #include "ui/gfx/native_widget_types.h" | 17 #include "ui/gfx/native_widget_types.h" |
24 #include "ui/gfx/point.h" | 18 #include "ui/gfx/point.h" |
25 #include "ui/views/bubble/bubble_delegate.h" | 19 #include "ui/views/bubble/bubble_delegate.h" |
26 #include "ui/views/controls/textfield/textfield.h" | 20 #include "ui/views/controls/textfield/textfield.h" |
27 #include "ui/views/test/test_views_delegate.h" | 21 #include "ui/views/test/test_views_delegate.h" |
28 #include "ui/views/test/widget_test.h" | 22 #include "ui/views/test/widget_test.h" |
29 #include "ui/views/views_delegate.h" | 23 #include "ui/views/views_delegate.h" |
30 #include "ui/views/widget/native_widget_aura.h" | |
31 #include "ui/views/widget/native_widget_delegate.h" | 24 #include "ui/views/widget/native_widget_delegate.h" |
32 #include "ui/views/widget/root_view.h" | 25 #include "ui/views/widget/root_view.h" |
33 #include "ui/views/widget/widget_deletion_observer.h" | |
34 #include "ui/views/window/dialog_delegate.h" | 26 #include "ui/views/window/dialog_delegate.h" |
35 #include "ui/views/window/native_frame_view.h" | 27 #include "ui/views/window/native_frame_view.h" |
36 | 28 |
| 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" |
37 #if !defined(OS_CHROMEOS) | 36 #if !defined(OS_CHROMEOS) |
38 #include "ui/views/widget/desktop_aura/desktop_native_widget_aura.h" | 37 #include "ui/views/widget/desktop_aura/desktop_native_widget_aura.h" |
39 #endif | 38 #endif |
| 39 #elif defined(OS_WIN) |
| 40 #include "ui/views/widget/native_widget_win.h" |
| 41 #endif |
40 | 42 |
41 #if defined(OS_WIN) | 43 #if defined(OS_WIN) |
42 #include "ui/views/win/hwnd_util.h" | 44 #include "ui/views/win/hwnd_util.h" |
43 #endif | 45 #endif |
44 | 46 |
45 namespace views { | 47 namespace views { |
46 namespace test { | 48 namespace test { |
47 | 49 |
48 // A view that keeps track of the events it receives, but consumes no events. | 50 // A view that keeps track of the events it receives, but consumes no events. |
49 class EventCountView : public View { | 51 class EventCountView : public View { |
(...skipping 103 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
153 private: | 155 private: |
154 void RecordEvent(const ui::Event& event) { | 156 void RecordEvent(const ui::Event& event) { |
155 ++event_count_[event.type()]; | 157 ++event_count_[event.type()]; |
156 } | 158 } |
157 | 159 |
158 std::map<ui::EventType, int> event_count_; | 160 std::map<ui::EventType, int> event_count_; |
159 | 161 |
160 DISALLOW_COPY_AND_ASSIGN(EventCountHandler); | 162 DISALLOW_COPY_AND_ASSIGN(EventCountHandler); |
161 }; | 163 }; |
162 | 164 |
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 | |
190 ui::WindowShowState GetWidgetShowState(const Widget* widget) { | 165 ui::WindowShowState GetWidgetShowState(const Widget* widget) { |
191 // Use IsMaximized/IsMinimized/IsFullScreen instead of GetWindowPlacement | 166 // Use IsMaximized/IsMinimized/IsFullScreen instead of GetWindowPlacement |
192 // because the former is implemented on all platforms but the latter is not. | 167 // because the former is implemented on all platforms but the latter is not. |
193 return widget->IsFullscreen() ? ui::SHOW_STATE_FULLSCREEN : | 168 return widget->IsFullscreen() ? ui::SHOW_STATE_FULLSCREEN : |
194 widget->IsMaximized() ? ui::SHOW_STATE_MAXIMIZED : | 169 widget->IsMaximized() ? ui::SHOW_STATE_MAXIMIZED : |
195 widget->IsMinimized() ? ui::SHOW_STATE_MINIMIZED : | 170 widget->IsMinimized() ? ui::SHOW_STATE_MINIMIZED : |
196 widget->IsActive() ? ui::SHOW_STATE_NORMAL : | 171 widget->IsActive() ? ui::SHOW_STATE_NORMAL : |
197 ui::SHOW_STATE_INACTIVE; | 172 ui::SHOW_STATE_INACTIVE; |
198 } | 173 } |
199 | 174 |
(...skipping 423 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
623 widget->IsMouseEventsEnabled(); | 598 widget->IsMouseEventsEnabled(); |
624 widget->SetNativeWindowProperty("xx", widget); | 599 widget->SetNativeWindowProperty("xx", widget); |
625 widget->GetNativeWindowProperty("xx"); | 600 widget->GetNativeWindowProperty("xx"); |
626 widget->GetFocusTraversable(); | 601 widget->GetFocusTraversable(); |
627 widget->GetLayer(); | 602 widget->GetLayer(); |
628 widget->ReorderNativeViews(); | 603 widget->ReorderNativeViews(); |
629 widget->SetCapture(widget->GetRootView()); | 604 widget->SetCapture(widget->GetRootView()); |
630 widget->ReleaseCapture(); | 605 widget->ReleaseCapture(); |
631 widget->HasCapture(); | 606 widget->HasCapture(); |
632 widget->GetWorkAreaBoundsInScreen(); | 607 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(); |
633 } | 614 } |
634 | 615 |
635 private: | 616 private: |
636 DISALLOW_COPY_AND_ASSIGN(WidgetWithDestroyedNativeViewTest); | 617 DISALLOW_COPY_AND_ASSIGN(WidgetWithDestroyedNativeViewTest); |
637 }; | 618 }; |
638 | 619 |
639 TEST_F(WidgetWithDestroyedNativeViewTest, Test) { | 620 TEST_F(WidgetWithDestroyedNativeViewTest, Test) { |
640 { | 621 { |
641 Widget widget; | 622 Widget widget; |
642 Widget::InitParams params = CreateParams(Widget::InitParams::TYPE_POPUP); | 623 Widget::InitParams params = CreateParams(Widget::InitParams::TYPE_POPUP); |
(...skipping 638 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1281 EXPECT_FALSE(widget.GetNativeView()->IsVisible()); | 1262 EXPECT_FALSE(widget.GetNativeView()->IsVisible()); |
1282 widget.Show(); | 1263 widget.Show(); |
1283 EXPECT_TRUE(widget.GetNativeView()->IsVisible()); | 1264 EXPECT_TRUE(widget.GetNativeView()->IsVisible()); |
1284 } | 1265 } |
1285 | 1266 |
1286 // The following code verifies we can correctly destroy a Widget from a mouse | 1267 // The following code verifies we can correctly destroy a Widget from a mouse |
1287 // enter/exit. We could test move/drag/enter/exit but in general we don't run | 1268 // enter/exit. We could test move/drag/enter/exit but in general we don't run |
1288 // nested message loops from such events, nor has the code ever really dealt | 1269 // nested message loops from such events, nor has the code ever really dealt |
1289 // with this situation. | 1270 // with this situation. |
1290 | 1271 |
| 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 |
1291 // Generates two moves (first generates enter, second real move), a press, drag | 1320 // Generates two moves (first generates enter, second real move), a press, drag |
1292 // and release stopping at |last_event_type|. | 1321 // and release stopping at |last_event_type|. |
1293 void GenerateMouseEvents(Widget* widget, ui::EventType last_event_type) { | 1322 void GenerateMouseEvents(Widget* widget, ui::EventType last_event_type) { |
1294 const gfx::Rect screen_bounds(widget->GetWindowBoundsInScreen()); | 1323 const gfx::Rect screen_bounds(widget->GetWindowBoundsInScreen()); |
1295 ui::MouseEvent move_event(ui::ET_MOUSE_MOVED, screen_bounds.CenterPoint(), | 1324 ui::MouseEvent move_event(ui::ET_MOUSE_MOVED, screen_bounds.CenterPoint(), |
1296 screen_bounds.CenterPoint(), 0, 0); | 1325 screen_bounds.CenterPoint(), 0, 0); |
1297 aura::WindowEventDispatcher* dispatcher = | 1326 aura::WindowEventDispatcher* dispatcher = |
1298 widget->GetNativeWindow()->GetDispatcher(); | 1327 widget->GetNativeWindow()->GetDispatcher(); |
1299 ui::EventDispatchDetails details = dispatcher->OnEventFromSource(&move_event); | 1328 ui::EventDispatchDetails details = dispatcher->OnEventFromSource(&move_event); |
1300 if (last_event_type == ui::ET_MOUSE_ENTERED || details.dispatcher_destroyed) | 1329 if (last_event_type == ui::ET_MOUSE_ENTERED || details.dispatcher_destroyed) |
(...skipping 378 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1679 set_views_delegate(delegate); // ViewsTestBase takes ownership. | 1708 set_views_delegate(delegate); // ViewsTestBase takes ownership. |
1680 scoped_ptr<Widget> widget(new Widget); | 1709 scoped_ptr<Widget> widget(new Widget); |
1681 Widget::InitParams params = | 1710 Widget::InitParams params = |
1682 CreateParams(views::Widget::InitParams::TYPE_POPUP); | 1711 CreateParams(views::Widget::InitParams::TYPE_POPUP); |
1683 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; | 1712 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; |
1684 widget->Init(params); | 1713 widget->Init(params); |
1685 EXPECT_TRUE(delegate->on_before_init_called()); | 1714 EXPECT_TRUE(delegate->on_before_init_called()); |
1686 EXPECT_TRUE(delegate->is_top_level()); | 1715 EXPECT_TRUE(delegate->is_top_level()); |
1687 } | 1716 } |
1688 | 1717 |
1689 TEST_F(WidgetTest, WidgetDeleted_InOnMousePressed) { | 1718 // A scumbag View that deletes its owning widget OnMousePressed. |
| 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) { |
1690 Widget* widget = new Widget; | 1734 Widget* widget = new Widget; |
1691 Widget::InitParams params = | 1735 Widget::InitParams params = |
1692 CreateParams(views::Widget::InitParams::TYPE_POPUP); | 1736 CreateParams(views::Widget::InitParams::TYPE_POPUP); |
| 1737 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; |
1693 widget->Init(params); | 1738 widget->Init(params); |
1694 | 1739 |
1695 widget->SetContentsView(new CloseWidgetView(ui::ET_MOUSE_PRESSED)); | 1740 widget->SetContentsView(new WidgetDeleterView); |
1696 | 1741 |
1697 widget->SetSize(gfx::Size(100, 100)); | 1742 widget->SetSize(gfx::Size(100, 100)); |
1698 widget->Show(); | 1743 widget->Show(); |
1699 | 1744 |
1700 aura::test::EventGenerator generator(GetContext(), widget->GetNativeWindow()); | 1745 gfx::Point click_location(45, 15); |
1701 | 1746 ui::MouseEvent press(ui::ET_MOUSE_PRESSED, click_location, click_location, |
1702 WidgetDeletionObserver deletion_observer(widget); | 1747 ui::EF_LEFT_MOUSE_BUTTON, ui::EF_LEFT_MOUSE_BUTTON); |
1703 generator.ClickLeftButton(); | 1748 widget->OnMouseEvent(&press); |
1704 EXPECT_FALSE(deletion_observer.IsWidgetAlive()); | |
1705 | 1749 |
1706 // Yay we did not crash! | 1750 // Yay we did not crash! |
1707 } | 1751 } |
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 } | |
1728 | 1752 |
1729 // See description of RunGetNativeThemeFromDestructor() for details. | 1753 // See description of RunGetNativeThemeFromDestructor() for details. |
1730 class GetNativeThemeFromDestructorView : public WidgetDelegateView { | 1754 class GetNativeThemeFromDestructorView : public WidgetDelegateView { |
1731 public: | 1755 public: |
1732 GetNativeThemeFromDestructorView() {} | 1756 GetNativeThemeFromDestructorView() {} |
1733 virtual ~GetNativeThemeFromDestructorView() { | 1757 virtual ~GetNativeThemeFromDestructorView() { |
1734 VerifyNativeTheme(); | 1758 VerifyNativeTheme(); |
1735 } | 1759 } |
1736 | 1760 |
1737 virtual View* GetContentsView() OVERRIDE { | 1761 virtual View* GetContentsView() OVERRIDE { |
(...skipping 664 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2402 child_widget.Init(child_params); | 2426 child_widget.Init(child_params); |
2403 child_widget.AddObserver(&observer); | 2427 child_widget.AddObserver(&observer); |
2404 child_widget.Show(); | 2428 child_widget.Show(); |
2405 | 2429 |
2406 parent_widget.CloseNow(); | 2430 parent_widget.CloseNow(); |
2407 } | 2431 } |
2408 #endif | 2432 #endif |
2409 | 2433 |
2410 } // namespace test | 2434 } // namespace test |
2411 } // namespace views | 2435 } // namespace views |
OLD | NEW |