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

Side by Side Diff: views/controls/menu/native_menu_gtk.cc

Issue 521063: Fixes possible crash in menus on views/gtk. The crash would happen if... (Closed) Base URL: svn://chrome-svn/chrome/trunk/src/
Patch Set: Created 10 years, 11 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
« no previous file with comments | « views/controls/menu/native_menu_gtk.h ('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) 2009 The Chromium Authors. All rights reserved. Use of this 1 // Copyright (c) 2009 The Chromium Authors. All rights reserved. Use of this
2 // source code is governed by a BSD-style license that can be found in the 2 // source code is governed by a BSD-style license that can be found in the
3 // LICENSE file. 3 // LICENSE file.
4 4
5 #include "views/controls/menu/native_menu_gtk.h" 5 #include "views/controls/menu/native_menu_gtk.h"
6 6
7 #include <map> 7 #include <map>
8 #include <string> 8 #include <string>
9 9
10 #include "app/gfx/gtk_util.h" 10 #include "app/gfx/gtk_util.h"
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after
59 59
60 namespace views { 60 namespace views {
61 61
62 //////////////////////////////////////////////////////////////////////////////// 62 ////////////////////////////////////////////////////////////////////////////////
63 // NativeMenuGtk, public: 63 // NativeMenuGtk, public:
64 64
65 NativeMenuGtk::NativeMenuGtk(menus::MenuModel* model) 65 NativeMenuGtk::NativeMenuGtk(menus::MenuModel* model)
66 : model_(model), 66 : model_(model),
67 menu_(NULL), 67 menu_(NULL),
68 menu_shown_(false), 68 menu_shown_(false),
69 suppress_activate_signal_(false) { 69 suppress_activate_signal_(false),
70 menu_activated_(false),
71 activated_index_(-1) {
70 } 72 }
71 73
72 NativeMenuGtk::~NativeMenuGtk() { 74 NativeMenuGtk::~NativeMenuGtk() {
73 gtk_widget_destroy(menu_); 75 gtk_widget_destroy(menu_);
74 } 76 }
75 77
76 //////////////////////////////////////////////////////////////////////////////// 78 ////////////////////////////////////////////////////////////////////////////////
77 // NativeMenuGtk, MenuWrapper implementation: 79 // NativeMenuGtk, MenuWrapper implementation:
78 80
79 void NativeMenuGtk::RunMenuAt(const gfx::Point& point, int alignment) { 81 void NativeMenuGtk::RunMenuAt(const gfx::Point& point, int alignment) {
82 menu_activated_ = false;
83
80 UpdateStates(); 84 UpdateStates();
81 Position position = { point, static_cast<Menu2::Alignment>(alignment) }; 85 Position position = { point, static_cast<Menu2::Alignment>(alignment) };
82 // TODO(beng): value of '1' will not work for context menus! 86 // TODO(beng): value of '1' will not work for context menus!
83 gtk_menu_popup(GTK_MENU(menu_), NULL, NULL, MenuPositionFunc, &position, 1, 87 gtk_menu_popup(GTK_MENU(menu_), NULL, NULL, MenuPositionFunc, &position, 1,
84 gtk_get_current_event_time()); 88 gtk_get_current_event_time());
85 89
86 DCHECK(!menu_shown_); 90 DCHECK(!menu_shown_);
87 menu_shown_ = true; 91 menu_shown_ = true;
88 // Listen for "hide" signal so that we know when to return from the blocking 92 // Listen for "hide" signal so that we know when to return from the blocking
89 // RunMenuAt call. 93 // RunMenuAt call.
90 gint handle_id = 94 gint handle_id =
91 g_signal_connect(G_OBJECT(menu_), "hide", G_CALLBACK(OnMenuHidden), this); 95 g_signal_connect(G_OBJECT(menu_), "hide", G_CALLBACK(OnMenuHidden), this);
92 96
93 // Block until menu is no longer shown by running a nested message loop. 97 // Block until menu is no longer shown by running a nested message loop.
94 MessageLoopForUI::current()->Run(NULL); 98 MessageLoopForUI::current()->Run(NULL);
95 99
96 g_signal_handler_disconnect(G_OBJECT(menu_), handle_id); 100 g_signal_handler_disconnect(G_OBJECT(menu_), handle_id);
97 menu_shown_ = false; 101 menu_shown_ = false;
102
103 // Call into the model after the nested message loop quits. This way if the
104 // model ends up deleting us, or MessageLoop::Quit takes a while, there aren't
105 // any problems.
106 if (menu_activated_ && model_->IsEnabledAt(activated_index_) &&
107 MenuTypeCanExecute(model_->GetTypeAt(activated_index_))) {
108 model_->ActivatedAt(activated_index_);
109 }
98 } 110 }
99 111
100 void NativeMenuGtk::CancelMenu() { 112 void NativeMenuGtk::CancelMenu() {
101 NOTIMPLEMENTED(); 113 NOTIMPLEMENTED();
102 } 114 }
103 115
104 void NativeMenuGtk::Rebuild() { 116 void NativeMenuGtk::Rebuild() {
105 ResetMenu(); 117 ResetMenu();
106 118
107 std::map<int, GtkRadioMenuItem*> radio_groups_; 119 std::map<int, GtkRadioMenuItem*> radio_groups_;
(...skipping 178 matching lines...) Expand 10 before | Expand all | Expand 10 after
286 // Ignore the signal if it's sent to an inactive checked radio item. 298 // Ignore the signal if it's sent to an inactive checked radio item.
287 // 299 //
288 // Suppose there are three radio items A, B, C, and A is now being 300 // Suppose there are three radio items A, B, C, and A is now being
289 // checked. If you click C, "activate" signal will be sent to A and C. 301 // checked. If you click C, "activate" signal will be sent to A and C.
290 // Here, we ignore the signal sent to A. 302 // Here, we ignore the signal sent to A.
291 if (GTK_IS_RADIO_MENU_ITEM(menu_item) && 303 if (GTK_IS_RADIO_MENU_ITEM(menu_item) &&
292 !gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menu_item))) { 304 !gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menu_item))) {
293 return; 305 return;
294 } 306 }
295 307
296 if (model_->IsEnabledAt(position) && 308 menu_activated_ = true;
297 MenuTypeCanExecute(model_->GetTypeAt(position))) { 309 activated_index_ = position;
298 model_->ActivatedAt(position);
299 }
300 } 310 }
301 311
302 // static 312 // static
303 void NativeMenuGtk::CallActivate(GtkMenuItem* menu_item, 313 void NativeMenuGtk::CallActivate(GtkMenuItem* menu_item,
304 NativeMenuGtk* native_menu) { 314 NativeMenuGtk* native_menu) {
305 native_menu->OnActivate(menu_item); 315 native_menu->OnActivate(menu_item);
306 } 316 }
307 317
308 //////////////////////////////////////////////////////////////////////////////// 318 ////////////////////////////////////////////////////////////////////////////////
309 // MenuWrapper, public: 319 // MenuWrapper, public:
310 320
311 // static 321 // static
312 MenuWrapper* MenuWrapper::CreateWrapper(Menu2* menu) { 322 MenuWrapper* MenuWrapper::CreateWrapper(Menu2* menu) {
313 return new NativeMenuGtk(menu->model()); 323 return new NativeMenuGtk(menu->model());
314 } 324 }
315 325
316 } // namespace views 326 } // namespace views
OLDNEW
« no previous file with comments | « views/controls/menu/native_menu_gtk.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698