2 * Copyright (C) 2008 - 2016 The Geeqie Team
4 * Author: Laurent Monin
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License along
17 * with this program; if not, write to the Free Software Foundation, Inc.,
18 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
21 #include "view-file.h"
24 #include <glib-object.h>
31 #include "history-list.h"
34 #include "main-defines.h"
40 #include "ui-fileops.h"
43 #include "ui-utildlg.h"
45 #include "view-file/view-file-icon.h"
46 #include "view-file/view-file-list.h"
50 *-----------------------------------------------------------------------------
52 *-----------------------------------------------------------------------------
55 void vf_send_update(ViewFile *vf)
57 if (vf->func_status) vf->func_status(vf, vf->data_status);
61 *-----------------------------------------------------------------------------
63 *-----------------------------------------------------------------------------
66 void vf_sort_set(ViewFile *vf, SortType type, gboolean ascend, gboolean case_sensitive)
70 case FILEVIEW_LIST: vflist_sort_set(vf, type, ascend, case_sensitive); break;
71 case FILEVIEW_ICON: vficon_sort_set(vf, type, ascend, case_sensitive); break;
76 *-----------------------------------------------------------------------------
78 *-----------------------------------------------------------------------------
81 FileData *vf_index_get_data(ViewFile *vf, gint row)
83 return static_cast<FileData *>(g_list_nth_data(vf->list, row));
86 gint vf_index_by_fd(ViewFile *vf, FileData *fd)
92 case FILEVIEW_LIST: ret = vflist_index_by_fd(vf, fd); break;
93 case FILEVIEW_ICON: ret = vficon_index_by_fd(vf, fd); break;
100 guint vf_count(ViewFile *vf, gint64 *bytes)
110 auto fd = static_cast<FileData *>(work->data);
119 return g_list_length(vf->list);
122 GList *vf_get_list(ViewFile *vf)
124 return filelist_copy(vf->list);
128 *-------------------------------------------------------------------
130 *-------------------------------------------------------------------
133 static gboolean vf_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
135 auto vf = static_cast<ViewFile *>(data);
140 case FILEVIEW_LIST: ret = vflist_press_key_cb(widget, event, data); break;
141 case FILEVIEW_ICON: ret = vficon_press_key_cb(widget, event, data); break;
142 default: ret = FALSE;
149 *-------------------------------------------------------------------
151 *-------------------------------------------------------------------
154 static gboolean vf_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
156 auto vf = static_cast<ViewFile *>(data);
161 case FILEVIEW_LIST: ret = vflist_press_cb(widget, bevent, data); break;
162 case FILEVIEW_ICON: ret = vficon_press_cb(widget, bevent, data); break;
163 default: ret = FALSE;
169 static gboolean vf_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
171 auto vf = static_cast<ViewFile *>(data);
176 case FILEVIEW_LIST: ret = vflist_release_cb(widget, bevent, data); break;
177 case FILEVIEW_ICON: ret = vficon_release_cb(widget, bevent, data); break;
178 default: ret = FALSE;
186 *-----------------------------------------------------------------------------
188 *-----------------------------------------------------------------------------
191 guint vf_selection_count(ViewFile *vf, gint64 *bytes)
197 case FILEVIEW_LIST: ret = vflist_selection_count(vf, bytes); break;
198 case FILEVIEW_ICON: ret = vficon_selection_count(vf, bytes); break;
205 GList *vf_selection_get_list(ViewFile *vf)
211 case FILEVIEW_LIST: ret = vflist_selection_get_list(vf); break;
212 case FILEVIEW_ICON: ret = vficon_selection_get_list(vf); break;
213 default: ret = nullptr;
219 GList *vf_selection_get_list_by_index(ViewFile *vf)
225 case FILEVIEW_LIST: ret = vflist_selection_get_list_by_index(vf); break;
226 case FILEVIEW_ICON: ret = vficon_selection_get_list_by_index(vf); break;
227 default: ret = nullptr;
233 void vf_selection_foreach(ViewFile *vf, const ViewFileSelectionCallback &func)
242 if (vf->type == FILEVIEW_ICON)
244 if (!VFICON(vf)->selection) return;
245 work = VFICON(vf)->selection;
249 GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vf->listview));
250 work = gtk_tree_selection_get_selected_rows(selection, &store);
253 for (; work; work = work->next)
255 if (vf->type == FILEVIEW_ICON)
257 fd_n = static_cast<FileData *>(work->data);
261 auto *tpath = static_cast<GtkTreePath *>(work->data);
262 gtk_tree_model_get_iter(store, &iter, tpath);
263 gtk_tree_model_get(store, &iter, VIEW_FILE_COLUMN_POINTER, &fd_n, -1);
270 void vf_select_all(ViewFile *vf)
274 case FILEVIEW_LIST: vflist_select_all(vf); break;
275 case FILEVIEW_ICON: vficon_select_all(vf); break;
279 void vf_select_none(ViewFile *vf)
283 case FILEVIEW_LIST: vflist_select_none(vf); break;
284 case FILEVIEW_ICON: vficon_select_none(vf); break;
288 void vf_select_invert(ViewFile *vf)
292 case FILEVIEW_LIST: vflist_select_invert(vf); break;
293 case FILEVIEW_ICON: vficon_select_invert(vf); break;
297 void vf_select_by_fd(ViewFile *vf, FileData *fd)
301 case FILEVIEW_LIST: vflist_select_by_fd(vf, fd); break;
302 case FILEVIEW_ICON: vficon_select_by_fd(vf, fd); break;
306 void vf_select_list(ViewFile *vf, GList *list)
310 case FILEVIEW_LIST: vflist_select_list(vf, list); break;
311 case FILEVIEW_ICON: vficon_select_list(vf, list); break;
315 void vf_mark_to_selection(ViewFile *vf, gint mark, MarkToSelectionMode mode)
319 case FILEVIEW_LIST: vflist_mark_to_selection(vf, mark, mode); break;
320 case FILEVIEW_ICON: vficon_mark_to_selection(vf, mark, mode); break;
324 void vf_selection_to_mark(ViewFile *vf, gint mark, SelectionToMarkMode mode)
328 case FILEVIEW_LIST: vflist_selection_to_mark(vf, mark, mode); break;
329 case FILEVIEW_ICON: vficon_selection_to_mark(vf, mark, mode); break;
334 *-----------------------------------------------------------------------------
336 *-----------------------------------------------------------------------------
340 static void vf_dnd_init(ViewFile *vf)
344 case FILEVIEW_LIST: vflist_dnd_init(vf); break;
345 case FILEVIEW_ICON: vficon_dnd_init(vf); break;
350 *-----------------------------------------------------------------------------
352 *-----------------------------------------------------------------------------
355 GList *vf_pop_menu_file_list(ViewFile *vf)
361 case FILEVIEW_LIST: ret = vflist_pop_menu_file_list(vf); break;
362 case FILEVIEW_ICON: ret = vficon_pop_menu_file_list(vf); break;
363 default: ret = nullptr;
369 GList *vf_selection_get_one(ViewFile *vf, FileData *fd)
375 case FILEVIEW_LIST: ret = vflist_selection_get_one(vf, fd); break;
376 case FILEVIEW_ICON: ret = vficon_selection_get_one(vf, fd); break;
377 default: ret = nullptr;
383 static void vf_pop_menu_edit_cb(GtkWidget *widget, gpointer data)
386 auto key = static_cast<const gchar *>(data);
388 vf = static_cast<ViewFile *>(submenu_item_get_data(widget));
392 file_util_start_editor_from_filelist(key, vf_pop_menu_file_list(vf), vf->dir_fd->path, vf->listview);
395 static void vf_pop_menu_view_cb(GtkWidget *widget, gpointer data)
397 auto vf = static_cast<ViewFile *>(data);
401 case FILEVIEW_LIST: vflist_pop_menu_view_cb(widget, data); break;
402 case FILEVIEW_ICON: vficon_pop_menu_view_cb(widget, data); break;
406 static void vf_pop_menu_open_archive_cb(GtkWidget *, gpointer data)
408 auto vf = static_cast<ViewFile *>(data);
409 LayoutWindow *lw_new;
410 FileData *fd = nullptr;
416 fd = (VFLIST(vf)->click_fd);
419 fd = (VFICON(vf)->click_fd);
423 dest_dir = open_archive(fd);
426 lw_new = layout_new_from_default();
427 layout_set_path(lw_new, dest_dir);
432 warning_dialog(_("Cannot open archive file"), _("See the Log Window"), GQ_ICON_DIALOG_WARNING, nullptr);
436 static void vf_pop_menu_copy_cb(GtkWidget *, gpointer data)
438 auto vf = static_cast<ViewFile *>(data);
440 file_util_copy(nullptr, vf_pop_menu_file_list(vf), nullptr, vf->listview);
443 static void vf_pop_menu_move_cb(GtkWidget *, gpointer data)
445 auto vf = static_cast<ViewFile *>(data);
447 file_util_move(nullptr, vf_pop_menu_file_list(vf), nullptr, vf->listview);
450 static void vf_pop_menu_rename_cb(GtkWidget *widget, gpointer data)
452 auto vf = static_cast<ViewFile *>(data);
456 case FILEVIEW_LIST: vflist_pop_menu_rename_cb(widget, data); break;
457 case FILEVIEW_ICON: vficon_pop_menu_rename_cb(widget, data); break;
461 static void vf_pop_menu_delete_cb(GtkWidget *, gpointer data)
463 auto vf = static_cast<ViewFile *>(data);
465 options->file_ops.safe_delete_enable = FALSE;
466 file_util_delete(nullptr, vf_pop_menu_file_list(vf), vf->listview);
469 static void vf_pop_menu_move_to_trash_cb(GtkWidget *, gpointer data)
471 auto vf = static_cast<ViewFile *>(data);
473 options->file_ops.safe_delete_enable = TRUE;
474 file_util_delete(nullptr, vf_pop_menu_file_list(vf), vf->listview);
477 static void vf_pop_menu_copy_path_cb(GtkWidget *, gpointer data)
479 auto vf = static_cast<ViewFile *>(data);
481 file_util_path_list_to_clipboard(vf_pop_menu_file_list(vf), TRUE, TRUE);
484 static void vf_pop_menu_copy_path_unquoted_cb(GtkWidget *, gpointer data)
486 auto vf = static_cast<ViewFile *>(data);
488 file_util_path_list_to_clipboard(vf_pop_menu_file_list(vf), FALSE, TRUE);
491 static void vf_pop_menu_cut_path_cb(GtkWidget *, gpointer data)
493 auto vf = static_cast<ViewFile *>(data);
495 file_util_path_list_to_clipboard(vf_pop_menu_file_list(vf), FALSE, FALSE);
498 static void vf_pop_menu_enable_grouping_cb(GtkWidget *, gpointer data)
500 auto vf = static_cast<ViewFile *>(data);
502 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), FALSE);
505 static void vf_pop_menu_duplicates_cb(GtkWidget *, gpointer data)
507 auto vf = static_cast<ViewFile *>(data);
510 dw = dupe_window_new();
511 dupe_window_add_files(dw, vf_pop_menu_file_list(vf), FALSE);
514 static void vf_pop_menu_disable_grouping_cb(GtkWidget *, gpointer data)
516 auto vf = static_cast<ViewFile *>(data);
518 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), TRUE);
521 static void vf_pop_menu_sort_cb(GtkWidget *widget, gpointer data)
526 if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget))) return;
528 vf = static_cast<ViewFile *>(submenu_item_get_data(widget));
531 type = static_cast<SortType>GPOINTER_TO_INT(data);
533 if (type == SORT_EXIFTIME || type == SORT_EXIFTIMEDIGITIZED || type == SORT_RATING)
535 vf_read_metadata_in_idle(vf);
540 layout_sort_set_files(vf->layout, type, vf->sort_ascend, vf->sort_case);
544 vf_sort_set(vf, type, vf->sort_ascend, vf->sort_case);
548 static void vf_pop_menu_sort_ascend_cb(GtkWidget *, gpointer data)
550 auto vf = static_cast<ViewFile *>(data);
554 layout_sort_set_files(vf->layout, vf->sort_method, !vf->sort_ascend, vf->sort_case);
558 vf_sort_set(vf, vf->sort_method, !vf->sort_ascend, vf->sort_case);
562 static void vf_pop_menu_sort_case_cb(GtkWidget *, gpointer data)
564 auto vf = static_cast<ViewFile *>(data);
568 layout_sort_set_files(vf->layout, vf->sort_method, vf->sort_ascend, !vf->sort_case);
572 vf_sort_set(vf, vf->sort_method, vf->sort_ascend, !vf->sort_case);
576 static void vf_pop_menu_sel_mark_cb(GtkWidget *, gpointer data)
578 auto vf = static_cast<ViewFile *>(data);
579 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_SET);
582 static void vf_pop_menu_sel_mark_and_cb(GtkWidget *, gpointer data)
584 auto vf = static_cast<ViewFile *>(data);
585 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_AND);
588 static void vf_pop_menu_sel_mark_or_cb(GtkWidget *, gpointer data)
590 auto vf = static_cast<ViewFile *>(data);
591 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_OR);
594 static void vf_pop_menu_sel_mark_minus_cb(GtkWidget *, gpointer data)
596 auto vf = static_cast<ViewFile *>(data);
597 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_MINUS);
600 static void vf_pop_menu_set_mark_sel_cb(GtkWidget *, gpointer data)
602 auto vf = static_cast<ViewFile *>(data);
603 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_SET);
606 static void vf_pop_menu_res_mark_sel_cb(GtkWidget *, gpointer data)
608 auto vf = static_cast<ViewFile *>(data);
609 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_RESET);
612 static void vf_pop_menu_toggle_mark_sel_cb(GtkWidget *, gpointer data)
614 auto vf = static_cast<ViewFile *>(data);
615 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_TOGGLE);
618 static void vf_pop_menu_toggle_view_type_cb(GtkWidget *widget, gpointer data)
620 auto vf = static_cast<ViewFile *>(data);
621 auto new_type = static_cast<FileViewType>(GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), "menu_item_radio_data")));
622 if (!vf->layout) return;
624 layout_views_set(vf->layout, vf->layout->options.dir_view_type, new_type);
627 static void vf_pop_menu_refresh_cb(GtkWidget *widget, gpointer data)
629 auto vf = static_cast<ViewFile *>(data);
633 case FILEVIEW_LIST: vflist_pop_menu_refresh_cb(widget, data); break;
634 case FILEVIEW_ICON: vficon_pop_menu_refresh_cb(widget, data); break;
638 static void vf_popup_destroy_cb(GtkWidget *widget, gpointer data)
640 auto vf = static_cast<ViewFile *>(data);
644 case FILEVIEW_LIST: vflist_popup_destroy_cb(widget, data); break;
645 case FILEVIEW_ICON: vficon_popup_destroy_cb(widget, data); break;
648 filelist_free(vf->editmenu_fd_list);
649 vf->editmenu_fd_list = nullptr;
653 * @brief Add file selection list to a collection
655 * @param[in] data Index to the collection list menu item selected, or -1 for new collection
659 static void vf_pop_menu_collections_cb(GtkWidget *widget, gpointer data)
662 GList *selection_list;
664 vf = static_cast<ViewFile *>(submenu_item_get_data(widget));
665 selection_list = vf_selection_get_list(vf);
666 pop_menu_collections(selection_list, data);
668 filelist_free(selection_list);
671 GtkWidget *vf_pop_menu(ViewFile *vf)
676 gboolean active = FALSE;
677 gboolean class_archive = FALSE;
678 GtkAccelGroup *accel_group;
683 vflist_color_set(vf, VFLIST(vf)->click_fd, TRUE);
684 active = (VFLIST(vf)->click_fd != nullptr);
685 class_archive = (VFLIST(vf)->click_fd != nullptr && VFLIST(vf)->click_fd->format_class == FORMAT_CLASS_ARCHIVE);
688 active = (VFICON(vf)->click_fd != nullptr);
689 class_archive = (VFICON(vf)->click_fd != nullptr && VFICON(vf)->click_fd->format_class == FORMAT_CLASS_ARCHIVE);
693 menu = popup_menu_short_lived();
695 accel_group = gtk_accel_group_new();
696 gtk_menu_set_accel_group(GTK_MENU(menu), accel_group);
698 g_object_set_data(G_OBJECT(menu), "window_keys", nullptr);
699 g_object_set_data(G_OBJECT(menu), "accel_group", accel_group);
701 g_signal_connect(G_OBJECT(menu), "destroy",
702 G_CALLBACK(vf_popup_destroy_cb), vf);
704 if (vf->clicked_mark > 0)
706 gint mark = vf->clicked_mark;
707 gchar *str_set_mark = g_strdup_printf(_("_Set mark %d"), mark);
708 gchar *str_res_mark = g_strdup_printf(_("_Reset mark %d"), mark);
709 gchar *str_toggle_mark = g_strdup_printf(_("_Toggle mark %d"), mark);
710 gchar *str_sel_mark = g_strdup_printf(_("_Select mark %d"), mark);
711 gchar *str_sel_mark_or = g_strdup_printf(_("_Add mark %d"), mark);
712 gchar *str_sel_mark_and = g_strdup_printf(_("_Intersection with mark %d"), mark);
713 gchar *str_sel_mark_minus = g_strdup_printf(_("_Unselect mark %d"), mark);
715 g_assert(mark >= 1 && mark <= FILEDATA_MARKS_SIZE);
717 vf->active_mark = mark;
718 vf->clicked_mark = 0;
720 menu_item_add_sensitive(menu, str_set_mark, active,
721 G_CALLBACK(vf_pop_menu_set_mark_sel_cb), vf);
723 menu_item_add_sensitive(menu, str_res_mark, active,
724 G_CALLBACK(vf_pop_menu_res_mark_sel_cb), vf);
726 menu_item_add_sensitive(menu, str_toggle_mark, active,
727 G_CALLBACK(vf_pop_menu_toggle_mark_sel_cb), vf);
729 menu_item_add_divider(menu);
731 menu_item_add_sensitive(menu, str_sel_mark, active,
732 G_CALLBACK(vf_pop_menu_sel_mark_cb), vf);
733 menu_item_add_sensitive(menu, str_sel_mark_or, active,
734 G_CALLBACK(vf_pop_menu_sel_mark_or_cb), vf);
735 menu_item_add_sensitive(menu, str_sel_mark_and, active,
736 G_CALLBACK(vf_pop_menu_sel_mark_and_cb), vf);
737 menu_item_add_sensitive(menu, str_sel_mark_minus, active,
738 G_CALLBACK(vf_pop_menu_sel_mark_minus_cb), vf);
740 menu_item_add_divider(menu);
742 g_free(str_set_mark);
743 g_free(str_res_mark);
744 g_free(str_toggle_mark);
745 g_free(str_sel_mark);
746 g_free(str_sel_mark_and);
747 g_free(str_sel_mark_or);
748 g_free(str_sel_mark_minus);
751 vf->editmenu_fd_list = vf_pop_menu_file_list(vf);
752 submenu_add_edit(menu, &item, G_CALLBACK(vf_pop_menu_edit_cb), vf, vf->editmenu_fd_list);
753 gtk_widget_set_sensitive(item, active);
755 menu_item_add_icon_sensitive(menu, _("View in _new window"), GQ_ICON_NEW, active,
756 G_CALLBACK(vf_pop_menu_view_cb), vf);
758 menu_item_add_icon_sensitive(menu, _("Open archive"), GQ_ICON_OPEN, active & class_archive, G_CALLBACK(vf_pop_menu_open_archive_cb), vf);
760 menu_item_add_divider(menu);
761 menu_item_add_icon_sensitive(menu, _("_Copy..."), GQ_ICON_COPY, active,
762 G_CALLBACK(vf_pop_menu_copy_cb), vf);
763 menu_item_add_sensitive(menu, _("_Move..."), active,
764 G_CALLBACK(vf_pop_menu_move_cb), vf);
765 menu_item_add_sensitive(menu, _("_Rename..."), active,
766 G_CALLBACK(vf_pop_menu_rename_cb), vf);
767 menu_item_add_sensitive(menu, _("_Copy to clipboard"), active,
768 G_CALLBACK(vf_pop_menu_copy_path_cb), vf);
769 menu_item_add_sensitive(menu, _("_Copy to clipboard (unquoted)"), active,
770 G_CALLBACK(vf_pop_menu_copy_path_unquoted_cb), vf);
771 menu_item_add_sensitive(menu, _("_Cut to clipboard"), active,
772 G_CALLBACK(vf_pop_menu_cut_path_cb), vf);
773 menu_item_add_divider(menu);
774 menu_item_add_icon_sensitive(menu,
775 options->file_ops.confirm_move_to_trash ? _("Move to Trash...") :
776 _("Move to Trash"), GQ_ICON_DELETE, active,
777 G_CALLBACK(vf_pop_menu_move_to_trash_cb), vf);
778 menu_item_add_icon_sensitive(menu,
779 options->file_ops.confirm_delete ? _("_Delete...") :
780 _("_Delete"), GQ_ICON_DELETE_SHRED, active,
781 G_CALLBACK(vf_pop_menu_delete_cb), vf);
782 menu_item_add_divider(menu);
784 menu_item_add_sensitive(menu, _("Enable file _grouping"), active,
785 G_CALLBACK(vf_pop_menu_enable_grouping_cb), vf);
786 menu_item_add_sensitive(menu, _("Disable file groupi_ng"), active,
787 G_CALLBACK(vf_pop_menu_disable_grouping_cb), vf);
789 menu_item_add_divider(menu);
790 menu_item_add_icon_sensitive(menu, _("_Find duplicates..."), GQ_ICON_FIND, active,
791 G_CALLBACK(vf_pop_menu_duplicates_cb), vf);
792 menu_item_add_divider(menu);
794 submenu = submenu_add_collections(menu, &item,
795 G_CALLBACK(vf_pop_menu_collections_cb), vf);
796 gtk_widget_set_sensitive(item, active);
797 menu_item_add_divider(menu);
799 submenu = submenu_add_sort(nullptr, G_CALLBACK(vf_pop_menu_sort_cb), vf,
800 FALSE, FALSE, TRUE, vf->sort_method);
801 menu_item_add_divider(submenu);
802 menu_item_add_check(submenu, _("Ascending"), vf->sort_ascend,
803 G_CALLBACK(vf_pop_menu_sort_ascend_cb), vf);
804 menu_item_add_check(submenu, _("Case"), vf->sort_ascend,
805 G_CALLBACK(vf_pop_menu_sort_case_cb), vf);
807 item = menu_item_add(menu, _("_Sort"), nullptr, nullptr);
808 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
810 item = menu_item_add_radio(menu, _("Images as List"), GINT_TO_POINTER(FILEVIEW_LIST), vf->type == FILEVIEW_LIST,
811 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
813 item = menu_item_add_radio(menu, _("Images as Icons"), GINT_TO_POINTER(FILEVIEW_ICON), vf->type == FILEVIEW_ICON,
814 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
819 menu_item_add_check(menu, _("Show _thumbnails"), VFLIST(vf)->thumbs_enabled,
820 G_CALLBACK(vflist_pop_menu_thumbs_cb), vf);
823 menu_item_add_check(menu, _("Show filename _text"), VFICON(vf)->show_text,
824 G_CALLBACK(vficon_pop_menu_show_names_cb), vf);
831 menu_item_add_check(menu, _("Show star rating"), options->show_star_rating,
832 G_CALLBACK(vflist_pop_menu_show_star_rating_cb), vf);
835 menu_item_add_check(menu, _("Show star rating"), options->show_star_rating,
836 G_CALLBACK(vficon_pop_menu_show_star_rating_cb), vf);
840 menu_item_add_icon(menu, _("Re_fresh"), GQ_ICON_REFRESH, G_CALLBACK(vf_pop_menu_refresh_cb), vf);
845 gboolean vf_refresh(ViewFile *vf)
851 case FILEVIEW_LIST: ret = vflist_refresh(vf); break;
852 case FILEVIEW_ICON: ret = vficon_refresh(vf); break;
853 default: ret = FALSE;
859 gboolean vf_set_fd(ViewFile *vf, FileData *dir_fd)
865 case FILEVIEW_LIST: ret = vflist_set_fd(vf, dir_fd); break;
866 case FILEVIEW_ICON: ret = vficon_set_fd(vf, dir_fd); break;
867 default: ret = FALSE;
873 static void vf_destroy_cb(GtkWidget *widget, gpointer data)
875 auto vf = static_cast<ViewFile *>(data);
879 case FILEVIEW_LIST: vflist_destroy_cb(widget, data); break;
880 case FILEVIEW_ICON: vficon_destroy_cb(widget, data); break;
885 g_signal_handlers_disconnect_matched(G_OBJECT(vf->popup), G_SIGNAL_MATCH_DATA,
886 0, 0, nullptr, nullptr, vf);
887 gq_gtk_widget_destroy(vf->popup);
890 if (vf->read_metadata_in_idle_id)
892 g_idle_remove_by_data(vf);
894 file_data_unref(vf->dir_fd);
899 static void vf_marks_filter_toggle_cb(GtkWidget *, gpointer data)
901 auto vf = static_cast<ViewFile *>(data);
905 struct MarksTextEntry {
908 GtkWidget *edit_widget;
913 static void vf_marks_tooltip_cancel_cb(GenericDialog *gd, gpointer data)
915 auto mte = static_cast<MarksTextEntry *>(data);
917 g_free(mte->text_entry);
918 generic_dialog_close(gd);
921 static void vf_marks_tooltip_ok_cb(GenericDialog *gd, gpointer data)
923 auto mte = static_cast<MarksTextEntry *>(data);
925 g_free(options->marks_tooltips[mte->mark_no]);
926 options->marks_tooltips[mte->mark_no] = g_strdup(gq_gtk_entry_get_text(GTK_ENTRY(mte->edit_widget)));
928 gtk_widget_set_tooltip_text(mte->parent, options->marks_tooltips[mte->mark_no]);
930 g_free(mte->text_entry);
931 generic_dialog_close(gd);
934 void vf_marks_filter_on_icon_press(GtkEntry *, GtkEntryIconPosition, GdkEvent *, gpointer userdata)
936 auto mte = static_cast<MarksTextEntry *>(userdata);
938 g_free(mte->text_entry);
939 mte->text_entry = g_strdup("");
940 gq_gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), "");
943 static void vf_marks_tooltip_help_cb(GenericDialog *, gpointer)
945 help_window_show("GuideImageMarks.html");
948 static gboolean vf_marks_tooltip_cb(GtkWidget *widget,
949 GdkEventButton *event,
953 gint i = GPOINTER_TO_INT(user_data);
955 if (event->button != MOUSE_BUTTON_RIGHT)
958 auto mte = g_new0(MarksTextEntry, 1);
960 mte->text_entry = g_strdup(options->marks_tooltips[i]);
961 mte->parent = widget;
963 mte->gd = generic_dialog_new(_("Mark text"), "mark_text",
965 vf_marks_tooltip_cancel_cb, mte);
966 generic_dialog_add_message(mte->gd, GQ_ICON_DIALOG_QUESTION, _("Set mark text"),
967 _("This will set or clear the mark text."), FALSE);
968 generic_dialog_add_button(mte->gd, GQ_ICON_OK, "OK",
969 vf_marks_tooltip_ok_cb, TRUE);
970 generic_dialog_add_button(mte->gd, GQ_ICON_HELP, _("Help"),
971 vf_marks_tooltip_help_cb, FALSE);
973 table = pref_table_new(mte->gd->vbox, 3, 1, FALSE, TRUE);
974 pref_table_label(table, 0, 0, g_strdup_printf("%s%d", _("Mark "), mte->mark_no + 1), GTK_ALIGN_END);
975 mte->edit_widget = gtk_entry_new();
976 gtk_widget_set_size_request(mte->edit_widget, 300, -1);
979 gq_gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), mte->text_entry);
981 gq_gtk_grid_attach_default(GTK_GRID(table), mte->edit_widget, 1, 2, 0, 1);
982 generic_dialog_attach_default(mte->gd, mte->edit_widget);
984 gtk_entry_set_icon_from_icon_name(GTK_ENTRY(mte->edit_widget),
985 GTK_ENTRY_ICON_SECONDARY, GQ_ICON_CLEAR);
986 gtk_entry_set_icon_tooltip_text(GTK_ENTRY(mte->edit_widget),
987 GTK_ENTRY_ICON_SECONDARY, _("Clear"));
988 g_signal_connect(GTK_ENTRY(mte->edit_widget), "icon-press",
989 G_CALLBACK(vf_marks_filter_on_icon_press), mte);
991 gtk_widget_show(mte->edit_widget);
992 gtk_widget_grab_focus(mte->edit_widget);
993 gtk_widget_show(GTK_WIDGET(mte->gd->dialog));
998 static void vf_file_filter_save_cb(GtkWidget *, gpointer data)
1000 auto vf = static_cast<ViewFile *>(data);
1002 gchar *remove_text = nullptr;
1003 gchar *index_text = nullptr;
1004 gboolean text_found = FALSE;
1007 entry_text = g_strdup(gq_gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo)))));
1009 if (entry_text[0] == '\0' && vf->file_filter.last_selected >= 0)
1011 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), vf->file_filter.last_selected);
1012 remove_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
1013 history_list_item_remove("file_filter", remove_text);
1014 gtk_combo_box_text_remove(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), vf->file_filter.last_selected);
1015 g_free(remove_text);
1017 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), -1);
1018 vf->file_filter.last_selected = - 1;
1019 gq_gtk_entry_set_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo))), "");
1020 vf->file_filter.count--;
1024 if (entry_text[0] != '\0')
1026 for (i = 0; i < vf->file_filter.count; i++)
1032 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), i);
1033 index_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
1035 if (g_strcmp0(index_text, entry_text) == 0)
1045 history_list_add_to_key("file_filter", entry_text, 10);
1046 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), entry_text);
1047 vf->file_filter.count++;
1048 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), vf->file_filter.count - 1);
1057 static void vf_file_filter_cb(GtkWidget *, gpointer data)
1059 auto vf = static_cast<ViewFile *>(data);
1064 static gboolean vf_file_filter_press_cb(GtkWidget *widget, GdkEventButton *, gpointer data)
1066 auto vf = static_cast<ViewFile *>(data);
1067 vf->file_filter.last_selected = gtk_combo_box_get_active(GTK_COMBO_BOX(vf->file_filter.combo));
1069 gtk_widget_grab_focus(widget);
1074 static GtkWidget *vf_marks_filter_init(ViewFile *vf)
1076 GtkWidget *frame = gtk_frame_new(nullptr);
1077 GtkWidget *hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
1081 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1083 GtkWidget *check = gtk_check_button_new();
1084 gq_gtk_box_pack_start(GTK_BOX(hbox), check, FALSE, FALSE, 0);
1085 g_signal_connect(G_OBJECT(check), "toggled",
1086 G_CALLBACK(vf_marks_filter_toggle_cb), vf);
1087 g_signal_connect(G_OBJECT(check), "button_press_event",
1088 G_CALLBACK(vf_marks_tooltip_cb), GINT_TO_POINTER(i));
1089 gtk_widget_set_tooltip_text(check, options->marks_tooltips[i]);
1091 gtk_widget_show(check);
1092 vf->filter_check[i] = check;
1094 gq_gtk_container_add(GTK_WIDGET(frame), hbox);
1095 gtk_widget_show(hbox);
1099 void vf_file_filter_set(ViewFile *vf, gboolean enable)
1103 gtk_widget_show(vf->file_filter.combo);
1104 gtk_widget_show(vf->file_filter.frame);
1108 gtk_widget_hide(vf->file_filter.combo);
1109 gtk_widget_hide(vf->file_filter.frame);
1115 static gboolean vf_file_filter_class_cb(GtkWidget *widget, gpointer data)
1117 auto vf = static_cast<ViewFile *>(data);
1120 gboolean state = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
1122 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1124 if (g_strcmp0(format_class_list[i], gtk_menu_item_get_label(GTK_MENU_ITEM(widget))) == 0)
1126 options->class_filter[i] = state;
1134 static gboolean vf_file_filter_class_set_all_cb(GtkWidget *widget, gpointer data)
1136 auto vf = static_cast<ViewFile *>(data);
1143 if (g_strcmp0(_("Select all"), gtk_menu_item_get_label(GTK_MENU_ITEM(widget))) == 0)
1152 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1154 options->class_filter[i] = state;
1158 parent = gtk_widget_get_parent(widget);
1159 children = gtk_container_get_children(GTK_CONTAINER(parent));
1162 child = static_cast<GtkWidget *>(children->data);
1163 if (i < FILE_FORMAT_CLASSES)
1165 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(child), state);
1168 children = children->next;
1170 g_list_free(children);
1176 static GtkWidget *class_filter_menu (ViewFile *vf)
1179 GtkWidget *menu_item;
1182 menu = gtk_menu_new();
1184 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1186 menu_item = gtk_check_menu_item_new_with_label(format_class_list[i]);
1187 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menu_item), options->class_filter[i]);
1188 g_signal_connect(G_OBJECT(menu_item), "toggled", G_CALLBACK(vf_file_filter_class_cb), vf);
1189 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1190 gtk_widget_show(menu_item);
1193 menu_item = gtk_menu_item_new_with_label(_("Select all"));
1194 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1195 gtk_widget_show(menu_item);
1196 g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(vf_file_filter_class_set_all_cb), vf);
1198 menu_item = gtk_menu_item_new_with_label(_("Select none"));
1199 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1200 gtk_widget_show(menu_item);
1201 g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(vf_file_filter_class_set_all_cb), vf);
1206 static void case_sensitive_cb(GtkWidget *widget, gpointer data)
1208 auto vf = static_cast<ViewFile *>(data);
1210 vf->file_filter.case_sensitive = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
1214 static void file_filter_clear_cb(GtkEntry *, GtkEntryIconPosition pos, GdkEvent *, gpointer userdata)
1216 if (pos == GTK_ENTRY_ICON_SECONDARY)
1218 gq_gtk_entry_set_text(GTK_ENTRY(userdata), "");
1219 gtk_widget_grab_focus(GTK_WIDGET(userdata));
1223 static GtkWidget *vf_file_filter_init(ViewFile *vf)
1225 GtkWidget *frame = gtk_frame_new(nullptr);
1226 GtkWidget *hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
1229 GtkWidget *combo_entry;
1231 GtkWidget *menuitem;
1232 GtkWidget *case_sensitive;
1237 vf->file_filter.combo = gtk_combo_box_text_new_with_entry();
1238 combo_entry = gtk_bin_get_child(GTK_BIN(vf->file_filter.combo));
1239 gtk_widget_show(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo)));
1240 gtk_widget_show((GTK_WIDGET(vf->file_filter.combo)));
1241 gtk_widget_set_tooltip_text(GTK_WIDGET(vf->file_filter.combo), _("Use regular expressions"));
1243 gtk_entry_set_icon_from_icon_name(GTK_ENTRY(combo_entry), GTK_ENTRY_ICON_SECONDARY, GQ_ICON_CLEAR);
1244 gtk_entry_set_icon_tooltip_text (GTK_ENTRY(combo_entry), GTK_ENTRY_ICON_SECONDARY, _("Clear"));
1245 g_signal_connect(GTK_ENTRY(combo_entry), "icon-press", G_CALLBACK(file_filter_clear_cb), combo_entry);
1247 work = history_list_get_by_key("file_filter");
1250 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), static_cast<gchar *>(work->data));
1253 vf->file_filter.count = n;
1255 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), 0);
1257 g_signal_connect(G_OBJECT(combo_entry), "activate",
1258 G_CALLBACK(vf_file_filter_save_cb), vf);
1260 g_signal_connect(G_OBJECT(vf->file_filter.combo), "changed",
1261 G_CALLBACK(vf_file_filter_cb), vf);
1263 g_signal_connect(G_OBJECT(combo_entry), "button_press_event",
1264 G_CALLBACK(vf_file_filter_press_cb), vf);
1266 gq_gtk_box_pack_start(GTK_BOX(hbox), vf->file_filter.combo, FALSE, FALSE, 0);
1267 gtk_widget_show(vf->file_filter.combo);
1268 gq_gtk_container_add(GTK_WIDGET(frame), hbox);
1269 gtk_widget_show(hbox);
1271 case_sensitive = gtk_check_button_new_with_label(_("Case"));
1272 gq_gtk_box_pack_start(GTK_BOX(hbox), case_sensitive, FALSE, FALSE, 0);
1273 gtk_widget_set_tooltip_text(GTK_WIDGET(case_sensitive), _("Case sensitive"));
1274 g_signal_connect(G_OBJECT(case_sensitive), "clicked", G_CALLBACK(case_sensitive_cb), vf);
1275 gtk_widget_show(case_sensitive);
1277 menubar = gtk_menu_bar_new();
1278 gq_gtk_box_pack_start(GTK_BOX(hbox), menubar, FALSE, TRUE, 0);
1279 gtk_widget_show(menubar);
1281 box = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, PREF_PAD_GAP);
1282 icon = gtk_image_new_from_icon_name(GQ_ICON_PAN_DOWN, GTK_ICON_SIZE_MENU);
1283 label = gtk_label_new(_("Class"));
1285 gq_gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
1286 gq_gtk_box_pack_end(GTK_BOX(box), icon, FALSE, FALSE, 0);
1288 menuitem = gtk_menu_item_new();
1290 gtk_widget_set_tooltip_text(GTK_WIDGET(menuitem), _("Select Class filter"));
1291 gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), class_filter_menu(vf));
1292 gtk_menu_shell_append(GTK_MENU_SHELL(menubar), menuitem);
1293 gq_gtk_container_add(GTK_WIDGET(menuitem), box);
1294 gq_gtk_widget_show_all(menuitem);
1299 void vf_mark_filter_toggle(ViewFile *vf, gint mark)
1302 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(vf->filter_check[n]),
1303 !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[n])));
1306 ViewFile *vf_new(FileViewType type, FileData *dir_fd)
1310 vf = g_new0(ViewFile, 1);
1313 vf->sort_method = SORT_NAME;
1314 vf->sort_ascend = TRUE;
1315 vf->read_metadata_in_idle_id = 0;
1317 vf->scrolled = gq_gtk_scrolled_window_new(nullptr, nullptr);
1318 gq_gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vf->scrolled), GTK_SHADOW_IN);
1319 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vf->scrolled),
1320 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1322 vf->filter = vf_marks_filter_init(vf);
1323 vf->file_filter.frame = vf_file_filter_init(vf);
1325 vf->widget = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
1326 gq_gtk_box_pack_start(GTK_BOX(vf->widget), vf->filter, FALSE, FALSE, 0);
1327 gq_gtk_box_pack_start(GTK_BOX(vf->widget), vf->file_filter.frame, FALSE, FALSE, 0);
1328 gq_gtk_box_pack_start(GTK_BOX(vf->widget), vf->scrolled, TRUE, TRUE, 0);
1329 gtk_widget_show(vf->scrolled);
1331 g_signal_connect(G_OBJECT(vf->widget), "destroy",
1332 G_CALLBACK(vf_destroy_cb), vf);
1336 case FILEVIEW_LIST: vf = vflist_new(vf, dir_fd); break;
1337 case FILEVIEW_ICON: vf = vficon_new(vf, dir_fd); break;
1342 g_signal_connect(G_OBJECT(vf->listview), "key_press_event",
1343 G_CALLBACK(vf_press_key_cb), vf);
1344 g_signal_connect(G_OBJECT(vf->listview), "button_press_event",
1345 G_CALLBACK(vf_press_cb), vf);
1346 g_signal_connect(G_OBJECT(vf->listview), "button_release_event",
1347 G_CALLBACK(vf_release_cb), vf);
1349 gq_gtk_container_add(GTK_WIDGET(vf->scrolled), vf->listview);
1350 gtk_widget_show(vf->listview);
1352 if (dir_fd) vf_set_fd(vf, dir_fd);
1357 void vf_set_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gpointer data), gpointer data)
1359 vf->func_status = func;
1360 vf->data_status = data;
1363 void vf_set_thumb_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gdouble val, const gchar *text, gpointer data), gpointer data)
1365 vf->func_thumb_status = func;
1366 vf->data_thumb_status = data;
1369 void vf_thumb_set(ViewFile *vf, gboolean enable)
1373 case FILEVIEW_LIST: vflist_thumb_set(vf, enable); break;
1374 case FILEVIEW_ICON: /*vficon_thumb_set(vf, enable);*/ break;
1379 static gboolean vf_thumb_next(ViewFile *vf);
1381 static gdouble vf_thumb_progress(ViewFile *vf)
1388 case FILEVIEW_LIST: vflist_thumb_progress_count(vf->list, &count, &done); break;
1389 case FILEVIEW_ICON: vficon_thumb_progress_count(vf->list, &count, &done); break;
1392 DEBUG_1("thumb progress: %d of %d", done, count);
1393 return static_cast<gdouble>(done) / count;
1396 static gdouble vf_read_metadata_in_idle_progress(ViewFile *vf)
1403 case FILEVIEW_LIST: vflist_read_metadata_progress_count(vf->list, &count, &done); break;
1404 case FILEVIEW_ICON: vficon_read_metadata_progress_count(vf->list, &count, &done); break;
1407 return static_cast<gdouble>(done) / count;
1410 static void vf_set_thumb_fd(ViewFile *vf, FileData *fd)
1414 case FILEVIEW_LIST: vflist_set_thumb_fd(vf, fd); break;
1415 case FILEVIEW_ICON: vficon_set_thumb_fd(vf, fd); break;
1419 static void vf_thumb_status(ViewFile *vf, gdouble val, const gchar *text)
1421 if (vf->func_thumb_status)
1423 vf->func_thumb_status(vf, val, text, vf->data_thumb_status);
1427 static void vf_thumb_do(ViewFile *vf, FileData *fd)
1431 vf_set_thumb_fd(vf, fd);
1432 vf_thumb_status(vf, vf_thumb_progress(vf), _("Loading thumbs..."));
1435 void vf_thumb_cleanup(ViewFile *vf)
1437 vf_thumb_status(vf, 0.0, nullptr);
1439 vf->thumbs_running = FALSE;
1441 thumb_loader_free(vf->thumbs_loader);
1442 vf->thumbs_loader = nullptr;
1444 vf->thumbs_filedata = nullptr;
1447 void vf_thumb_stop(ViewFile *vf)
1449 if (vf->thumbs_running) vf_thumb_cleanup(vf);
1452 static void vf_thumb_common_cb(ThumbLoader *tl, gpointer data)
1454 auto vf = static_cast<ViewFile *>(data);
1456 if (vf->thumbs_filedata && vf->thumbs_loader == tl)
1458 vf_thumb_do(vf, vf->thumbs_filedata);
1461 while (vf_thumb_next(vf));
1464 static void vf_thumb_error_cb(ThumbLoader *tl, gpointer data)
1466 vf_thumb_common_cb(tl, data);
1469 static void vf_thumb_done_cb(ThumbLoader *tl, gpointer data)
1471 vf_thumb_common_cb(tl, data);
1474 static gboolean vf_thumb_next(ViewFile *vf)
1476 FileData *fd = nullptr;
1478 if (!gtk_widget_get_realized(vf->listview))
1480 vf_thumb_status(vf, 0.0, nullptr);
1486 case FILEVIEW_LIST: fd = vflist_thumb_next_fd(vf); break;
1487 case FILEVIEW_ICON: fd = vficon_thumb_next_fd(vf); break;
1493 vf_thumb_cleanup(vf);
1497 vf->thumbs_filedata = fd;
1499 thumb_loader_free(vf->thumbs_loader);
1501 vf->thumbs_loader = thumb_loader_new(options->thumbnails.max_width, options->thumbnails.max_height);
1502 thumb_loader_set_callbacks(vf->thumbs_loader,
1508 if (!thumb_loader_start(vf->thumbs_loader, fd))
1510 /* set icon to unknown, continue */
1511 DEBUG_1("thumb loader start failed %s", fd->path);
1512 vf_thumb_do(vf, fd);
1520 static void vf_thumb_reset_all(ViewFile *vf)
1524 for (work = vf->list; work; work = work->next)
1526 auto fd = static_cast<FileData *>(work->data);
1527 if (fd->thumb_pixbuf)
1529 g_object_unref(fd->thumb_pixbuf);
1530 fd->thumb_pixbuf = nullptr;
1535 void vf_thumb_update(ViewFile *vf)
1539 if (vf->type == FILEVIEW_LIST && !VFLIST(vf)->thumbs_enabled) return;
1541 vf_thumb_status(vf, 0.0, _("Loading thumbs..."));
1542 vf->thumbs_running = TRUE;
1544 if (thumb_format_changed)
1546 vf_thumb_reset_all(vf);
1547 thumb_format_changed = FALSE;
1550 while (vf_thumb_next(vf));
1553 void vf_star_cleanup(ViewFile *vf)
1555 if (vf->stars_id != 0)
1557 g_source_remove(vf->stars_id);
1561 vf->stars_filedata = nullptr;
1564 void vf_star_stop(ViewFile *vf)
1566 vf_star_cleanup(vf);
1569 static void vf_set_star_fd(ViewFile *vf, FileData *fd)
1573 case FILEVIEW_LIST: vflist_set_star_fd(vf, fd); break;
1574 case FILEVIEW_ICON: vficon_set_star_fd(vf, fd); break;
1579 static void vf_star_do(ViewFile *vf, FileData *fd)
1583 vf_set_star_fd(vf, fd);
1586 static gboolean vf_star_next(ViewFile *vf)
1588 FileData *fd = nullptr;
1592 case FILEVIEW_LIST: fd = vflist_star_next_fd(vf); break;
1593 case FILEVIEW_ICON: fd = vficon_star_next_fd(vf); break;
1600 vf_star_cleanup(vf);
1607 gboolean vf_stars_cb(gpointer data)
1609 auto vf = static_cast<ViewFile *>(data);
1610 FileData *fd = vf->stars_filedata;
1614 read_rating_data(fd);
1618 if (vf_star_next(vf))
1620 return G_SOURCE_CONTINUE;
1623 vf->stars_filedata = nullptr;
1625 return G_SOURCE_REMOVE;
1628 return G_SOURCE_REMOVE;
1631 void vf_star_update(ViewFile *vf)
1635 if (!options->show_star_rating)
1643 void vf_marks_set(ViewFile *vf, gboolean enable)
1645 if (vf->marks_enabled == enable) return;
1647 vf->marks_enabled = enable;
1651 case FILEVIEW_LIST: vflist_marks_set(vf, enable); break;
1652 case FILEVIEW_ICON: vficon_marks_set(vf, enable); break;
1655 gtk_widget_show(vf->filter);
1657 gtk_widget_hide(vf->filter);
1659 vf_refresh_idle(vf);
1661 #pragma GCC diagnostic push
1662 #pragma GCC diagnostic ignored "-Wunused-function"
1663 void vf_star_rating_set_unused(ViewFile *vf, gboolean enable)
1665 if (options->show_star_rating == enable) return;
1666 options->show_star_rating = enable;
1670 case FILEVIEW_LIST: vflist_star_rating_set(vf, enable); break;
1671 case FILEVIEW_ICON: vficon_star_rating_set(vf, enable); break;
1673 vf_refresh_idle(vf);
1675 #pragma GCC diagnostic pop
1677 guint vf_marks_get_filter(ViewFile *vf)
1681 if (!vf->marks_enabled) return 0;
1683 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1685 if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[i])))
1693 GRegex *vf_file_filter_get_filter(ViewFile *vf)
1695 GRegex *ret = nullptr;
1696 GError *error = nullptr;
1697 gchar *file_filter_text = nullptr;
1699 if (!gtk_widget_get_visible(vf->file_filter.combo))
1701 return g_regex_new("", static_cast<GRegexCompileFlags>(0), static_cast<GRegexMatchFlags>(0), nullptr);
1704 file_filter_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
1706 if (file_filter_text[0] != '\0')
1708 ret = g_regex_new(file_filter_text, vf->file_filter.case_sensitive ? static_cast<GRegexCompileFlags>(0) : G_REGEX_CASELESS, static_cast<GRegexMatchFlags>(0), &error);
1711 log_printf("Error: could not compile regular expression %s\n%s\n", file_filter_text, error->message);
1712 g_error_free(error);
1714 ret = g_regex_new("", static_cast<GRegexCompileFlags>(0), static_cast<GRegexMatchFlags>(0), nullptr);
1716 g_free(file_filter_text);
1720 ret = g_regex_new("", static_cast<GRegexCompileFlags>(0), static_cast<GRegexMatchFlags>(0), nullptr);
1726 guint vf_class_get_filter(ViewFile *vf)
1731 if (!gtk_widget_get_visible(vf->file_filter.combo))
1736 for ( i = 0; i < FILE_FORMAT_CLASSES; i++)
1738 if (options->class_filter[i])
1747 void vf_set_layout(ViewFile *vf, LayoutWindow *layout)
1749 vf->layout = layout;
1754 *-----------------------------------------------------------------------------
1755 * maintenance (for rename, move, remove)
1756 *-----------------------------------------------------------------------------
1759 static gboolean vf_refresh_idle_cb(gpointer data)
1761 auto vf = static_cast<ViewFile *>(data);
1764 vf->refresh_idle_id = 0;
1765 return G_SOURCE_REMOVE;
1768 void vf_refresh_idle_cancel(ViewFile *vf)
1770 if (vf->refresh_idle_id)
1772 g_source_remove(vf->refresh_idle_id);
1773 vf->refresh_idle_id = 0;
1778 void vf_refresh_idle(ViewFile *vf)
1780 if (!vf->refresh_idle_id)
1782 vf->time_refresh_set = time(nullptr);
1783 /* file operations run with G_PRIORITY_DEFAULT_IDLE */
1784 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE + 50, vf_refresh_idle_cb, vf, nullptr);
1786 else if (time(nullptr) - vf->time_refresh_set > 1)
1788 /* more than 1 sec since last update - increase priority */
1789 vf_refresh_idle_cancel(vf);
1790 vf->time_refresh_set = time(nullptr);
1791 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE - 50, vf_refresh_idle_cb, vf, nullptr);
1795 void vf_notify_cb(FileData *fd, NotifyType type, gpointer data)
1797 auto vf = static_cast<ViewFile *>(data);
1800 auto interested = static_cast<NotifyType>(NOTIFY_CHANGE | NOTIFY_REREAD | NOTIFY_GROUPING);
1801 if (vf->marks_enabled) interested = static_cast<NotifyType>(interested | NOTIFY_MARKS | NOTIFY_METADATA);
1802 /** @FIXME NOTIFY_METADATA should be checked by the keyword-to-mark functions and converted to NOTIFY_MARKS only if there was a change */
1804 if (!(type & interested) || vf->refresh_idle_id || !vf->dir_fd) return;
1806 refresh = (fd == vf->dir_fd);
1810 gchar *base = remove_level_from_path(fd->path);
1811 refresh = (g_strcmp0(base, vf->dir_fd->path) == 0);
1815 if ((type & NOTIFY_CHANGE) && fd->change)
1817 if (!refresh && fd->change->dest)
1819 gchar *dest_base = remove_level_from_path(fd->change->dest);
1820 refresh = (g_strcmp0(dest_base, vf->dir_fd->path) == 0);
1824 if (!refresh && fd->change->source)
1826 gchar *source_base = remove_level_from_path(fd->change->source);
1827 refresh = (g_strcmp0(source_base, vf->dir_fd->path) == 0);
1828 g_free(source_base);
1834 DEBUG_1("Notify vf: %s %04x", fd->path, type);
1835 vf_refresh_idle(vf);
1839 static gboolean vf_read_metadata_in_idle_cb(gpointer data)
1842 auto vf = static_cast<ViewFile *>(data);
1845 vf_thumb_status(vf, vf_read_metadata_in_idle_progress(vf), _("Loading meta..."));
1851 fd = static_cast<FileData *>(work->data);
1853 if (fd && !fd->metadata_in_idle_loaded)
1857 read_exif_time_data(fd);
1859 if (!fd->exifdate_digitized)
1861 read_exif_time_digitized_data(fd);
1863 if (fd->rating == STAR_RATING_NOT_READ)
1865 read_rating_data(fd);
1867 fd->metadata_in_idle_loaded = TRUE;
1868 return G_SOURCE_CONTINUE;
1873 vf_thumb_status(vf, 0.0, nullptr);
1874 vf->read_metadata_in_idle_id = 0;
1876 return G_SOURCE_REMOVE;
1879 static void vf_read_metadata_in_idle_finished_cb(gpointer data)
1881 auto vf = static_cast<ViewFile *>(data);
1883 vf_thumb_status(vf, 0.0, _("Loading meta..."));
1884 vf->read_metadata_in_idle_id = 0;
1887 void vf_read_metadata_in_idle(ViewFile *vf)
1891 if (vf->read_metadata_in_idle_id)
1893 g_idle_remove_by_data(vf);
1895 vf->read_metadata_in_idle_id = 0;
1899 vf->read_metadata_in_idle_id = g_idle_add_full(G_PRIORITY_LOW, vf_read_metadata_in_idle_cb, vf, vf_read_metadata_in_idle_finished_cb);
1903 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */