Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(182)

Side by Side Diff: ui/views/widget/widget_unittest.cc

Issue 169443005: Fix crash which occurs when a widget destroys itself as a result of ET_GESTURE_TAP_DOWN (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« ui/views/widget/root_view.cc ('K') | « ui/views/widget/root_view.cc ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
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
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
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
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
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
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
OLDNEW
« ui/views/widget/root_view.cc ('K') | « ui/views/widget/root_view.cc ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698