2 * Copyright (C) 2004 John Ellis
3 * Copyright (C) 2008 - 2017 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.
26 #include <gdk-pixbuf/gdk-pixbuf.h>
27 #include <glib-object.h>
35 #include "layout-util.h"
37 #include "main-defines.h"
38 #include "ui-fileops.h"
42 /** Implements the user-definable toolbar function
43 * Called from the Preferences/toolbar tab
50 GtkWidget *add_button;
55 struct ToolbarButtonData
58 GtkWidget *button_label;
61 const gchar *name; /* GtkActionEntry terminology */
62 const gchar *stock_id;
65 static ToolbarData *toolbarlist[2];
67 struct UseableToolbarItems
69 const gchar *name; /* GtkActionEntry terminology */
71 const gchar *stock_id;
76 * @param widget Not used
77 * @param data Pointer to vbox list item
78 * @param up Up/Down movement
79 * @param single_step Move up/down one step, or to top/bottom
82 static void toolbar_item_move(GtkWidget *, gpointer data, gboolean up, gboolean single_step)
84 auto list_item = static_cast<GtkWidget *>(data);
88 if (!list_item) return;
89 box = gtk_widget_get_ancestor(list_item, GTK_TYPE_BOX);
92 gtk_container_child_get(GTK_CONTAINER(box), list_item, "position", &pos, NULL);
96 pos = up ? (pos - 1) : (pos + 1);
104 gtk_box_reorder_child(GTK_BOX(box), list_item, pos);
107 static void toolbar_item_move_up_cb(GtkWidget *widget, gpointer data)
109 toolbar_item_move(widget, data, TRUE, TRUE);
112 static void toolbar_item_move_down_cb(GtkWidget *widget, gpointer data)
114 toolbar_item_move(widget, data, FALSE, TRUE);
117 static void toolbar_item_move_top_cb(GtkWidget *widget, gpointer data)
119 toolbar_item_move(widget, data, TRUE, FALSE);
122 static void toolbar_item_move_bottom_cb(GtkWidget *widget, gpointer data)
124 toolbar_item_move(widget, data, FALSE, FALSE);
127 static void toolbar_item_delete_cb(GtkWidget *, gpointer data)
129 gtk_container_remove(GTK_CONTAINER(gtk_widget_get_parent(GTK_WIDGET(data))), GTK_WIDGET(data));
132 static void toolbar_menu_popup(GtkWidget *widget)
136 menu = popup_menu_short_lived();
140 menu_item_add_icon(menu, _("Move to _top"), GQ_ICON_GO_TOP, G_CALLBACK(toolbar_item_move_top_cb), widget);
141 menu_item_add_icon(menu, _("Move _up"), GQ_ICON_GO_UP, G_CALLBACK(toolbar_item_move_up_cb), widget);
142 menu_item_add_icon(menu, _("Move _down"), GQ_ICON_GO_DOWN, G_CALLBACK(toolbar_item_move_down_cb), widget);
143 menu_item_add_icon(menu, _("Move to _bottom"), GQ_ICON_GO_BOTTOM, G_CALLBACK(toolbar_item_move_bottom_cb), widget);
144 menu_item_add_divider(menu);
145 menu_item_add_icon(menu, _("Remove"), GQ_ICON_DELETE, G_CALLBACK(toolbar_item_delete_cb), widget);
146 menu_item_add_divider(menu);
149 gtk_menu_popup_at_pointer(GTK_MENU(menu), nullptr);
152 static gboolean toolbar_press_cb(GtkGesture *, int, double, double, gpointer data)
154 auto button_data = static_cast<ToolbarButtonData *>(data);
156 toolbar_menu_popup(button_data->button);
161 static void get_toolbar_item(const gchar *name, gchar **label, gchar **stock_id)
163 ActionItem *action_item;
169 list = get_action_items();
174 action_item = static_cast<ActionItem *>(work->data);
175 if (g_strcmp0(action_item->name, name) == 0)
177 *label = g_strdup(action_item->label);
178 *stock_id = g_strdup(action_item->icon_name);
185 action_items_free(list);
188 static void toolbar_item_free(ToolbarButtonData *tbbd)
192 g_free(const_cast<gchar *>(tbbd->name));
193 g_free(const_cast<gchar *>(tbbd->stock_id));
194 g_free(const_cast<ToolbarButtonData *>(tbbd));
197 static void toolbar_button_free(GtkWidget *widget)
199 g_free(g_object_get_data(G_OBJECT(widget), "toolbar_add_name"));
200 g_free(g_object_get_data(G_OBJECT(widget), "toolbar_add_label"));
201 g_free(g_object_get_data(G_OBJECT(widget), "toolbar_add_stock_id"));
204 static void toolbarlist_add_button(const gchar *name, const gchar *label,
205 const gchar *stock_id, GtkBox *box)
207 ToolbarButtonData *toolbar_entry;
211 toolbar_entry = g_new(ToolbarButtonData,1);
212 toolbar_entry->button = gtk_button_new();
213 gtk_button_set_relief(GTK_BUTTON(toolbar_entry->button), GTK_RELIEF_NONE);
214 gq_gtk_box_pack_start(GTK_BOX(box), toolbar_entry->button, FALSE, FALSE, 0);
215 gtk_widget_show(toolbar_entry->button);
217 g_object_set_data_full(G_OBJECT(toolbar_entry->button), "toolbarbuttondata",
218 toolbar_entry, reinterpret_cast<GDestroyNotify>(toolbar_item_free));
220 hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, PREF_PAD_BUTTON_GAP);
221 gq_gtk_container_add(GTK_WIDGET(toolbar_entry->button), hbox);
222 gtk_widget_show(hbox);
224 toolbar_entry->button_label = gtk_label_new(label);
225 toolbar_entry->name = g_strdup(name);
226 toolbar_entry->stock_id = g_strdup(stock_id);
229 gesture = gtk_gesture_click_new();
230 gtk_widget_add_controller(toolbar_entry->button, GTK_EVENT_CONTROLLER(gesture));
232 gesture = gtk_gesture_multi_press_new(toolbar_entry->button);
234 gtk_gesture_single_set_button(GTK_GESTURE_SINGLE(gesture), MOUSE_BUTTON_RIGHT);
235 g_signal_connect(gesture, "released", G_CALLBACK(toolbar_press_cb), toolbar_entry);
237 if (toolbar_entry->stock_id)
241 iconl = path_from_utf8(toolbar_entry->stock_id);
242 pixbuf = gdk_pixbuf_new_from_file(iconl, nullptr);
251 gtk_icon_size_lookup(GTK_ICON_SIZE_BUTTON, &w, &h);
253 scaled = gdk_pixbuf_scale_simple(pixbuf, w, h,
254 GDK_INTERP_BILINEAR);
255 toolbar_entry->image = gtk_image_new_from_pixbuf(scaled);
257 g_object_unref(scaled);
258 g_object_unref(pixbuf);
262 toolbar_entry->image = gtk_image_new_from_stock(toolbar_entry->stock_id,
263 GTK_ICON_SIZE_BUTTON);
268 toolbar_entry->image = gtk_image_new_from_icon_name(GQ_ICON_GO_JUMP,
269 GTK_ICON_SIZE_BUTTON);
271 gq_gtk_box_pack_start(GTK_BOX(hbox), toolbar_entry->image, FALSE, FALSE, 0);
272 gtk_widget_show(toolbar_entry->image);
273 gq_gtk_box_pack_start(GTK_BOX(hbox), toolbar_entry->button_label, FALSE, FALSE, 0);
274 gtk_widget_show(toolbar_entry->button_label);
277 static void toolbarlist_add_cb(GtkWidget *widget, gpointer data)
279 auto name = static_cast<const gchar *>(g_object_get_data(G_OBJECT(widget), "toolbar_add_name"));
280 auto label = static_cast<const gchar *>(g_object_get_data(G_OBJECT(widget), "toolbar_add_label"));
281 auto stock_id = static_cast<const gchar *>(g_object_get_data(G_OBJECT(widget), "toolbar_add_stock_id"));
282 auto tbbd = static_cast<ToolbarData *>(data);
284 toolbarlist_add_button(name, label, stock_id, GTK_BOX(tbbd->vbox));
287 static void get_desktop_data(const gchar *name, gchar **label, gchar **stock_id)
294 editors_list = editor_list_get();
298 auto editor = static_cast<const EditorDescription *>(work->data);
300 if (g_strcmp0(name, editor->key) == 0)
302 *label = g_strdup(editor->name);
303 *stock_id = g_strconcat(editor->icon, ".desktop", NULL);
308 g_list_free(editors_list);
311 static void toolbar_menu_add_popup(GtkWidget *, gpointer data)
313 ActionItem *action_item;
314 auto toolbarlist = static_cast<ToolbarData *>(data);
320 menu = popup_menu_short_lived();
322 item = menu_item_add_stock(menu, "Separator", "Separator", G_CALLBACK(toolbarlist_add_cb), toolbarlist);
323 g_object_set_data(G_OBJECT(item), "toolbar_add_name", g_strdup("Separator"));
324 g_object_set_data(G_OBJECT(item), "toolbar_add_label", g_strdup("Separator"));
325 g_object_set_data(G_OBJECT(item), "toolbar_add_stock_id", g_strdup("no-icon"));
326 g_signal_connect(G_OBJECT(item), "destroy", G_CALLBACK(toolbar_button_free), item);
328 list = get_action_items();
333 action_item = static_cast<ActionItem *>(work->data);
335 item = menu_item_add_stock(menu, action_item->label, action_item->icon_name, G_CALLBACK(toolbarlist_add_cb), toolbarlist);
336 g_object_set_data(G_OBJECT(item), "toolbar_add_name", g_strdup(action_item->name));
337 g_object_set_data(G_OBJECT(item), "toolbar_add_label", g_strdup(action_item->label));
338 g_object_set_data(G_OBJECT(item), "toolbar_add_stock_id", g_strdup(action_item->icon_name));
339 g_signal_connect(G_OBJECT(item), "destroy", G_CALLBACK(toolbar_button_free), item);
344 action_items_free(list);
346 gtk_menu_popup_at_pointer(GTK_MENU(menu), nullptr);
349 static gboolean toolbar_menu_add_cb(GtkWidget *widget, gpointer data)
351 auto toolbarlist = static_cast<ToolbarData *>(data);
353 toolbar_menu_add_popup(widget, toolbarlist);
358 * @brief For each layoutwindow, clear toolbar and reload with current selection
359 * @param bar Main or Status toolbar
362 void toolbar_apply(ToolbarType bar)
368 work_windows = layout_window_list;
371 lw = static_cast<LayoutWindow *>(work_windows->data);
373 layout_toolbar_clear(lw, bar);
375 work_toolbar = gtk_container_get_children(GTK_CONTAINER(toolbarlist[bar]->vbox));
378 auto button = static_cast<GtkButton *>(work_toolbar->data);
379 ToolbarButtonData *tbbd;
381 tbbd = static_cast<ToolbarButtonData *>(g_object_get_data(G_OBJECT(button),"toolbarbuttondata"));
382 layout_toolbar_add(lw, bar, tbbd->name);
384 work_toolbar = work_toolbar->next;
386 g_list_free(work_toolbar);
388 work_windows = work_windows->next;
394 * @brief Load the current toolbar items into the vbox
396 * @param box The vbox displayed in the preferences Toolbar tab
397 * @param bar Main or Status toolbar
399 * Get the current contents of the toolbar, both menu items
400 * and desktop items, and load them into the vbox
402 static void toolbarlist_populate(LayoutWindow *lw, GtkBox *box, ToolbarType bar)
404 GList *work = g_list_first(lw->toolbar_actions[bar]);
408 auto name = static_cast<gchar *>(work->data);
413 if (file_extension_match(name, ".desktop"))
415 get_desktop_data(name, &label, &icon);
419 get_toolbar_item(name, &label, &icon);
422 if (g_strcmp0(name, "Separator") != 0)
424 toolbarlist_add_button(name, label, icon, box);
428 toolbarlist_add_button(name, name, "no-icon", box);
433 GtkWidget *toolbar_select_new(LayoutWindow *lw, ToolbarType bar)
439 if (!lw) return nullptr;
441 if (!toolbarlist[bar])
443 toolbarlist[bar] = g_new0(ToolbarData, 1);
445 toolbarlist[bar]->lw = lw;
447 toolbarlist[bar]->widget = gtk_box_new(GTK_ORIENTATION_VERTICAL, PREF_PAD_GAP);
448 gtk_widget_show(toolbarlist[bar]->widget);
450 scrolled = gq_gtk_scrolled_window_new(nullptr, nullptr);
451 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled),
452 GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
453 gq_gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolled), GTK_SHADOW_NONE);
454 gq_gtk_box_pack_start(GTK_BOX(toolbarlist[bar]->widget), scrolled, TRUE, TRUE, 0);
455 gtk_widget_show(scrolled);
457 toolbarlist[bar]->vbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
458 gtk_widget_show(toolbarlist[bar]->vbox);
459 gq_gtk_container_add(GTK_WIDGET(scrolled), toolbarlist[bar]->vbox);
460 gtk_viewport_set_shadow_type(GTK_VIEWPORT(gtk_bin_get_child(GTK_BIN(scrolled))),
463 add_box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
464 gtk_widget_show(add_box);
465 gq_gtk_box_pack_end(GTK_BOX(toolbarlist[bar]->widget), add_box, FALSE, FALSE, 0);
466 tbar = pref_toolbar_new(add_box);
467 toolbarlist[bar]->add_button = pref_toolbar_button(tbar, GQ_ICON_ADD, _("Add"), FALSE,
468 _("Add Toolbar Item"),
469 G_CALLBACK(toolbar_menu_add_cb), toolbarlist[bar]);
470 gtk_widget_show(toolbarlist[bar]->add_button);
472 toolbarlist_populate(lw,GTK_BOX(toolbarlist[bar]->vbox), bar);
474 return toolbarlist[bar]->widget;
477 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */