2 * Copyright (C) 2004 John Ellis
3 * Copyright (C) 2008 - 2016 The Geeqie Team
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
29 *-----------------------------------------------------------------------------
31 *-----------------------------------------------------------------------------
35 * @brief Add accelerator key to a window popup menu
40 * This is used only so that the user can see the applicable
41 * shortcut key displayed in the menu. The actual handling of
42 * the keystroke is done elsewhere in the code.
44 static void menu_item_add_accelerator(GtkWidget *menu, GtkAccelGroup *accel_group, hard_coded_window_keys *window_keys)
48 gchar **label_stripped;
51 label = g_strdup(gtk_menu_item_get_label(GTK_MENU_ITEM(menu)));
53 pango_parse_markup(label, -1, '_', nullptr, &label_text, nullptr, nullptr);
55 label_stripped = g_strsplit(label_text, "...", 2);
57 while (window_keys[i].text != nullptr)
59 if (g_strcmp0(window_keys[i].text, label_stripped[0]) == 0)
61 gtk_widget_add_accelerator(menu, "activate", accel_group, window_keys[i].key_value, window_keys[i].mask, GTK_ACCEL_VISIBLE);
70 g_strfreev(label_stripped);
74 * @brief Callback for the actions GList sort function
79 * Sort the action entries so that the non-shifted and non-control
80 * entries are at the start of the list. The user then sees the basic
81 * non-modified key shortcuts displayed in the menus.
83 static gint actions_sort_cb(gconstpointer a, gconstpointer b)
85 const gchar *accel_path_a;
87 const gchar *accel_path_b;
90 accel_path_a = gtk_action_get_accel_path(GTK_ACTION(a));
91 accel_path_b = gtk_action_get_accel_path(GTK_ACTION(b));
93 if (accel_path_a && gtk_accel_map_lookup_entry(accel_path_a, &key_a) && accel_path_b && gtk_accel_map_lookup_entry(accel_path_b, &key_b))
95 if (key_a.accel_mods < key_b.accel_mods) return -1;
96 if (key_a.accel_mods > key_b.accel_mods) return 1;
103 * @brief Add accelerator key to main window popup menu
107 * This is used only so that the user can see the applicable
108 * shortcut key displayed in the menu. The actual handling of
109 * the keystroke is done elsewhere in the code.
111 static void menu_item_add_main_window_accelerator(GtkWidget *menu, GtkAccelGroup *accel_group)
114 gchar *menu_label_text;
116 gchar *action_label_text;
121 const gchar *accel_path;
124 menu_label = g_strdup(gtk_menu_item_get_label(GTK_MENU_ITEM(menu)));
126 pango_parse_markup(menu_label, -1, '_', nullptr, &menu_label_text, nullptr, nullptr);
128 lw = static_cast<LayoutWindow *>(layout_window_list->data); /* get the actions from the first window, it should not matter, they should be the same in all windows */
130 g_assert(lw && lw->ui_manager);
131 groups = gtk_ui_manager_get_action_groups(lw->ui_manager);
135 actions = gtk_action_group_list_actions(GTK_ACTION_GROUP(groups->data));
136 actions = g_list_sort(actions, actions_sort_cb);
140 action = GTK_ACTION(actions->data);
141 accel_path = gtk_action_get_accel_path(action);
142 if (accel_path && gtk_accel_map_lookup_entry(accel_path, &key))
144 g_object_get(action, "label", &action_label, NULL);
146 pango_parse_markup(action_label, -1, '_', nullptr, &action_label_text, nullptr, nullptr);
148 if (g_strcmp0(action_label_text, menu_label_text) == 0)
150 if (key.accel_key != 0)
152 gtk_widget_add_accelerator(menu, "activate", accel_group, key.accel_key, key.accel_mods, GTK_ACCEL_VISIBLE);
157 g_free(action_label_text);
158 g_free(action_label);
160 actions = actions->next;
162 groups = groups->next;
166 g_free(menu_label_text);
169 static void menu_item_finish(GtkWidget *menu, GtkWidget *item, GCallback func, gpointer data)
171 if (func) g_signal_connect(G_OBJECT(item), "activate", func, data);
172 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
173 gtk_widget_show(item);
176 GtkWidget *menu_item_add(GtkWidget *menu, const gchar *label,
177 GCallback func, gpointer data)
180 GtkAccelGroup *accel_group;
181 hard_coded_window_keys *window_keys;
183 item = gtk_menu_item_new_with_mnemonic(label);
184 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
185 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
187 if (accel_group && window_keys)
189 menu_item_add_accelerator(item, accel_group, window_keys);
191 else if (accel_group)
193 menu_item_add_main_window_accelerator(item, accel_group);
196 menu_item_finish(menu, item, func, data);
201 GtkWidget *menu_item_add_stock(GtkWidget *menu, const gchar *label, const gchar *stock_id,
202 GCallback func, gpointer data)
206 GtkAccelGroup *accel_group;
207 hard_coded_window_keys *window_keys;
209 item = gtk_image_menu_item_new_with_mnemonic(label);
210 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
211 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
213 image = gtk_image_new_from_stock(stock_id, GTK_ICON_SIZE_MENU);
214 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), image);
216 if (accel_group && window_keys)
218 menu_item_add_accelerator(item, accel_group, window_keys);
220 else if (accel_group)
222 menu_item_add_main_window_accelerator(item, accel_group);
225 gtk_widget_show(image);
226 menu_item_finish(menu, item, func, data);
231 GtkWidget *menu_item_add_icon(GtkWidget *menu, const gchar *label, const gchar *icon_name,
232 GCallback func, gpointer data)
236 GtkAccelGroup *accel_group;
237 hard_coded_window_keys *window_keys;
239 item = gtk_image_menu_item_new_with_mnemonic(label);
240 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
241 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
243 image = gtk_image_new_from_icon_name(icon_name, GTK_ICON_SIZE_MENU);
244 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), image);
246 if (accel_group && window_keys)
248 menu_item_add_accelerator(item, accel_group, window_keys);
250 else if (accel_group)
252 menu_item_add_main_window_accelerator(item, accel_group);
255 gtk_widget_show(image);
256 menu_item_finish(menu, item, func, data);
261 GtkWidget *menu_item_add_sensitive(GtkWidget *menu, const gchar *label, gboolean sensitive,
262 GCallback func, gpointer data)
265 GtkAccelGroup *accel_group;
266 hard_coded_window_keys *window_keys;
268 item = menu_item_add(menu, label, func, data);
269 gtk_widget_set_sensitive(item, sensitive);
270 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
271 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
272 if (accel_group && window_keys)
274 menu_item_add_accelerator(item, accel_group, window_keys);
276 else if (accel_group)
278 menu_item_add_main_window_accelerator(item, accel_group);
284 GtkWidget *menu_item_add_stock_sensitive(GtkWidget *menu, const gchar *label, const gchar *stock_id, gboolean sensitive,
285 GCallback func, gpointer data)
288 GtkAccelGroup *accel_group;
289 hard_coded_window_keys *window_keys;
291 item = menu_item_add_stock(menu, label, stock_id, func, data);
292 gtk_widget_set_sensitive(item, sensitive);
293 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
294 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
295 if (accel_group && window_keys)
297 menu_item_add_accelerator(item, accel_group, window_keys);
299 else if (accel_group)
301 menu_item_add_main_window_accelerator(item, accel_group);
307 GtkWidget *menu_item_add_icon_sensitive(GtkWidget *menu, const gchar *label, const gchar *icon_name, gboolean sensitive,
308 GCallback func, gpointer data)
311 GtkAccelGroup *accel_group;
312 hard_coded_window_keys *window_keys;
314 item = menu_item_add_icon(menu, label, icon_name, func, data);
315 gtk_widget_set_sensitive(item, sensitive);
316 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
317 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
318 if (accel_group && window_keys)
320 menu_item_add_accelerator(item, accel_group, window_keys);
322 else if (accel_group)
324 menu_item_add_main_window_accelerator(item, accel_group);
330 GtkWidget *menu_item_add_check(GtkWidget *menu, const gchar *label, gboolean active,
331 GCallback func, gpointer data)
334 GtkAccelGroup *accel_group;
335 hard_coded_window_keys *window_keys;
337 item = gtk_check_menu_item_new_with_mnemonic(label);
338 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
339 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
341 if (accel_group && window_keys)
343 menu_item_add_accelerator(item, accel_group, window_keys);
345 else if (accel_group)
347 menu_item_add_main_window_accelerator(item, accel_group);
350 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(item), active);
351 menu_item_finish(menu, item, func, data);
356 GtkWidget *menu_item_add_radio(GtkWidget *menu, const gchar *label, gpointer item_data, gboolean active,
357 GCallback func, gpointer data)
359 GtkAccelGroup *accel_group;
360 hard_coded_window_keys *window_keys;
362 GtkWidget *item = menu_item_add_check(menu, label, active, func, data);
363 g_object_set_data(G_OBJECT(item), "menu_item_radio_data", item_data);
364 g_object_set(G_OBJECT(item), "draw-as-radio", TRUE, NULL);
366 window_keys = static_cast<hard_coded_window_keys *>(g_object_get_data(G_OBJECT(menu), "window_keys"));
367 accel_group = static_cast<GtkAccelGroup *>(g_object_get_data(G_OBJECT(menu), "accel_group"));
368 if (accel_group && window_keys)
370 menu_item_add_accelerator(item, accel_group, window_keys);
372 else if (accel_group)
374 menu_item_add_main_window_accelerator(item, accel_group);
380 void menu_item_add_divider(GtkWidget *menu)
382 GtkWidget *item = gtk_separator_menu_item_new();
383 gtk_widget_set_sensitive(item, FALSE);
384 gtk_menu_shell_append(GTK_MENU_SHELL(menu),item);
385 gtk_widget_show(item);
388 GtkWidget *menu_item_add_simple(GtkWidget *menu, const gchar *label,
389 GCallback func, gpointer data)
391 GtkWidget *item = gtk_menu_item_new_with_label(label);
392 menu_item_finish(menu, item, func, data);
398 *-----------------------------------------------------------------------------
400 *-----------------------------------------------------------------------------
403 static void popup_menu_short_lived_cb(GtkWidget *, gpointer data)
405 /* destroy the menu */
406 g_object_unref(G_OBJECT(data));
409 GtkWidget *popup_menu_short_lived()
413 menu = gtk_menu_new();
415 /* take ownership of menu */
416 #ifdef GTK_OBJECT_FLOATING
418 g_object_ref(G_OBJECT(menu));
419 gtk_object_sink(GTK_OBJECT(menu));
422 g_object_ref_sink(G_OBJECT(menu));
425 g_signal_connect(G_OBJECT(menu), "selection_done",
426 G_CALLBACK(popup_menu_short_lived_cb), menu);
430 gboolean popup_menu_position_clamp(GtkMenu *menu, gint *x, gint *y, gint height)
432 gboolean adjusted = FALSE;
437 GtkRequisition requisition;
439 gtk_widget_get_requisition(GTK_WIDGET(menu), &requisition);
440 w = requisition.width;
441 h = requisition.height;
442 xw = gdk_screen_width();
443 xh = gdk_screen_height();
454 *y = MAX(0, *y - h - height);
476 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */