| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  | // Copyright 2013 The Chromium Authors. All rights reserved.
 | 
					
						
							|  |  |  | // Use of this source code is governed by a BSD-style license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "shell/browser/ui/gtk/menu_util.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <gdk/gdk.h>
 | 
					
						
							|  |  |  | #include <gtk/gtk.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-05 05:18:31 -06:00
										 |  |  | #include "base/containers/flat_map.h"
 | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  | #include "base/strings/utf_string_conversions.h"
 | 
					
						
							|  |  |  | #include "chrome/app/chrome_command_ids.h"
 | 
					
						
							|  |  |  | #include "shell/browser/ui/gtk_util.h"
 | 
					
						
							|  |  |  | #include "third_party/skia/include/core/SkBitmap.h"
 | 
					
						
							|  |  |  | #include "ui/base/accelerators/accelerator.h"
 | 
					
						
							|  |  |  | #include "ui/base/accelerators/menu_label_accelerator_util_linux.h"
 | 
					
						
							|  |  |  | #include "ui/base/models/image_model.h"
 | 
					
						
							|  |  |  | #include "ui/base/models/menu_model.h"
 | 
					
						
							|  |  |  | #include "ui/events/event_constants.h"
 | 
					
						
							|  |  |  | #include "ui/events/keycodes/keyboard_code_conversion_x.h"
 | 
					
						
							| 
									
										
										
										
											2022-03-28 18:48:50 +02:00
										 |  |  | #include "ui/ozone/public/ozone_platform.h"
 | 
					
						
							| 
									
										
										
										
											2020-10-20 20:24:52 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 12:55:47 -07:00
										 |  |  | namespace electron::gtkui { | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int EventFlagsFromGdkState(guint state) { | 
					
						
							|  |  |  |   int flags = ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= (state & GDK_SHIFT_MASK) ? ui::EF_SHIFT_DOWN : ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= (state & GDK_LOCK_MASK) ? ui::EF_CAPS_LOCK_ON : ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= (state & GDK_CONTROL_MASK) ? ui::EF_CONTROL_DOWN : ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= (state & GDK_MOD1_MASK) ? ui::EF_ALT_DOWN : ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= (state & GDK_BUTTON1_MASK) ? ui::EF_LEFT_MOUSE_BUTTON : ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= | 
					
						
							|  |  |  |       (state & GDK_BUTTON2_MASK) ? ui::EF_MIDDLE_MOUSE_BUTTON : ui::EF_NONE; | 
					
						
							|  |  |  |   flags |= (state & GDK_BUTTON3_MASK) ? ui::EF_RIGHT_MOUSE_BUTTON : ui::EF_NONE; | 
					
						
							|  |  |  |   return flags; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | guint GetGdkKeyCodeForAccelerator(const ui::Accelerator& accelerator) { | 
					
						
							|  |  |  |   // The second parameter is false because accelerator keys are expressed in
 | 
					
						
							|  |  |  |   // terms of the non-shift-modified key.
 | 
					
						
							|  |  |  |   return XKeysymForWindowsKeyCode(accelerator.key_code(), false); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GdkModifierType GetGdkModifierForAccelerator( | 
					
						
							|  |  |  |     const ui::Accelerator& accelerator) { | 
					
						
							|  |  |  |   int event_flag = accelerator.modifiers(); | 
					
						
							|  |  |  |   int modifier = 0; | 
					
						
							|  |  |  |   if (event_flag & ui::EF_SHIFT_DOWN) | 
					
						
							|  |  |  |     modifier |= GDK_SHIFT_MASK; | 
					
						
							|  |  |  |   if (event_flag & ui::EF_CONTROL_DOWN) | 
					
						
							|  |  |  |     modifier |= GDK_CONTROL_MASK; | 
					
						
							|  |  |  |   if (event_flag & ui::EF_ALT_DOWN) | 
					
						
							|  |  |  |     modifier |= GDK_MOD1_MASK; | 
					
						
							|  |  |  |   return static_cast<GdkModifierType>(modifier); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-16 11:05:26 -05:00
										 |  |  | GtkWidget* BuildMenuItemWithImage(const std::string& label, | 
					
						
							|  |  |  |                                   const gfx::Image& icon) { | 
					
						
							|  |  |  | // GTK4 removed support for menuitem icons.
 | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  | #if GTK_CHECK_VERSION(3, 90, 0)
 | 
					
						
							|  |  |  |   return gtk_menu_item_new_with_mnemonic(label.c_str()); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   G_GNUC_BEGIN_IGNORE_DEPRECATIONS; | 
					
						
							|  |  |  |   GtkWidget* menu_item = gtk_image_menu_item_new_with_mnemonic(label.c_str()); | 
					
						
							| 
									
										
										
										
											2025-05-16 11:05:26 -05:00
										 |  |  |   GdkPixbuf* pixbuf = gtk_util::GdkPixbufFromSkBitmap(*icon.ToSkBitmap()); | 
					
						
							|  |  |  |   GtkWidget* image = gtk_image_new_from_pixbuf(pixbuf); | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |   gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menu_item), image); | 
					
						
							| 
									
										
										
										
											2025-05-16 11:05:26 -05:00
										 |  |  |   g_object_unref(pixbuf); | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |   G_GNUC_END_IGNORE_DEPRECATIONS; | 
					
						
							|  |  |  |   return menu_item; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GtkWidget* BuildMenuItemWithLabel(const std::string& label) { | 
					
						
							|  |  |  |   return gtk_menu_item_new_with_mnemonic(label.c_str()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ui::MenuModel* ModelForMenuItem(GtkMenuItem* menu_item) { | 
					
						
							|  |  |  |   return reinterpret_cast<ui::MenuModel*>( | 
					
						
							|  |  |  |       g_object_get_data(G_OBJECT(menu_item), "model")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GtkWidget* AppendMenuItemToMenu(int index, | 
					
						
							|  |  |  |                                 ui::MenuModel* model, | 
					
						
							|  |  |  |                                 GtkWidget* menu_item, | 
					
						
							|  |  |  |                                 GtkWidget* menu, | 
					
						
							|  |  |  |                                 bool connect_to_activate, | 
					
						
							| 
									
										
										
										
											2023-10-05 19:59:39 -04:00
										 |  |  |                                 MenuActivatedCallback item_activated_cb, | 
					
						
							|  |  |  |                                 std::vector<ScopedGSignal>* signals) { | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |   // Set the ID of a menu item.
 | 
					
						
							|  |  |  |   // Add 1 to the menu_id to avoid setting zero (null) to "menu-id".
 | 
					
						
							|  |  |  |   g_object_set_data(G_OBJECT(menu_item), "menu-id", GINT_TO_POINTER(index + 1)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Native menu items do their own thing, so only selectively listen for the
 | 
					
						
							|  |  |  |   // activate signal.
 | 
					
						
							|  |  |  |   if (connect_to_activate) { | 
					
						
							| 
									
										
										
										
											2023-10-05 19:59:39 -04:00
										 |  |  |     signals->emplace_back(menu_item, "activate", item_activated_cb); | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // AppendMenuItemToMenu is used both internally when we control menu creation
 | 
					
						
							|  |  |  |   // from a model (where the model can choose to hide certain menu items), and
 | 
					
						
							|  |  |  |   // with immediate commands which don't provide the option.
 | 
					
						
							|  |  |  |   if (model) { | 
					
						
							|  |  |  |     if (model->IsVisibleAt(index)) | 
					
						
							|  |  |  |       gtk_widget_show(menu_item); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     gtk_widget_show(menu_item); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   gtk_menu_shell_append(GTK_MENU_SHELL(menu), menu_item); | 
					
						
							|  |  |  |   return menu_item; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GetMenuItemID(GtkWidget* menu_item, int* menu_id) { | 
					
						
							|  |  |  |   gpointer id_ptr = g_object_get_data(G_OBJECT(menu_item), "menu-id"); | 
					
						
							|  |  |  |   if (id_ptr != nullptr) { | 
					
						
							|  |  |  |     *menu_id = GPOINTER_TO_INT(id_ptr) - 1; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void ExecuteCommand(ui::MenuModel* model, int id) { | 
					
						
							|  |  |  |   GdkEvent* event = gtk_get_current_event(); | 
					
						
							|  |  |  |   int event_flags = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (event && event->type == GDK_BUTTON_RELEASE) | 
					
						
							|  |  |  |     event_flags = EventFlagsFromGdkState(event->button.state); | 
					
						
							| 
									
										
										
										
											2022-08-17 11:35:53 -07:00
										 |  |  |   model->ActivatedAt(static_cast<int>(id), event_flags); | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (event) | 
					
						
							|  |  |  |     gdk_event_free(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void BuildSubmenuFromModel(ui::MenuModel* model, | 
					
						
							|  |  |  |                            GtkWidget* menu, | 
					
						
							| 
									
										
										
										
											2023-10-05 19:59:39 -04:00
										 |  |  |                            MenuActivatedCallback item_activated_cb, | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |                            bool* block_activation, | 
					
						
							| 
									
										
										
										
											2023-10-05 19:59:39 -04:00
										 |  |  |                            std::vector<ScopedGSignal>* signals) { | 
					
						
							| 
									
										
										
										
											2024-01-05 05:18:31 -06:00
										 |  |  |   base::flat_map<int, GtkWidget*> radio_groups; | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |   GtkWidget* menu_item = nullptr; | 
					
						
							| 
									
										
										
										
											2022-08-17 11:35:53 -07:00
										 |  |  |   for (size_t i = 0; i < model->GetItemCount(); ++i) { | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |     std::string label = ui::ConvertAcceleratorsFromWindowsStyle( | 
					
						
							|  |  |  |         base::UTF16ToUTF8(model->GetLabelAt(i))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool connect_to_activate = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (model->GetTypeAt(i)) { | 
					
						
							|  |  |  |       case ui::MenuModel::TYPE_SEPARATOR: | 
					
						
							|  |  |  |         menu_item = gtk_separator_menu_item_new(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case ui::MenuModel::TYPE_CHECK: | 
					
						
							|  |  |  |         menu_item = gtk_check_menu_item_new_with_mnemonic(label.c_str()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case ui::MenuModel::TYPE_RADIO: { | 
					
						
							|  |  |  |         auto iter = radio_groups.find(model->GetGroupIdAt(i)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (iter == radio_groups.end()) { | 
					
						
							|  |  |  |           menu_item = | 
					
						
							|  |  |  |               gtk_radio_menu_item_new_with_mnemonic(nullptr, label.c_str()); | 
					
						
							|  |  |  |           radio_groups[model->GetGroupIdAt(i)] = menu_item; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |           menu_item = gtk_radio_menu_item_new_with_mnemonic_from_widget( | 
					
						
							|  |  |  |               GTK_RADIO_MENU_ITEM(iter->second), label.c_str()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case ui::MenuModel::TYPE_BUTTON_ITEM: { | 
					
						
							|  |  |  |         NOTIMPLEMENTED(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case ui::MenuModel::TYPE_SUBMENU: | 
					
						
							|  |  |  |       case ui::MenuModel::TYPE_COMMAND: { | 
					
						
							|  |  |  |         auto icon_model = model->GetIconAt(i); | 
					
						
							|  |  |  |         if (!icon_model.IsEmpty()) | 
					
						
							|  |  |  |           menu_item = BuildMenuItemWithImage(label, icon_model.GetImage()); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           menu_item = BuildMenuItemWithLabel(label); | 
					
						
							|  |  |  | #if !GTK_CHECK_VERSION(3, 90, 0)
 | 
					
						
							|  |  |  |         G_GNUC_BEGIN_IGNORE_DEPRECATIONS; | 
					
						
							|  |  |  |         if (GTK_IS_IMAGE_MENU_ITEM(menu_item)) { | 
					
						
							|  |  |  |           gtk_image_menu_item_set_always_show_image( | 
					
						
							|  |  |  |               GTK_IMAGE_MENU_ITEM(menu_item), TRUE); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         G_GNUC_END_IGNORE_DEPRECATIONS; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         NOTREACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (model->GetTypeAt(i) == ui::MenuModel::TYPE_SUBMENU) { | 
					
						
							|  |  |  |       GtkWidget* submenu = gtk_menu_new(); | 
					
						
							|  |  |  |       ui::MenuModel* submenu_model = model->GetSubmenuModelAt(i); | 
					
						
							|  |  |  |       BuildSubmenuFromModel(submenu_model, submenu, item_activated_cb, | 
					
						
							| 
									
										
										
										
											2023-10-05 19:59:39 -04:00
										 |  |  |                             block_activation, signals); | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |       gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), submenu); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Update all the menu item info in the newly-generated menu.
 | 
					
						
							|  |  |  |       gtk_container_foreach(GTK_CONTAINER(submenu), SetMenuItemInfo, | 
					
						
							|  |  |  |                             block_activation); | 
					
						
							|  |  |  |       submenu_model->MenuWillShow(); | 
					
						
							|  |  |  |       connect_to_activate = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-28 18:48:50 +02:00
										 |  |  |     if (ui::OzonePlatform::GetInstance() | 
					
						
							|  |  |  |             ->GetPlatformProperties() | 
					
						
							|  |  |  |             .electron_can_call_x11) { | 
					
						
							|  |  |  |       ui::Accelerator accelerator; | 
					
						
							|  |  |  |       if (model->GetAcceleratorAt(i, &accelerator)) { | 
					
						
							|  |  |  |         gtk_widget_add_accelerator(menu_item, "activate", nullptr, | 
					
						
							|  |  |  |                                    GetGdkKeyCodeForAccelerator(accelerator), | 
					
						
							|  |  |  |                                    GetGdkModifierForAccelerator(accelerator), | 
					
						
							|  |  |  |                                    GTK_ACCEL_VISIBLE); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     g_object_set_data(G_OBJECT(menu_item), "model", model); | 
					
						
							|  |  |  |     AppendMenuItemToMenu(i, model, menu_item, menu, connect_to_activate, | 
					
						
							| 
									
										
										
										
											2023-10-05 19:59:39 -04:00
										 |  |  |                          item_activated_cb, signals); | 
					
						
							| 
									
										
										
										
											2020-06-02 17:00:20 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     menu_item = nullptr; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SetMenuItemInfo(GtkWidget* widget, void* block_activation_ptr) { | 
					
						
							|  |  |  |   if (GTK_IS_SEPARATOR_MENU_ITEM(widget)) { | 
					
						
							|  |  |  |     // We need to explicitly handle this case because otherwise we'll ask the
 | 
					
						
							|  |  |  |     // menu delegate about something with an invalid id.
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int id; | 
					
						
							|  |  |  |   if (!GetMenuItemID(widget, &id)) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ui::MenuModel* model = ModelForMenuItem(GTK_MENU_ITEM(widget)); | 
					
						
							|  |  |  |   if (!model) { | 
					
						
							|  |  |  |     // If we're not providing the sub menu, then there's no model.  For
 | 
					
						
							|  |  |  |     // example, the IME submenu doesn't have a model.
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   bool* block_activation = static_cast<bool*>(block_activation_ptr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (GTK_IS_CHECK_MENU_ITEM(widget)) { | 
					
						
							|  |  |  |     GtkCheckMenuItem* item = GTK_CHECK_MENU_ITEM(widget); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // gtk_check_menu_item_set_active() will send the activate signal. Touching
 | 
					
						
							|  |  |  |     // the underlying "active" property will also call the "activate" handler
 | 
					
						
							|  |  |  |     // for this menu item. So we prevent the "activate" handler from
 | 
					
						
							|  |  |  |     // being called while we set the checkbox.
 | 
					
						
							|  |  |  |     // Why not use one of the glib signal-blocking functions?  Because when we
 | 
					
						
							|  |  |  |     // toggle a radio button, it will deactivate one of the other radio buttons,
 | 
					
						
							|  |  |  |     // which we don't have a pointer to.
 | 
					
						
							|  |  |  |     *block_activation = true; | 
					
						
							|  |  |  |     gtk_check_menu_item_set_active(item, model->IsItemCheckedAt(id)); | 
					
						
							|  |  |  |     *block_activation = false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (GTK_IS_MENU_ITEM(widget)) { | 
					
						
							|  |  |  |     gtk_widget_set_sensitive(widget, model->IsEnabledAt(id)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (model->IsVisibleAt(id)) { | 
					
						
							|  |  |  |       // Update the menu item label if it is dynamic.
 | 
					
						
							|  |  |  |       if (model->IsItemDynamicAt(id)) { | 
					
						
							|  |  |  |         std::string label = ui::ConvertAcceleratorsFromWindowsStyle( | 
					
						
							|  |  |  |             base::UTF16ToUTF8(model->GetLabelAt(id))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         gtk_menu_item_set_label(GTK_MENU_ITEM(widget), label.c_str()); | 
					
						
							|  |  |  | #if !GTK_CHECK_VERSION(3, 90, 0)
 | 
					
						
							|  |  |  |         G_GNUC_BEGIN_IGNORE_DEPRECATIONS; | 
					
						
							|  |  |  |         if (GTK_IS_IMAGE_MENU_ITEM(widget)) { | 
					
						
							|  |  |  |           auto icon_model = model->GetIconAt(id); | 
					
						
							|  |  |  |           if (!icon_model.IsEmpty()) { | 
					
						
							|  |  |  |             GdkPixbuf* pixbuf = gtk_util::GdkPixbufFromSkBitmap( | 
					
						
							|  |  |  |                 *icon_model.GetImage().ToSkBitmap()); | 
					
						
							|  |  |  |             gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(widget), | 
					
						
							|  |  |  |                                           gtk_image_new_from_pixbuf(pixbuf)); | 
					
						
							|  |  |  |             g_object_unref(pixbuf); | 
					
						
							|  |  |  |           } else { | 
					
						
							|  |  |  |             gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(widget), nullptr); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         G_GNUC_END_IGNORE_DEPRECATIONS; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       gtk_widget_show(widget); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       gtk_widget_hide(widget); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     GtkWidget* submenu = gtk_menu_item_get_submenu(GTK_MENU_ITEM(widget)); | 
					
						
							|  |  |  |     if (submenu) { | 
					
						
							|  |  |  |       gtk_container_foreach(GTK_CONTAINER(submenu), &SetMenuItemInfo, | 
					
						
							|  |  |  |                             block_activation_ptr); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-29 12:55:47 -07:00
										 |  |  | }  // namespace electron::gtkui
 |