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.
26 #include "collect-io.h"
28 #include "history-list.h"
30 #include "layout-image.h"
33 #include "ui-bookmark.h"
34 #include "ui-fileops.h"
41 *-------------------------------------------------------------------
43 *-------------------------------------------------------------------
53 GtkWidget *dialog_name_entry;
56 SortActionType action;
59 SortSelectionType selection;
61 GtkWidget *folder_group;
62 GtkWidget *collection_group;
64 GtkWidget *add_button;
65 GtkWidget *help_button;
66 GtkWidget *undo_button;
67 SortActionType undo_action;
69 GList *undo_dest_list;
70 gchar *undo_collection;
74 #define SORT_KEY_FOLDERS "sort_manager"
75 #define SORT_KEY_COLLECTIONS "sort_manager_collections"
78 static void bar_sort_undo_set(SortData *sd, GList *src_list, const gchar *dest);
79 static void bar_sort_add_close(SortData *sd);
82 static void bar_sort_collection_list_build(GtkWidget *bookmarks)
88 history_list_free_key(SORT_KEY_COLLECTIONS);
89 bookmark_list_set_key(bookmarks, SORT_KEY_COLLECTIONS);
91 dir_fd = file_data_new_dir(get_collections_dir());
92 filelist_read(dir_fd, &list, nullptr);
93 file_data_unref(dir_fd);
95 list = filelist_sort_path(list);
103 fd = static_cast<FileData *>(work->data);
106 if (file_extension_match(fd->path, GQ_COLLECTION_EXT))
108 name = remove_extension_from_path(fd->name);
112 name = g_strdup(fd->name);
114 bookmark_list_add(bookmarks, name, fd->path);
121 static void bar_sort_mode_sync(SortData *sd, SortModeType mode)
123 gboolean folder_mode;
125 if (sd->mode == mode) return;
128 folder_mode = (sd->mode == BAR_SORT_MODE_FOLDER);
130 bookmark_list_set_no_defaults(sd->bookmarks, !folder_mode);
131 bookmark_list_set_editable(sd->bookmarks, folder_mode);
132 bookmark_list_set_only_directories(sd->bookmarks, folder_mode);
136 gtk_widget_hide(sd->collection_group);
137 gtk_widget_show(sd->folder_group);
138 bookmark_list_set_key(sd->bookmarks, SORT_KEY_FOLDERS);
142 gtk_widget_hide(sd->folder_group);
143 gtk_widget_show(sd->collection_group);
144 bar_sort_collection_list_build(sd->bookmarks);
147 bar_sort_add_close(sd);
149 bar_sort_undo_set(sd, nullptr, nullptr);
152 static void bar_sort_mode_cb(GtkWidget *combo, gpointer data)
154 auto sd = static_cast<SortData *>(data);
156 if (gtk_combo_box_get_active(GTK_COMBO_BOX(combo)) == BAR_SORT_MODE_FOLDER)
158 bar_sort_mode_sync(sd, BAR_SORT_MODE_FOLDER);
162 bar_sort_mode_sync(sd, BAR_SORT_MODE_COLLECTION);
166 /* this takes control of src_list */
167 static void bar_sort_undo_set(SortData *sd, GList *src_list, const gchar *dest)
169 g_list_free_full(sd->undo_src_list, g_free);
170 sd->undo_src_list = filelist_to_path_list(src_list);
174 /* we should create the undo_dest_list to use it later... */
175 g_list_free_full(sd->undo_dest_list, g_free);
176 sd->undo_dest_list=nullptr;
178 GList *work = sd->undo_src_list;
181 gchar *filename = g_strdup(filename_from_path(static_cast<const gchar *>(work->data)));
182 gchar *dest_path = g_build_filename(g_strdup(dest), filename, NULL);
183 sd->undo_dest_list = g_list_prepend(sd->undo_dest_list, g_strdup(dest_path));
186 sd->undo_dest_list = g_list_reverse(sd->undo_dest_list);
189 sd->undo_action = sd->action;
193 gtk_widget_set_sensitive(sd->undo_button,
194 ((sd->undo_src_list ) && sd->undo_dest_list));
198 static void bar_sort_undo_folder(SortData *sd, GtkWidget *button)
202 if (!(sd->undo_src_list && sd->undo_dest_list)) return;
204 switch (sd->undo_action)
212 if (sd->undo_src_list)
214 GList *work = nullptr;
216 src_path = g_strdup(static_cast<const gchar *>(sd->undo_src_list->data));
217 src_dir = remove_level_from_path(src_path);
218 list = sd->undo_dest_list;
221 work = g_list_prepend(work, file_data_new_group(static_cast<const gchar *>(list->data)));
224 file_util_move_simple(work, src_dir, sd->lw->window);
232 case BAR_SORT_FILTER:
233 if (sd->undo_src_list)
236 GList *work = nullptr;
238 delete_list = sd->undo_dest_list;
241 work = g_list_append(work, file_data_new_group(static_cast<const gchar *>(delete_list->data)));
242 delete_list = delete_list->next;
244 options->file_ops.safe_delete_enable = TRUE;
245 file_util_delete(nullptr, work, button);
253 layout_refresh(sd->lw);
254 origin = static_cast<gchar *>((sd->undo_src_list)->data);
258 layout_image_set_fd(sd->lw, file_data_new_group(origin));
261 bar_sort_undo_set(sd, nullptr, nullptr);
264 static void bar_sort_undo_collection(SortData *sd)
268 work = sd->undo_src_list;
272 source = static_cast<gchar *>(work->data);
274 collect_manager_remove(file_data_new_group(source), sd->undo_collection);
277 bar_sort_undo_set(sd, nullptr, nullptr);
280 static void bar_sort_undo_cb(GtkWidget *button, gpointer data)
282 auto sd = static_cast<SortData *>(data);
284 if (sd->mode == BAR_SORT_MODE_FOLDER)
286 bar_sort_undo_folder(sd, button);
290 bar_sort_undo_collection(sd);
294 static void bar_sort_bookmark_select_folder(SortData *sd, FileData *UNUSED(source), const gchar *path)
298 GList *undo_src_list;
300 if (!isdir(path)) return;
302 orig_list = layout_selection_list(sd->lw);
303 action_list = orig_list;
304 undo_src_list = orig_list;
307 bar_sort_undo_set(sd, undo_src_list, path);
312 file_util_copy_simple(action_list, path, sd->lw->window);
313 action_list = nullptr;
314 layout_image_next(sd->lw);
318 file_util_move_simple(action_list, path, sd->lw->window);
319 action_list = nullptr;
322 case BAR_SORT_FILTER:
323 file_util_start_filter_from_filelist(sd->filter_key, action_list, path, sd->lw->window);
324 layout_image_next(sd->lw);
332 static void bar_sort_bookmark_select_collection(SortData *sd, FileData *source, const gchar *path)
334 GList *list = nullptr;
336 switch (sd->selection)
338 case BAR_SORT_SELECTION_IMAGE:
339 list = g_list_append(nullptr, file_data_ref(source));
341 case BAR_SORT_SELECTION_SELECTED:
342 list = layout_selection_list(sd->lw);
350 bar_sort_undo_set(sd, nullptr, nullptr);
354 bar_sort_undo_set(sd, list, path);
355 sd->undo_collection = g_strdup(path);
361 image_fd = static_cast<FileData *>(list->data);
363 collect_manager_add(image_fd, path);
367 static void bar_sort_bookmark_select(const gchar *path, gpointer data)
369 auto sd = static_cast<SortData *>(data);
372 source = layout_image_get_fd(sd->lw);
373 if (!path || !source) return;
375 if (sd->mode == BAR_SORT_MODE_FOLDER)
377 bar_sort_bookmark_select_folder(sd, source, path);
381 bar_sort_bookmark_select_collection(sd, source, path);
385 static void bar_sort_set_action(SortData *sd, SortActionType action, const gchar *filter_key)
388 if (action == BAR_SORT_FILTER)
390 if (!filter_key) filter_key = "";
391 sd->filter_key = g_strdup(filter_key);
395 sd->filter_key = nullptr;
399 static void bar_sort_set_copy_cb(GtkWidget *button, gpointer data)
401 auto sd = static_cast<SortData *>(data);
402 if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button))) return;
403 bar_sort_set_action(sd, BAR_SORT_COPY, nullptr);
406 static void bar_sort_set_move_cb(GtkWidget *button, gpointer data)
408 auto sd = static_cast<SortData *>(data);
409 if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button))) return;
410 bar_sort_set_action(sd, BAR_SORT_MOVE, nullptr);
413 static void bar_sort_set_filter_cb(GtkWidget *button, gpointer data)
415 auto sd = static_cast<SortData *>(data);
418 if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button))) return;
419 key = static_cast<const gchar *>(g_object_get_data(G_OBJECT(button), "filter_key"));
420 bar_sort_set_action(sd, BAR_SORT_FILTER, key);
423 static void bar_filter_help_cb(GenericDialog *UNUSED(gd), gpointer UNUSED(data))
425 help_window_show("GuidePluginsConfig.html#Geeqieextensions");
428 static void bar_filter_help_dialog()
432 gd = generic_dialog_new(_("Sort Manager Operations"),
433 "sort_manager_operations", nullptr, TRUE, nullptr, nullptr);
434 generic_dialog_add_message(gd, GTK_STOCK_DIALOG_INFO,
435 "Sort Manager Operations", _("Additional operations utilising plugins\nmay be included by setting:\n\nX-Geeqie-Filter=true\n\nin the plugin file."), TRUE);
436 generic_dialog_add_button(gd, GTK_STOCK_HELP, nullptr, bar_filter_help_cb, TRUE);
437 generic_dialog_add_button(gd, GTK_STOCK_OK, nullptr, nullptr, TRUE);
439 gtk_widget_show(gd->dialog);
442 static gboolean bar_filter_message_cb(GtkWidget *UNUSED(widget), GdkEventButton *event, gpointer UNUSED(data))
444 if (event->button != MOUSE_BUTTON_RIGHT) return FALSE;
446 bar_filter_help_dialog();
451 static void bar_sort_help_cb(gpointer UNUSED(data))
453 bar_filter_help_dialog();
456 static void bar_sort_set_selection(SortData *sd, SortSelectionType selection)
458 sd->selection = selection;
461 static void bar_sort_set_selection_image_cb(GtkWidget *button, gpointer data)
463 auto sd = static_cast<SortData *>(data);
464 if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button))) return;
465 bar_sort_set_selection(sd, BAR_SORT_SELECTION_IMAGE);
468 static void bar_sort_set_selection_selected_cb(GtkWidget *button, gpointer data)
470 auto sd = static_cast<SortData *>(data);
471 if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button))) return;
472 bar_sort_set_selection(sd, BAR_SORT_SELECTION_SELECTED);
475 static void bar_sort_add_close(SortData *sd)
477 if (sd->dialog) file_dialog_close(sd->dialog);
478 sd->dialog_name_entry = nullptr;
479 sd->dialog = nullptr;
482 static void bar_sort_add_ok_cb(FileDialog *fd, gpointer data)
484 auto sd = static_cast<SortData *>(data);
485 const gchar *name = gtk_entry_get_text(GTK_ENTRY(sd->dialog_name_entry));
486 gboolean empty_name = (name[0] == '\0');
488 name = gtk_entry_get_text(GTK_ENTRY(sd->dialog_name_entry));
489 if (sd->mode == BAR_SORT_MODE_FOLDER)
493 name = filename_from_path(fd->dest_path);
496 bookmark_list_add(sd->bookmarks, name, fd->dest_path);
501 gboolean has_extension;
502 auto filename = const_cast<gchar *>(name);
504 if (empty_name) return;
506 has_extension = file_extension_match(name, GQ_COLLECTION_EXT);
509 filename = g_strconcat(name, GQ_COLLECTION_EXT, NULL);
512 path = g_build_filename(get_collections_dir(), filename, NULL);
515 gchar *text = g_strdup_printf(_("The collection:\n%s\nalready exists."), filename);
516 file_util_warning_dialog(_("Collection exists"), text, GTK_STOCK_DIALOG_INFO, nullptr);
523 cd = collection_new(path);
524 if (collection_save(cd, path))
526 bar_sort_collection_list_build(sd->bookmarks);
530 gchar *text = g_strdup_printf(_("Failed to save the collection:\n%s"), path);
531 file_util_warning_dialog(_("Save Failed"), text,
532 GTK_STOCK_DIALOG_ERROR, GENERIC_DIALOG(fd)->dialog);
535 collection_unref(cd);
538 if (!has_extension) g_free(filename);
542 bar_sort_add_close(sd);
545 static void bar_sort_add_cancel_cb(FileDialog *UNUSED(fd), gpointer data)
547 auto sd = static_cast<SortData *>(data);
549 bar_sort_add_close(sd);
552 static void bar_sort_add_cb(GtkWidget *button, gpointer data)
554 auto sd = static_cast<SortData *>(data);
560 gtk_window_present(GTK_WINDOW(GENERIC_DIALOG(sd->dialog)->dialog));
564 if (sd->mode == BAR_SORT_MODE_FOLDER)
566 title = _("Add Bookmark");
570 title = _("Add Collection");
573 sd->dialog = file_util_file_dlg(title,
574 "add_bookmark", button,
575 bar_sort_add_cancel_cb, sd);
576 file_dialog_add_button(sd->dialog, GTK_STOCK_OK, nullptr, bar_sort_add_ok_cb, TRUE);
578 generic_dialog_add_message(GENERIC_DIALOG(sd->dialog), nullptr, title, nullptr, FALSE);
580 if (sd->mode == BAR_SORT_MODE_FOLDER)
582 file_dialog_add_path_widgets(sd->dialog, nullptr, nullptr, "add_bookmark", nullptr, nullptr);
585 hbox = pref_box_new(GENERIC_DIALOG(sd->dialog)->vbox, FALSE, GTK_ORIENTATION_HORIZONTAL, PREF_PAD_GAP);
587 pref_label_new(hbox, _("Name:"));
589 sd->dialog_name_entry = gtk_entry_new();
590 gtk_box_pack_start(GTK_BOX(hbox), sd->dialog_name_entry, TRUE, TRUE, 0);
591 generic_dialog_attach_default(GENERIC_DIALOG(sd->dialog), sd->dialog_name_entry);
592 gtk_widget_show(sd->dialog_name_entry);
594 if (sd->mode == BAR_SORT_MODE_COLLECTION)
596 gtk_widget_grab_focus(sd->dialog_name_entry);
599 gtk_widget_show(GENERIC_DIALOG(sd->dialog)->dialog);
602 void bar_sort_close(GtkWidget *bar)
606 sd = static_cast<SortData *>(g_object_get_data(G_OBJECT(bar), "bar_sort_data"));
609 gtk_widget_destroy(sd->vbox);
612 static void bar_sort_destroy(GtkWidget *UNUSED(widget), gpointer data)
614 auto sd = static_cast<SortData *>(data);
616 bar_sort_add_close(sd);
618 g_free(sd->filter_key);
619 g_list_free_full(sd->undo_src_list, g_free);
620 g_list_free_full(sd->undo_dest_list, g_free);
621 g_free(sd->undo_collection);
625 static void bar_sort_edit_button_free(gpointer data)
630 static GtkWidget *bar_sort_new(LayoutWindow *lw, SortActionType action,
631 SortModeType mode, SortSelectionType selection,
632 const gchar *filter_key)
635 GtkWidget *buttongrp;
639 GList *editors_list, *work;
640 gboolean have_filter;
643 if (!lw) return nullptr;
645 sd = g_new0(SortData, 1);
651 if (sd->action == BAR_SORT_FILTER && (!filter_key || !filter_key[0]))
653 sd->action = BAR_SORT_COPY;
656 sd->selection = selection;
657 sd->undo_src_list = nullptr;
658 sd->undo_dest_list = nullptr;
659 sd->undo_collection = nullptr;
661 sd->vbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, PREF_PAD_GAP);
662 DEBUG_NAME(sd->vbox);
663 g_object_set_data(G_OBJECT(sd->vbox), "bar_sort_data", sd);
664 g_signal_connect(G_OBJECT(sd->vbox), "destroy",
665 G_CALLBACK(bar_sort_destroy), sd);
667 label = gtk_label_new(_("Sort Manager"));
668 pref_label_bold(label, TRUE, FALSE);
669 gtk_box_pack_start(GTK_BOX(sd->vbox), label, FALSE, FALSE, 0);
670 gtk_widget_show(label);
672 combo = gtk_combo_box_text_new();
673 gtk_box_pack_start(GTK_BOX(sd->vbox), combo, FALSE, FALSE, 0);
674 gtk_widget_show(combo);
676 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(combo), _("Folders"));
677 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(combo), _("Collections"));
679 g_signal_connect(G_OBJECT(combo), "changed",
680 G_CALLBACK(bar_sort_mode_cb), sd);
682 sd->folder_group = pref_box_new(sd->vbox, FALSE, GTK_ORIENTATION_VERTICAL, 0);
683 DEBUG_NAME(sd->folder_group);
684 gtk_widget_set_tooltip_text(sd->folder_group, _("See the Help file for additional functions"));
686 buttongrp = pref_radiobutton_new(sd->folder_group, nullptr,
687 _("Copy"), (sd->action == BAR_SORT_COPY),
688 G_CALLBACK(bar_sort_set_copy_cb), sd);
689 g_signal_connect(G_OBJECT(buttongrp), "button_press_event", G_CALLBACK(bar_filter_message_cb), NULL);
690 button = pref_radiobutton_new(sd->folder_group, buttongrp,
691 _("Move"), (sd->action == BAR_SORT_MOVE),
692 G_CALLBACK(bar_sort_set_move_cb), sd);
693 g_signal_connect(G_OBJECT(button), "button_press_event", G_CALLBACK(bar_filter_message_cb), NULL);
697 editors_list = editor_list_get();
702 auto editor = static_cast<EditorDescription *>(work->data);
704 gboolean select = FALSE;
708 if (!editor_is_filter(editor->key)) continue;
710 key = g_strdup(editor->key);
711 if (sd->action == BAR_SORT_FILTER && strcmp(key, filter_key) == 0)
713 bar_sort_set_action(sd, sd->action, key);
718 button = pref_radiobutton_new(sd->folder_group, buttongrp,
719 editor->name, select,
720 G_CALLBACK(bar_sort_set_filter_cb), sd);
721 g_signal_connect(G_OBJECT(button), "button_press_event", G_CALLBACK(bar_filter_message_cb), NULL);
723 g_object_set_data_full(G_OBJECT(button), "filter_key", key, bar_sort_edit_button_free);
725 g_list_free(editors_list);
727 if (sd->action == BAR_SORT_FILTER && !have_filter) sd->action = BAR_SORT_COPY;
729 sd->collection_group = pref_box_new(sd->vbox, FALSE, GTK_ORIENTATION_VERTICAL, 0);
731 buttongrp = pref_radiobutton_new(sd->collection_group, nullptr,
732 _("Add image"), (sd->selection == BAR_SORT_SELECTION_IMAGE),
733 G_CALLBACK(bar_sort_set_selection_image_cb), sd);
734 pref_radiobutton_new(sd->collection_group, buttongrp,
735 _("Add selection"), (sd->selection == BAR_SORT_SELECTION_SELECTED),
736 G_CALLBACK(bar_sort_set_selection_selected_cb), sd);
738 sd->bookmarks = bookmark_list_new(SORT_KEY_FOLDERS, bar_sort_bookmark_select, sd);
739 DEBUG_NAME(sd->bookmarks);
740 gtk_box_pack_start(GTK_BOX(sd->vbox), sd->bookmarks, TRUE, TRUE, 0);
741 gtk_widget_show(sd->bookmarks);
743 tbar = pref_toolbar_new(sd->vbox, GTK_TOOLBAR_ICONS);
746 sd->add_button = pref_toolbar_button(tbar, GTK_STOCK_ADD, nullptr, FALSE,
748 G_CALLBACK(bar_sort_add_cb), sd);
749 sd->undo_button = pref_toolbar_button(tbar, GTK_STOCK_UNDO, nullptr, FALSE,
750 _("Undo last image"),
751 G_CALLBACK(bar_sort_undo_cb), sd);
752 sd->help_button = pref_toolbar_button(tbar, GTK_STOCK_HELP, nullptr, FALSE,
753 _("Functions additional to Copy and Move"),
754 G_CALLBACK(bar_sort_help_cb), sd);
756 sd->mode = static_cast<SortModeType>(-1);
757 bar_sort_mode_sync(sd, mode);
758 gtk_combo_box_set_active(GTK_COMBO_BOX(combo), static_cast<gint>(sd->mode));
763 GtkWidget *bar_sort_new_from_config(LayoutWindow *lw, const gchar **UNUSED(attribute_names), const gchar **UNUSED(attribute_values))
767 bar = bar_sort_new(lw, lw->options.action, lw->options.mode, lw->options.selection, lw->options.filter_key);
769 if (lw->bar_sort_enabled) gtk_widget_show(bar);
774 * @brief Sets the bar_sort_enabled flag
776 * @param attribute_names
777 * @param attribute_values
779 * Called from rcfile when processing geeqierc.xml on start-up.
780 * It is necessary to set the bar_sort_enabled flag because
781 * the sort manager and desktop files are set up in the idle loop, and
782 * setup is not yet completed during initialisation.
783 * The flag is checked in layout_editors_reload_idle_cb.
785 void bar_sort_cold_start(LayoutWindow *lw, const gchar **attribute_names, const gchar **attribute_values)
787 gboolean enabled = TRUE;
788 gint action = BAR_SORT_COPY;
789 gint mode = BAR_SORT_MODE_FOLDER;
790 gint selection = BAR_SORT_SELECTION_IMAGE;
791 gchar *filter_key = nullptr;
793 while (attribute_names && *attribute_names)
795 const gchar *option = *attribute_names++;
796 const gchar *value = *attribute_values++;
798 if (READ_BOOL_FULL("enabled", enabled)) continue;
799 if (READ_INT_CLAMP_FULL("action", action, 0, BAR_SORT_ACTION_COUNT - 1)) continue;
800 if (READ_INT_CLAMP_FULL("mode", mode, 0, BAR_SORT_MODE_COUNT - 1)) continue;
801 if (READ_INT_CLAMP_FULL("selection", selection, 0, BAR_SORT_SELECTION_COUNT - 1)) continue;
802 if (READ_CHAR_FULL("filter_key", filter_key)) continue;
804 log_printf("unknown attribute %s = %s\n", option, value);
807 lw->options.action = static_cast<SortActionType>(action);
808 lw->options.mode = static_cast<SortModeType>(mode);
809 lw->options.selection = static_cast<SortSelectionType>(selection);
810 lw->options.filter_key = g_strdup(filter_key);
811 lw->bar_sort_enabled = enabled;
816 GtkWidget *bar_sort_new_default(LayoutWindow *lw)
818 return bar_sort_new_from_config(lw, nullptr, nullptr);
821 void bar_sort_write_config(GtkWidget *bar, GString *outstr, gint indent)
826 sd = static_cast<SortData *>(g_object_get_data(G_OBJECT(bar), "bar_sort_data"));
829 WRITE_NL(); WRITE_STRING("<bar_sort ");
830 write_bool_option(outstr, indent, "enabled", gtk_widget_get_visible(bar));
831 WRITE_INT(*sd, mode);
832 WRITE_INT(*sd, action);
833 WRITE_INT(*sd, selection);
834 WRITE_CHAR(*sd, filter_key);
839 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */