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.
22 #include "view-file.h"
25 #include "history-list.h"
29 #include "pixbuf-util.h"
32 #include "ui-fileops.h"
35 #include "view-file/view-file-list.h"
36 #include "view-file/view-file-icon.h"
40 *-----------------------------------------------------------------------------
42 *-----------------------------------------------------------------------------
45 void vf_send_update(ViewFile *vf)
47 if (vf->func_status) vf->func_status(vf, vf->data_status);
51 *-----------------------------------------------------------------------------
53 *-----------------------------------------------------------------------------
56 void vf_sort_set(ViewFile *vf, SortType type, gboolean ascend)
60 case FILEVIEW_LIST: vflist_sort_set(vf, type, ascend); break;
61 case FILEVIEW_ICON: vficon_sort_set(vf, type, ascend); break;
66 *-----------------------------------------------------------------------------
68 *-----------------------------------------------------------------------------
71 FileData *vf_index_get_data(ViewFile *vf, gint row)
73 return static_cast<FileData *>(g_list_nth_data(vf->list, row));
76 gint vf_index_by_fd(ViewFile *vf, FileData *fd)
82 case FILEVIEW_LIST: ret = vflist_index_by_fd(vf, fd); break;
83 case FILEVIEW_ICON: ret = vficon_index_by_fd(vf, fd); break;
90 guint vf_count(ViewFile *vf, gint64 *bytes)
100 auto fd = static_cast<FileData *>(work->data);
109 return g_list_length(vf->list);
112 GList *vf_get_list(ViewFile *vf)
114 GList *list = nullptr;
116 for (work = vf->list; work; work = work->next)
118 auto fd = static_cast<FileData *>(work->data);
119 list = g_list_prepend(list, file_data_ref(fd));
122 return g_list_reverse(list);
126 *-------------------------------------------------------------------
128 *-------------------------------------------------------------------
131 static gboolean vf_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
133 auto vf = static_cast<ViewFile *>(data);
138 case FILEVIEW_LIST: ret = vflist_press_key_cb(widget, event, data); break;
139 case FILEVIEW_ICON: ret = vficon_press_key_cb(widget, event, data); break;
140 default: ret = FALSE;
147 *-------------------------------------------------------------------
149 *-------------------------------------------------------------------
152 static gboolean vf_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
154 auto vf = static_cast<ViewFile *>(data);
159 case FILEVIEW_LIST: ret = vflist_press_cb(widget, bevent, data); break;
160 case FILEVIEW_ICON: ret = vficon_press_cb(widget, bevent, data); break;
161 default: ret = FALSE;
167 static gboolean vf_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
169 auto vf = static_cast<ViewFile *>(data);
174 case FILEVIEW_LIST: ret = vflist_release_cb(widget, bevent, data); break;
175 case FILEVIEW_ICON: ret = vficon_release_cb(widget, bevent, data); break;
176 default: ret = FALSE;
184 *-----------------------------------------------------------------------------
186 *-----------------------------------------------------------------------------
189 guint vf_selection_count(ViewFile *vf, gint64 *bytes)
195 case FILEVIEW_LIST: ret = vflist_selection_count(vf, bytes); break;
196 case FILEVIEW_ICON: ret = vficon_selection_count(vf, bytes); break;
203 GList *vf_selection_get_list(ViewFile *vf)
209 case FILEVIEW_LIST: ret = vflist_selection_get_list(vf); break;
210 case FILEVIEW_ICON: ret = vficon_selection_get_list(vf); break;
211 default: ret = nullptr;
217 GList *vf_selection_get_list_by_index(ViewFile *vf)
223 case FILEVIEW_LIST: ret = vflist_selection_get_list_by_index(vf); break;
224 case FILEVIEW_ICON: ret = vficon_selection_get_list_by_index(vf); break;
225 default: ret = nullptr;
231 void vf_select_all(ViewFile *vf)
235 case FILEVIEW_LIST: vflist_select_all(vf); break;
236 case FILEVIEW_ICON: vficon_select_all(vf); break;
240 void vf_select_none(ViewFile *vf)
244 case FILEVIEW_LIST: vflist_select_none(vf); break;
245 case FILEVIEW_ICON: vficon_select_none(vf); break;
249 void vf_select_invert(ViewFile *vf)
253 case FILEVIEW_LIST: vflist_select_invert(vf); break;
254 case FILEVIEW_ICON: vficon_select_invert(vf); break;
258 void vf_select_by_fd(ViewFile *vf, FileData *fd)
262 case FILEVIEW_LIST: vflist_select_by_fd(vf, fd); break;
263 case FILEVIEW_ICON: vficon_select_by_fd(vf, fd); break;
267 void vf_select_list(ViewFile *vf, GList *list)
271 case FILEVIEW_LIST: vflist_select_list(vf, list); break;
272 case FILEVIEW_ICON: vficon_select_list(vf, list); break;
276 void vf_mark_to_selection(ViewFile *vf, gint mark, MarkToSelectionMode mode)
280 case FILEVIEW_LIST: vflist_mark_to_selection(vf, mark, mode); break;
281 case FILEVIEW_ICON: vficon_mark_to_selection(vf, mark, mode); break;
285 void vf_selection_to_mark(ViewFile *vf, gint mark, SelectionToMarkMode mode)
289 case FILEVIEW_LIST: vflist_selection_to_mark(vf, mark, mode); break;
290 case FILEVIEW_ICON: vficon_selection_to_mark(vf, mark, mode); break;
295 *-----------------------------------------------------------------------------
297 *-----------------------------------------------------------------------------
301 static void vf_dnd_init(ViewFile *vf)
305 case FILEVIEW_LIST: vflist_dnd_init(vf); break;
306 case FILEVIEW_ICON: vficon_dnd_init(vf); break;
311 *-----------------------------------------------------------------------------
313 *-----------------------------------------------------------------------------
316 GList *vf_pop_menu_file_list(ViewFile *vf)
322 case FILEVIEW_LIST: ret = vflist_pop_menu_file_list(vf); break;
323 case FILEVIEW_ICON: ret = vficon_pop_menu_file_list(vf); break;
324 default: ret = nullptr;
330 GList *vf_selection_get_one(ViewFile *vf, FileData *fd)
336 case FILEVIEW_LIST: ret = vflist_selection_get_one(vf, fd); break;
337 case FILEVIEW_ICON: ret = vficon_selection_get_one(vf, fd); break;
338 default: ret = nullptr;
344 static void vf_pop_menu_edit_cb(GtkWidget *widget, gpointer data)
347 auto key = static_cast<const gchar *>(data);
349 vf = static_cast<ViewFile *>(submenu_item_get_data(widget));
353 file_util_start_editor_from_filelist(key, vf_pop_menu_file_list(vf), vf->dir_fd->path, vf->listview);
356 static void vf_pop_menu_view_cb(GtkWidget *widget, gpointer data)
358 auto vf = static_cast<ViewFile *>(data);
362 case FILEVIEW_LIST: vflist_pop_menu_view_cb(widget, data); break;
363 case FILEVIEW_ICON: vficon_pop_menu_view_cb(widget, data); break;
367 static void vf_pop_menu_open_archive_cb(GtkWidget *UNUSED(widget), gpointer data)
369 auto vf = static_cast<ViewFile *>(data);
370 LayoutWindow *lw_new;
371 FileData *fd = nullptr;
377 fd = (VFLIST(vf)->click_fd);
380 fd = (VFICON(vf)->click_fd);
384 dest_dir = open_archive(fd);
387 lw_new = layout_new_from_default();
388 layout_set_path(lw_new, dest_dir);
393 warning_dialog(_("Cannot open archive file"), _("See the Log Window"), GTK_STOCK_DIALOG_WARNING, nullptr);
397 static void vf_pop_menu_copy_cb(GtkWidget *UNUSED(widget), gpointer data)
399 auto vf = static_cast<ViewFile *>(data);
401 file_util_copy(nullptr, vf_pop_menu_file_list(vf), nullptr, vf->listview);
404 static void vf_pop_menu_move_cb(GtkWidget *UNUSED(widget), gpointer data)
406 auto vf = static_cast<ViewFile *>(data);
408 file_util_move(nullptr, vf_pop_menu_file_list(vf), nullptr, vf->listview);
411 static void vf_pop_menu_rename_cb(GtkWidget *widget, gpointer data)
413 auto vf = static_cast<ViewFile *>(data);
417 case FILEVIEW_LIST: vflist_pop_menu_rename_cb(widget, data); break;
418 case FILEVIEW_ICON: vficon_pop_menu_rename_cb(widget, data); break;
422 static void vf_pop_menu_delete_cb(GtkWidget *UNUSED(widget), gpointer data)
424 auto vf = static_cast<ViewFile *>(data);
426 options->file_ops.safe_delete_enable = FALSE;
427 file_util_delete(nullptr, vf_pop_menu_file_list(vf), vf->listview);
430 static void vf_pop_menu_move_to_trash_cb(GtkWidget *UNUSED(widget), gpointer data)
432 auto vf = static_cast<ViewFile *>(data);
434 options->file_ops.safe_delete_enable = TRUE;
435 file_util_delete(nullptr, vf_pop_menu_file_list(vf), vf->listview);
438 static void vf_pop_menu_copy_path_cb(GtkWidget *UNUSED(widget), gpointer data)
440 auto vf = static_cast<ViewFile *>(data);
442 file_util_copy_path_list_to_clipboard(vf_pop_menu_file_list(vf), TRUE);
445 static void vf_pop_menu_copy_path_unquoted_cb(GtkWidget *UNUSED(widget), gpointer data)
447 auto vf = static_cast<ViewFile *>(data);
449 file_util_copy_path_list_to_clipboard(vf_pop_menu_file_list(vf), FALSE);
452 static void vf_pop_menu_enable_grouping_cb(GtkWidget *UNUSED(widget), gpointer data)
454 auto vf = static_cast<ViewFile *>(data);
456 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), FALSE);
459 static void vf_pop_menu_duplicates_cb(GtkWidget *UNUSED(widget), gpointer data)
461 auto vf = static_cast<ViewFile *>(data);
464 dw = dupe_window_new();
465 dupe_window_add_files(dw, vf_pop_menu_file_list(vf), FALSE);
468 static void vf_pop_menu_disable_grouping_cb(GtkWidget *UNUSED(widget), gpointer data)
470 auto vf = static_cast<ViewFile *>(data);
472 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), TRUE);
475 static void vf_pop_menu_sort_cb(GtkWidget *widget, gpointer data)
480 if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget))) return;
482 vf = static_cast<ViewFile *>(submenu_item_get_data(widget));
485 type = static_cast<SortType>GPOINTER_TO_INT(data);
487 if (type == SORT_EXIFTIME || type == SORT_EXIFTIMEDIGITIZED || type == SORT_RATING)
489 vf_read_metadata_in_idle(vf);
494 layout_sort_set(vf->layout, type, vf->sort_ascend);
498 vf_sort_set(vf, type, vf->sort_ascend);
502 static void vf_pop_menu_sort_ascend_cb(GtkWidget *UNUSED(widget), gpointer data)
504 auto vf = static_cast<ViewFile *>(data);
508 layout_sort_set(vf->layout, vf->sort_method, !vf->sort_ascend);
512 vf_sort_set(vf, vf->sort_method, !vf->sort_ascend);
516 static void vf_pop_menu_sel_mark_cb(GtkWidget *UNUSED(widget), gpointer data)
518 auto vf = static_cast<ViewFile *>(data);
519 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_SET);
522 static void vf_pop_menu_sel_mark_and_cb(GtkWidget *UNUSED(widget), gpointer data)
524 auto vf = static_cast<ViewFile *>(data);
525 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_AND);
528 static void vf_pop_menu_sel_mark_or_cb(GtkWidget *UNUSED(widget), gpointer data)
530 auto vf = static_cast<ViewFile *>(data);
531 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_OR);
534 static void vf_pop_menu_sel_mark_minus_cb(GtkWidget *UNUSED(widget), gpointer data)
536 auto vf = static_cast<ViewFile *>(data);
537 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_MINUS);
540 static void vf_pop_menu_set_mark_sel_cb(GtkWidget *UNUSED(widget), gpointer data)
542 auto vf = static_cast<ViewFile *>(data);
543 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_SET);
546 static void vf_pop_menu_res_mark_sel_cb(GtkWidget *UNUSED(widget), gpointer data)
548 auto vf = static_cast<ViewFile *>(data);
549 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_RESET);
552 static void vf_pop_menu_toggle_mark_sel_cb(GtkWidget *UNUSED(widget), gpointer data)
554 auto vf = static_cast<ViewFile *>(data);
555 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_TOGGLE);
558 static void vf_pop_menu_toggle_view_type_cb(GtkWidget *widget, gpointer data)
560 auto vf = static_cast<ViewFile *>(data);
561 auto new_type = static_cast<FileViewType>(GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), "menu_item_radio_data")));
562 if (!vf->layout) return;
564 layout_views_set(vf->layout, vf->layout->options.dir_view_type, new_type);
567 static void vf_pop_menu_refresh_cb(GtkWidget *widget, gpointer data)
569 auto vf = static_cast<ViewFile *>(data);
573 case FILEVIEW_LIST: vflist_pop_menu_refresh_cb(widget, data); break;
574 case FILEVIEW_ICON: vficon_pop_menu_refresh_cb(widget, data); break;
578 static void vf_popup_destroy_cb(GtkWidget *widget, gpointer data)
580 auto vf = static_cast<ViewFile *>(data);
584 case FILEVIEW_LIST: vflist_popup_destroy_cb(widget, data); break;
585 case FILEVIEW_ICON: vficon_popup_destroy_cb(widget, data); break;
588 filelist_free(vf->editmenu_fd_list);
589 vf->editmenu_fd_list = nullptr;
593 * @brief Add file selection list to a collection
595 * @param[in] data Index to the collection list menu item selected, or -1 for new collection
599 static void vf_pop_menu_collections_cb(GtkWidget *widget, gpointer data)
602 GList *selection_list;
604 vf = static_cast<ViewFile *>(submenu_item_get_data(widget));
605 selection_list = vf_selection_get_list(vf);
606 pop_menu_collections(selection_list, data);
608 filelist_free(selection_list);
611 GtkWidget *vf_pop_menu(ViewFile *vf)
616 gboolean active = FALSE;
617 gboolean class_archive = FALSE;
618 GtkAccelGroup *accel_group;
623 vflist_color_set(vf, VFLIST(vf)->click_fd, TRUE);
624 active = (VFLIST(vf)->click_fd != nullptr);
625 class_archive = (VFLIST(vf)->click_fd != nullptr && VFLIST(vf)->click_fd->format_class == FORMAT_CLASS_ARCHIVE);
628 active = (VFICON(vf)->click_fd != nullptr);
629 class_archive = (VFICON(vf)->click_fd != nullptr && VFICON(vf)->click_fd->format_class == FORMAT_CLASS_ARCHIVE);
633 menu = popup_menu_short_lived();
635 accel_group = gtk_accel_group_new();
636 gtk_menu_set_accel_group(GTK_MENU(menu), accel_group);
638 g_object_set_data(G_OBJECT(menu), "window_keys", nullptr);
639 g_object_set_data(G_OBJECT(menu), "accel_group", accel_group);
641 g_signal_connect(G_OBJECT(menu), "destroy",
642 G_CALLBACK(vf_popup_destroy_cb), vf);
644 if (vf->clicked_mark > 0)
646 gint mark = vf->clicked_mark;
647 gchar *str_set_mark = g_strdup_printf(_("_Set mark %d"), mark);
648 gchar *str_res_mark = g_strdup_printf(_("_Reset mark %d"), mark);
649 gchar *str_toggle_mark = g_strdup_printf(_("_Toggle mark %d"), mark);
650 gchar *str_sel_mark = g_strdup_printf(_("_Select mark %d"), mark);
651 gchar *str_sel_mark_or = g_strdup_printf(_("_Add mark %d"), mark);
652 gchar *str_sel_mark_and = g_strdup_printf(_("_Intersection with mark %d"), mark);
653 gchar *str_sel_mark_minus = g_strdup_printf(_("_Unselect mark %d"), mark);
655 g_assert(mark >= 1 && mark <= FILEDATA_MARKS_SIZE);
657 vf->active_mark = mark;
658 vf->clicked_mark = 0;
660 menu_item_add_sensitive(menu, str_set_mark, active,
661 G_CALLBACK(vf_pop_menu_set_mark_sel_cb), vf);
663 menu_item_add_sensitive(menu, str_res_mark, active,
664 G_CALLBACK(vf_pop_menu_res_mark_sel_cb), vf);
666 menu_item_add_sensitive(menu, str_toggle_mark, active,
667 G_CALLBACK(vf_pop_menu_toggle_mark_sel_cb), vf);
669 menu_item_add_divider(menu);
671 menu_item_add_sensitive(menu, str_sel_mark, active,
672 G_CALLBACK(vf_pop_menu_sel_mark_cb), vf);
673 menu_item_add_sensitive(menu, str_sel_mark_or, active,
674 G_CALLBACK(vf_pop_menu_sel_mark_or_cb), vf);
675 menu_item_add_sensitive(menu, str_sel_mark_and, active,
676 G_CALLBACK(vf_pop_menu_sel_mark_and_cb), vf);
677 menu_item_add_sensitive(menu, str_sel_mark_minus, active,
678 G_CALLBACK(vf_pop_menu_sel_mark_minus_cb), vf);
680 menu_item_add_divider(menu);
682 g_free(str_set_mark);
683 g_free(str_res_mark);
684 g_free(str_toggle_mark);
685 g_free(str_sel_mark);
686 g_free(str_sel_mark_and);
687 g_free(str_sel_mark_or);
688 g_free(str_sel_mark_minus);
691 vf->editmenu_fd_list = vf_pop_menu_file_list(vf);
692 submenu_add_edit(menu, &item, G_CALLBACK(vf_pop_menu_edit_cb), vf, vf->editmenu_fd_list);
693 gtk_widget_set_sensitive(item, active);
695 menu_item_add_stock_sensitive(menu, _("View in _new window"), GTK_STOCK_NEW, active,
696 G_CALLBACK(vf_pop_menu_view_cb), vf);
698 menu_item_add_stock_sensitive(menu, _("Open archive"), GTK_STOCK_OPEN, active & class_archive, G_CALLBACK(vf_pop_menu_open_archive_cb), vf);
700 menu_item_add_divider(menu);
701 menu_item_add_stock_sensitive(menu, _("_Copy..."), GTK_STOCK_COPY, active,
702 G_CALLBACK(vf_pop_menu_copy_cb), vf);
703 menu_item_add_sensitive(menu, _("_Move..."), active,
704 G_CALLBACK(vf_pop_menu_move_cb), vf);
705 menu_item_add_sensitive(menu, _("_Rename..."), active,
706 G_CALLBACK(vf_pop_menu_rename_cb), vf);
707 menu_item_add_sensitive(menu, _("_Copy path to clipboard"), active,
708 G_CALLBACK(vf_pop_menu_copy_path_cb), vf);
709 menu_item_add_sensitive(menu, _("_Copy path unquoted to clipboard"), active,
710 G_CALLBACK(vf_pop_menu_copy_path_unquoted_cb), vf);
711 menu_item_add_divider(menu);
712 menu_item_add_stock_sensitive(menu,
713 options->file_ops.confirm_move_to_trash ? _("Move to Trash...") :
714 _("Move to Trash"), PIXBUF_INLINE_ICON_TRASH, active,
715 G_CALLBACK(vf_pop_menu_move_to_trash_cb), vf);
716 menu_item_add_stock_sensitive(menu,
717 options->file_ops.confirm_delete ? _("_Delete...") :
718 _("_Delete"), GTK_STOCK_DELETE, active,
719 G_CALLBACK(vf_pop_menu_delete_cb), vf);
720 menu_item_add_divider(menu);
722 menu_item_add_sensitive(menu, _("Enable file _grouping"), active,
723 G_CALLBACK(vf_pop_menu_enable_grouping_cb), vf);
724 menu_item_add_sensitive(menu, _("Disable file groupi_ng"), active,
725 G_CALLBACK(vf_pop_menu_disable_grouping_cb), vf);
727 menu_item_add_divider(menu);
728 menu_item_add_stock_sensitive(menu, _("_Find duplicates..."), GTK_STOCK_FIND, active,
729 G_CALLBACK(vf_pop_menu_duplicates_cb), vf);
730 menu_item_add_divider(menu);
732 submenu = submenu_add_collections(menu, &item,
733 G_CALLBACK(vf_pop_menu_collections_cb), vf);
734 gtk_widget_set_sensitive(item, active);
735 menu_item_add_divider(menu);
737 submenu = submenu_add_sort(nullptr, G_CALLBACK(vf_pop_menu_sort_cb), vf,
738 FALSE, FALSE, TRUE, vf->sort_method);
739 menu_item_add_divider(submenu);
740 menu_item_add_check(submenu, _("Ascending"), vf->sort_ascend,
741 G_CALLBACK(vf_pop_menu_sort_ascend_cb), vf);
743 item = menu_item_add(menu, _("_Sort"), nullptr, nullptr);
744 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
746 item = menu_item_add_radio(menu, _("Images as List"), GINT_TO_POINTER(FILEVIEW_LIST), vf->type == FILEVIEW_LIST,
747 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
749 item = menu_item_add_radio(menu, _("Images as Icons"), GINT_TO_POINTER(FILEVIEW_ICON), vf->type == FILEVIEW_ICON,
750 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
755 menu_item_add_check(menu, _("Show _thumbnails"), VFLIST(vf)->thumbs_enabled,
756 G_CALLBACK(vflist_pop_menu_thumbs_cb), vf);
759 menu_item_add_check(menu, _("Show filename _text"), VFICON(vf)->show_text,
760 G_CALLBACK(vficon_pop_menu_show_names_cb), vf);
767 menu_item_add_check(menu, _("Show star rating"), options->show_star_rating,
768 G_CALLBACK(vflist_pop_menu_show_star_rating_cb), vf);
771 menu_item_add_check(menu, _("Show star rating"), options->show_star_rating,
772 G_CALLBACK(vficon_pop_menu_show_star_rating_cb), vf);
776 menu_item_add_stock(menu, _("Re_fresh"), GTK_STOCK_REFRESH, G_CALLBACK(vf_pop_menu_refresh_cb), vf);
781 gboolean vf_refresh(ViewFile *vf)
787 case FILEVIEW_LIST: ret = vflist_refresh(vf); break;
788 case FILEVIEW_ICON: ret = vficon_refresh(vf); break;
789 default: ret = FALSE;
795 gboolean vf_set_fd(ViewFile *vf, FileData *dir_fd)
801 case FILEVIEW_LIST: ret = vflist_set_fd(vf, dir_fd); break;
802 case FILEVIEW_ICON: ret = vficon_set_fd(vf, dir_fd); break;
803 default: ret = FALSE;
809 static void vf_destroy_cb(GtkWidget *widget, gpointer data)
811 auto vf = static_cast<ViewFile *>(data);
815 case FILEVIEW_LIST: vflist_destroy_cb(widget, data); break;
816 case FILEVIEW_ICON: vficon_destroy_cb(widget, data); break;
821 g_signal_handlers_disconnect_matched(G_OBJECT(vf->popup), G_SIGNAL_MATCH_DATA,
822 0, 0, nullptr, nullptr, vf);
823 gtk_widget_destroy(vf->popup);
826 if (vf->read_metadata_in_idle_id)
828 g_idle_remove_by_data(vf);
830 file_data_unref(vf->dir_fd);
835 static void vf_marks_filter_toggle_cb(GtkWidget *UNUSED(widget), gpointer data)
837 auto vf = static_cast<ViewFile *>(data);
841 struct MarksTextEntry {
844 GtkWidget *edit_widget;
849 static void vf_marks_tooltip_cancel_cb(GenericDialog *gd, gpointer data)
851 auto mte = static_cast<MarksTextEntry *>(data);
853 g_free(mte->text_entry);
854 generic_dialog_close(gd);
857 static void vf_marks_tooltip_ok_cb(GenericDialog *gd, gpointer data)
859 auto mte = static_cast<MarksTextEntry *>(data);
861 g_free(options->marks_tooltips[mte->mark_no]);
862 options->marks_tooltips[mte->mark_no] = g_strdup(gtk_entry_get_text(GTK_ENTRY(mte->edit_widget)));
864 gtk_widget_set_tooltip_text(mte->parent, options->marks_tooltips[mte->mark_no]);
866 g_free(mte->text_entry);
867 generic_dialog_close(gd);
870 void vf_marks_filter_on_icon_press(GtkEntry *UNUSED(entry), GtkEntryIconPosition UNUSED(pos),
871 GdkEvent *UNUSED(event), gpointer userdata)
873 auto mte = static_cast<MarksTextEntry *>(userdata);
875 g_free(mte->text_entry);
876 mte->text_entry = g_strdup("");
877 gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), "");
880 static void vf_marks_tooltip_help_cb(GenericDialog *UNUSED(gd), gpointer UNUSED(data))
882 help_window_show("GuideImageMarks.html");
885 static gboolean vf_marks_tooltip_cb(GtkWidget *widget,
886 GdkEventButton *event,
890 gint i = GPOINTER_TO_INT(user_data);
892 if (event->button != MOUSE_BUTTON_RIGHT)
895 auto mte = g_new0(MarksTextEntry, 1);
897 mte->text_entry = g_strdup(options->marks_tooltips[i]);
898 mte->parent = widget;
900 mte->gd = generic_dialog_new(_("Mark text"), "mark_text",
902 vf_marks_tooltip_cancel_cb, mte);
903 generic_dialog_add_message(mte->gd, GTK_STOCK_DIALOG_QUESTION, _("Set mark text"),
904 _("This will set or clear the mark text."), FALSE);
905 generic_dialog_add_button(mte->gd, GTK_STOCK_OK, nullptr,
906 vf_marks_tooltip_ok_cb, TRUE);
907 generic_dialog_add_button(mte->gd, GTK_STOCK_HELP, nullptr,
908 vf_marks_tooltip_help_cb, FALSE);
910 table = pref_table_new(mte->gd->vbox, 3, 1, FALSE, TRUE);
911 pref_table_label(table, 0, 0, g_strdup_printf("%s%d", _("Mark "), mte->mark_no + 1), 1.0);
912 mte->edit_widget = gtk_entry_new();
913 gtk_widget_set_size_request(mte->edit_widget, 300, -1);
916 gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), mte->text_entry);
918 gtk_table_attach_defaults(GTK_TABLE(table), mte->edit_widget, 1, 2, 0, 1);
919 generic_dialog_attach_default(mte->gd, mte->edit_widget);
921 gtk_entry_set_icon_from_icon_name(GTK_ENTRY(mte->edit_widget),
922 GTK_ENTRY_ICON_SECONDARY, "edit-clear");
923 gtk_entry_set_icon_tooltip_text(GTK_ENTRY(mte->edit_widget),
924 GTK_ENTRY_ICON_SECONDARY, "Clear");
925 g_signal_connect(GTK_ENTRY(mte->edit_widget), "icon-press",
926 G_CALLBACK(vf_marks_filter_on_icon_press), mte);
928 gtk_widget_show(mte->edit_widget);
929 gtk_widget_grab_focus(mte->edit_widget);
930 gtk_widget_show(GTK_WIDGET(mte->gd->dialog));
935 static void vf_file_filter_save_cb(GtkWidget *UNUSED(widget), gpointer data)
937 auto vf = static_cast<ViewFile *>(data);
939 gchar *remove_text = nullptr;
940 gchar *index_text = nullptr;
941 gboolean text_found = FALSE;
944 entry_text = g_strdup(gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo)))));
946 if (entry_text[0] == '\0' && vf->file_filter.last_selected >= 0)
948 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), vf->file_filter.last_selected);
949 remove_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
950 history_list_item_remove("file_filter", remove_text);
951 gtk_combo_box_text_remove(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), vf->file_filter.last_selected);
954 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), -1);
955 vf->file_filter.last_selected = - 1;
956 gtk_entry_set_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo))), "");
957 vf->file_filter.count--;
961 if (entry_text[0] != '\0')
963 for (i = 0; i < vf->file_filter.count; i++)
969 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), i);
970 index_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
972 if (g_strcmp0(index_text, entry_text) == 0)
982 history_list_add_to_key("file_filter", entry_text, 10);
983 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), entry_text);
984 vf->file_filter.count++;
985 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), vf->file_filter.count - 1);
994 static void vf_file_filter_cb(GtkWidget *UNUSED(widget), gpointer data)
996 auto vf = static_cast<ViewFile *>(data);
1001 static gboolean vf_file_filter_press_cb(GtkWidget *widget, GdkEventButton *UNUSED(bevent), gpointer data)
1003 auto vf = static_cast<ViewFile *>(data);
1004 vf->file_filter.last_selected = gtk_combo_box_get_active(GTK_COMBO_BOX(vf->file_filter.combo));
1006 gtk_widget_grab_focus(widget);
1011 static GtkWidget *vf_marks_filter_init(ViewFile *vf)
1013 GtkWidget *frame = gtk_frame_new(nullptr);
1014 GtkWidget *hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
1018 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1020 GtkWidget *check = gtk_check_button_new();
1021 gtk_box_pack_start(GTK_BOX(hbox), check, FALSE, FALSE, 0);
1022 g_signal_connect(G_OBJECT(check), "toggled",
1023 G_CALLBACK(vf_marks_filter_toggle_cb), vf);
1024 g_signal_connect(G_OBJECT(check), "button_press_event",
1025 G_CALLBACK(vf_marks_tooltip_cb), GINT_TO_POINTER(i));
1026 gtk_widget_set_tooltip_text(check, options->marks_tooltips[i]);
1028 gtk_widget_show(check);
1029 vf->filter_check[i] = check;
1031 gtk_container_add(GTK_CONTAINER(frame), hbox);
1032 gtk_widget_show(hbox);
1036 void vf_file_filter_set(ViewFile *vf, gboolean enable)
1040 gtk_widget_show(vf->file_filter.combo);
1041 gtk_widget_show(vf->file_filter.frame);
1045 gtk_widget_hide(vf->file_filter.combo);
1046 gtk_widget_hide(vf->file_filter.frame);
1052 static gboolean vf_file_filter_class_cb(GtkWidget *widget, gpointer data)
1054 auto vf = static_cast<ViewFile *>(data);
1057 gboolean state = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
1059 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1061 if (g_strcmp0(format_class_list[i], gtk_menu_item_get_label(GTK_MENU_ITEM(widget))) == 0)
1063 options->class_filter[i] = state;
1071 static gboolean vf_file_filter_class_set_all_cb(GtkWidget *widget, gpointer data)
1073 auto vf = static_cast<ViewFile *>(data);
1080 if (g_strcmp0(_("Select all"), gtk_menu_item_get_label(GTK_MENU_ITEM(widget))) == 0)
1089 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1091 options->class_filter[i] = state;
1095 parent = gtk_widget_get_parent(widget);
1096 children = gtk_container_get_children(GTK_CONTAINER(parent));
1099 child = static_cast<GtkWidget *>(children->data);
1100 if (i < FILE_FORMAT_CLASSES)
1102 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(child), state);
1105 children = children->next;
1107 g_list_free(children);
1113 static GtkWidget *class_filter_menu (ViewFile *vf)
1116 GtkWidget *menu_item;
1119 menu = gtk_menu_new();
1121 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1123 menu_item = gtk_check_menu_item_new_with_label(format_class_list[i]);
1124 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menu_item), options->class_filter[i]);
1125 g_signal_connect(G_OBJECT(menu_item), "toggled", G_CALLBACK(vf_file_filter_class_cb), vf);
1126 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1127 gtk_widget_show(menu_item);
1130 menu_item = gtk_menu_item_new_with_label(_("Select all"));
1131 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1132 gtk_widget_show(menu_item);
1133 g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(vf_file_filter_class_set_all_cb), vf);
1135 menu_item = gtk_menu_item_new_with_label(_("Select none"));
1136 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1137 gtk_widget_show(menu_item);
1138 g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(vf_file_filter_class_set_all_cb), vf);
1143 static void case_sensitive_cb(GtkWidget *widget, gpointer data)
1145 auto vf = static_cast<ViewFile *>(data);
1147 vf->file_filter.case_sensitive = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
1151 static void file_filter_clear_cb(GtkEntry *UNUSED(entry), GtkEntryIconPosition pos, GdkEvent *UNUSED(event), gpointer userdata)
1153 if (pos == GTK_ENTRY_ICON_SECONDARY)
1155 gtk_entry_set_text(GTK_ENTRY(userdata), "");
1156 gtk_widget_grab_focus(GTK_WIDGET(userdata));
1160 static GtkWidget *vf_file_filter_init(ViewFile *vf)
1162 GtkWidget *frame = gtk_frame_new(nullptr);
1163 GtkWidget *hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
1166 GtkWidget *combo_entry;
1168 GtkWidget *menuitem;
1169 GtkWidget *case_sensitive;
1174 vf->file_filter.combo = gtk_combo_box_text_new_with_entry();
1175 combo_entry = gtk_bin_get_child(GTK_BIN(vf->file_filter.combo));
1176 gtk_widget_show(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo)));
1177 gtk_widget_show((GTK_WIDGET(vf->file_filter.combo)));
1178 gtk_widget_set_tooltip_text(GTK_WIDGET(vf->file_filter.combo), _("Use regular expressions"));
1180 gtk_entry_set_icon_from_icon_name(GTK_ENTRY(combo_entry), GTK_ENTRY_ICON_SECONDARY, "edit-clear");
1181 gtk_entry_set_icon_tooltip_text (GTK_ENTRY(combo_entry), GTK_ENTRY_ICON_SECONDARY, _("Clear"));
1182 g_signal_connect(GTK_ENTRY(combo_entry), "icon-press", G_CALLBACK(file_filter_clear_cb), combo_entry);
1184 work = history_list_get_by_key("file_filter");
1187 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), static_cast<gchar *>(work->data));
1190 vf->file_filter.count = n;
1192 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), 0);
1194 g_signal_connect(G_OBJECT(combo_entry), "activate",
1195 G_CALLBACK(vf_file_filter_save_cb), vf);
1197 g_signal_connect(G_OBJECT(vf->file_filter.combo), "changed",
1198 G_CALLBACK(vf_file_filter_cb), vf);
1200 g_signal_connect(G_OBJECT(combo_entry), "button_press_event",
1201 G_CALLBACK(vf_file_filter_press_cb), vf);
1203 gtk_box_pack_start(GTK_BOX(hbox), vf->file_filter.combo, FALSE, FALSE, 0);
1204 gtk_widget_show(vf->file_filter.combo);
1205 gtk_container_add(GTK_CONTAINER(frame), hbox);
1206 gtk_widget_show(hbox);
1208 case_sensitive = gtk_check_button_new_with_label(_("Case"));
1209 gtk_box_pack_start(GTK_BOX(hbox), case_sensitive, FALSE, FALSE, 0);
1210 gtk_widget_set_tooltip_text(GTK_WIDGET(case_sensitive), _("Case sensitive"));
1211 g_signal_connect(G_OBJECT(case_sensitive), "clicked", G_CALLBACK(case_sensitive_cb), vf);
1212 gtk_widget_show(case_sensitive);
1214 menubar = gtk_menu_bar_new();
1215 gtk_box_pack_start(GTK_BOX(hbox), menubar, FALSE, TRUE, 0);
1216 gtk_widget_show(menubar);
1218 box = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, PREF_PAD_GAP);
1219 icon = gtk_image_new_from_icon_name("pan-down", GTK_ICON_SIZE_MENU);
1220 label = gtk_label_new(_("Class"));
1222 gtk_box_pack_start(GTK_BOX(box), label, FALSE, FALSE, 0);
1223 gtk_box_pack_end(GTK_BOX(box), icon, FALSE, FALSE, 0);
1225 menuitem = gtk_menu_item_new();
1227 gtk_widget_set_tooltip_text(GTK_WIDGET(menuitem), _("Select Class filter"));
1228 gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), class_filter_menu(vf));
1229 gtk_menu_shell_append(GTK_MENU_SHELL(menubar), menuitem);
1230 gtk_container_add(GTK_CONTAINER(menuitem), box);
1231 gtk_widget_show_all(menuitem);
1236 void vf_mark_filter_toggle(ViewFile *vf, gint mark)
1239 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(vf->filter_check[n]),
1240 !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[n])));
1243 ViewFile *vf_new(FileViewType type, FileData *dir_fd)
1247 vf = g_new0(ViewFile, 1);
1250 vf->sort_method = SORT_NAME;
1251 vf->sort_ascend = TRUE;
1252 vf->read_metadata_in_idle_id = 0;
1254 vf->scrolled = gtk_scrolled_window_new(nullptr, nullptr);
1255 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vf->scrolled), GTK_SHADOW_IN);
1256 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vf->scrolled),
1257 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1259 vf->filter = vf_marks_filter_init(vf);
1260 vf->file_filter.frame = vf_file_filter_init(vf);
1262 vf->widget = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
1263 gtk_box_pack_start(GTK_BOX(vf->widget), vf->filter, FALSE, FALSE, 0);
1264 gtk_box_pack_start(GTK_BOX(vf->widget), vf->file_filter.frame, FALSE, FALSE, 0);
1265 gtk_box_pack_start(GTK_BOX(vf->widget), vf->scrolled, TRUE, TRUE, 0);
1266 gtk_widget_show(vf->scrolled);
1268 g_signal_connect(G_OBJECT(vf->widget), "destroy",
1269 G_CALLBACK(vf_destroy_cb), vf);
1273 case FILEVIEW_LIST: vf = vflist_new(vf, dir_fd); break;
1274 case FILEVIEW_ICON: vf = vficon_new(vf, dir_fd); break;
1279 g_signal_connect(G_OBJECT(vf->listview), "key_press_event",
1280 G_CALLBACK(vf_press_key_cb), vf);
1281 g_signal_connect(G_OBJECT(vf->listview), "button_press_event",
1282 G_CALLBACK(vf_press_cb), vf);
1283 g_signal_connect(G_OBJECT(vf->listview), "button_release_event",
1284 G_CALLBACK(vf_release_cb), vf);
1286 gtk_container_add(GTK_CONTAINER(vf->scrolled), vf->listview);
1287 gtk_widget_show(vf->listview);
1289 if (dir_fd) vf_set_fd(vf, dir_fd);
1294 void vf_set_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gpointer data), gpointer data)
1296 vf->func_status = func;
1297 vf->data_status = data;
1300 void vf_set_thumb_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gdouble val, const gchar *text, gpointer data), gpointer data)
1302 vf->func_thumb_status = func;
1303 vf->data_thumb_status = data;
1306 void vf_thumb_set(ViewFile *vf, gboolean enable)
1310 case FILEVIEW_LIST: vflist_thumb_set(vf, enable); break;
1311 case FILEVIEW_ICON: /*vficon_thumb_set(vf, enable);*/ break;
1316 static gboolean vf_thumb_next(ViewFile *vf);
1318 static gdouble vf_thumb_progress(ViewFile *vf)
1325 case FILEVIEW_LIST: vflist_thumb_progress_count(vf->list, &count, &done); break;
1326 case FILEVIEW_ICON: vficon_thumb_progress_count(vf->list, &count, &done); break;
1329 DEBUG_1("thumb progress: %d of %d", done, count);
1330 return static_cast<gdouble>(done) / count;
1333 static gdouble vf_read_metadata_in_idle_progress(ViewFile *vf)
1340 case FILEVIEW_LIST: vflist_read_metadata_progress_count(vf->list, &count, &done); break;
1341 case FILEVIEW_ICON: vficon_read_metadata_progress_count(vf->list, &count, &done); break;
1344 return static_cast<gdouble>(done) / count;
1347 static void vf_set_thumb_fd(ViewFile *vf, FileData *fd)
1351 case FILEVIEW_LIST: vflist_set_thumb_fd(vf, fd); break;
1352 case FILEVIEW_ICON: vficon_set_thumb_fd(vf, fd); break;
1356 static void vf_thumb_status(ViewFile *vf, gdouble val, const gchar *text)
1358 if (vf->func_thumb_status)
1360 vf->func_thumb_status(vf, val, text, vf->data_thumb_status);
1364 static void vf_thumb_do(ViewFile *vf, FileData *fd)
1368 vf_set_thumb_fd(vf, fd);
1369 vf_thumb_status(vf, vf_thumb_progress(vf), _("Loading thumbs..."));
1372 void vf_thumb_cleanup(ViewFile *vf)
1374 vf_thumb_status(vf, 0.0, nullptr);
1376 vf->thumbs_running = FALSE;
1378 thumb_loader_free(vf->thumbs_loader);
1379 vf->thumbs_loader = nullptr;
1381 vf->thumbs_filedata = nullptr;
1384 void vf_thumb_stop(ViewFile *vf)
1386 if (vf->thumbs_running) vf_thumb_cleanup(vf);
1389 static void vf_thumb_common_cb(ThumbLoader *tl, gpointer data)
1391 auto vf = static_cast<ViewFile *>(data);
1393 if (vf->thumbs_filedata && vf->thumbs_loader == tl)
1395 vf_thumb_do(vf, vf->thumbs_filedata);
1398 while (vf_thumb_next(vf));
1401 static void vf_thumb_error_cb(ThumbLoader *tl, gpointer data)
1403 vf_thumb_common_cb(tl, data);
1406 static void vf_thumb_done_cb(ThumbLoader *tl, gpointer data)
1408 vf_thumb_common_cb(tl, data);
1411 static gboolean vf_thumb_next(ViewFile *vf)
1413 FileData *fd = nullptr;
1415 if (!gtk_widget_get_realized(vf->listview))
1417 vf_thumb_status(vf, 0.0, nullptr);
1423 case FILEVIEW_LIST: fd = vflist_thumb_next_fd(vf); break;
1424 case FILEVIEW_ICON: fd = vficon_thumb_next_fd(vf); break;
1430 vf_thumb_cleanup(vf);
1434 vf->thumbs_filedata = fd;
1436 thumb_loader_free(vf->thumbs_loader);
1438 vf->thumbs_loader = thumb_loader_new(options->thumbnails.max_width, options->thumbnails.max_height);
1439 thumb_loader_set_callbacks(vf->thumbs_loader,
1445 if (!thumb_loader_start(vf->thumbs_loader, fd))
1447 /* set icon to unknown, continue */
1448 DEBUG_1("thumb loader start failed %s", fd->path);
1449 vf_thumb_do(vf, fd);
1457 static void vf_thumb_reset_all(ViewFile *vf)
1461 for (work = vf->list; work; work = work->next)
1463 auto fd = static_cast<FileData *>(work->data);
1464 if (fd->thumb_pixbuf)
1466 g_object_unref(fd->thumb_pixbuf);
1467 fd->thumb_pixbuf = nullptr;
1472 void vf_thumb_update(ViewFile *vf)
1476 if (vf->type == FILEVIEW_LIST && !VFLIST(vf)->thumbs_enabled) return;
1478 vf_thumb_status(vf, 0.0, _("Loading thumbs..."));
1479 vf->thumbs_running = TRUE;
1481 if (thumb_format_changed)
1483 vf_thumb_reset_all(vf);
1484 thumb_format_changed = FALSE;
1487 while (vf_thumb_next(vf));
1490 void vf_star_cleanup(ViewFile *vf)
1492 if (vf->stars_id != 0)
1494 g_source_remove(vf->stars_id);
1498 vf->stars_filedata = nullptr;
1501 void vf_star_stop(ViewFile *vf)
1503 vf_star_cleanup(vf);
1506 static void vf_set_star_fd(ViewFile *vf, FileData *fd)
1510 case FILEVIEW_LIST: vflist_set_star_fd(vf, fd); break;
1511 case FILEVIEW_ICON: vficon_set_star_fd(vf, fd); break;
1516 static void vf_star_do(ViewFile *vf, FileData *fd)
1520 vf_set_star_fd(vf, fd);
1523 static gboolean vf_star_next(ViewFile *vf)
1525 FileData *fd = nullptr;
1529 case FILEVIEW_LIST: fd = vflist_star_next_fd(vf); break;
1530 case FILEVIEW_ICON: fd = vficon_star_next_fd(vf); break;
1537 vf_star_cleanup(vf);
1544 gboolean vf_stars_cb(gpointer data)
1546 auto vf = static_cast<ViewFile *>(data);
1547 FileData *fd = vf->stars_filedata;
1551 read_rating_data(fd);
1555 if (vf_star_next(vf))
1557 return G_SOURCE_CONTINUE;
1561 vf->stars_filedata = nullptr;
1563 return G_SOURCE_REMOVE;
1567 return G_SOURCE_REMOVE;
1570 void vf_star_update(ViewFile *vf)
1574 if (!options->show_star_rating)
1582 void vf_marks_set(ViewFile *vf, gboolean enable)
1584 if (vf->marks_enabled == enable) return;
1586 vf->marks_enabled = enable;
1590 case FILEVIEW_LIST: vflist_marks_set(vf, enable); break;
1591 case FILEVIEW_ICON: vficon_marks_set(vf, enable); break;
1594 gtk_widget_show(vf->filter);
1596 gtk_widget_hide(vf->filter);
1598 vf_refresh_idle(vf);
1600 #pragma GCC diagnostic push
1601 #pragma GCC diagnostic ignored "-Wunused-function"
1602 void vf_star_rating_set_unused(ViewFile *vf, gboolean enable)
1604 if (options->show_star_rating == enable) return;
1605 options->show_star_rating = enable;
1609 case FILEVIEW_LIST: vflist_star_rating_set(vf, enable); break;
1610 case FILEVIEW_ICON: vficon_star_rating_set(vf, enable); break;
1612 vf_refresh_idle(vf);
1614 #pragma GCC diagnostic pop
1616 guint vf_marks_get_filter(ViewFile *vf)
1620 if (!vf->marks_enabled) return 0;
1622 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1624 if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[i])))
1632 GRegex *vf_file_filter_get_filter(ViewFile *vf)
1634 GRegex *ret = nullptr;
1635 GError *error = nullptr;
1636 gchar *file_filter_text = nullptr;
1638 if (!gtk_widget_get_visible(vf->file_filter.combo))
1640 return g_regex_new("", static_cast<GRegexCompileFlags>(0), static_cast<GRegexMatchFlags>(0), nullptr);
1643 file_filter_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
1645 if (file_filter_text[0] != '\0')
1647 ret = g_regex_new(file_filter_text, vf->file_filter.case_sensitive ? static_cast<GRegexCompileFlags>(0) : G_REGEX_CASELESS, static_cast<GRegexMatchFlags>(0), &error);
1650 log_printf("Error: could not compile regular expression %s\n%s\n", file_filter_text, error->message);
1651 g_error_free(error);
1653 ret = g_regex_new("", static_cast<GRegexCompileFlags>(0), static_cast<GRegexMatchFlags>(0), nullptr);
1655 g_free(file_filter_text);
1659 ret = g_regex_new("", static_cast<GRegexCompileFlags>(0), static_cast<GRegexMatchFlags>(0), nullptr);
1665 guint vf_class_get_filter(ViewFile *vf)
1670 if (!gtk_widget_get_visible(vf->file_filter.combo))
1675 for ( i = 0; i < FILE_FORMAT_CLASSES; i++)
1677 if (options->class_filter[i])
1686 void vf_set_layout(ViewFile *vf, LayoutWindow *layout)
1688 vf->layout = layout;
1693 *-----------------------------------------------------------------------------
1694 * maintenance (for rename, move, remove)
1695 *-----------------------------------------------------------------------------
1698 static gboolean vf_refresh_idle_cb(gpointer data)
1700 auto vf = static_cast<ViewFile *>(data);
1703 vf->refresh_idle_id = 0;
1704 return G_SOURCE_REMOVE;
1707 void vf_refresh_idle_cancel(ViewFile *vf)
1709 if (vf->refresh_idle_id)
1711 g_source_remove(vf->refresh_idle_id);
1712 vf->refresh_idle_id = 0;
1717 void vf_refresh_idle(ViewFile *vf)
1719 if (!vf->refresh_idle_id)
1721 vf->time_refresh_set = time(nullptr);
1722 /* file operations run with G_PRIORITY_DEFAULT_IDLE */
1723 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE + 50, vf_refresh_idle_cb, vf, nullptr);
1725 else if (time(nullptr) - vf->time_refresh_set > 1)
1727 /* more than 1 sec since last update - increase priority */
1728 vf_refresh_idle_cancel(vf);
1729 vf->time_refresh_set = time(nullptr);
1730 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE - 50, vf_refresh_idle_cb, vf, nullptr);
1734 void vf_notify_cb(FileData *fd, NotifyType type, gpointer data)
1736 auto vf = static_cast<ViewFile *>(data);
1739 auto interested = static_cast<NotifyType>(NOTIFY_CHANGE | NOTIFY_REREAD | NOTIFY_GROUPING);
1740 if (vf->marks_enabled) interested = static_cast<NotifyType>(interested | NOTIFY_MARKS | NOTIFY_METADATA);
1741 /** @FIXME NOTIFY_METADATA should be checked by the keyword-to-mark functions and converted to NOTIFY_MARKS only if there was a change */
1743 if (!(type & interested) || vf->refresh_idle_id || !vf->dir_fd) return;
1745 refresh = (fd == vf->dir_fd);
1749 gchar *base = remove_level_from_path(fd->path);
1750 refresh = (g_strcmp0(base, vf->dir_fd->path) == 0);
1754 if ((type & NOTIFY_CHANGE) && fd->change)
1756 if (!refresh && fd->change->dest)
1758 gchar *dest_base = remove_level_from_path(fd->change->dest);
1759 refresh = (g_strcmp0(dest_base, vf->dir_fd->path) == 0);
1763 if (!refresh && fd->change->source)
1765 gchar *source_base = remove_level_from_path(fd->change->source);
1766 refresh = (g_strcmp0(source_base, vf->dir_fd->path) == 0);
1767 g_free(source_base);
1773 DEBUG_1("Notify vf: %s %04x", fd->path, type);
1774 vf_refresh_idle(vf);
1778 static gboolean vf_read_metadata_in_idle_cb(gpointer data)
1781 auto vf = static_cast<ViewFile *>(data);
1784 vf_thumb_status(vf, vf_read_metadata_in_idle_progress(vf), _("Loading meta..."));
1790 fd = static_cast<FileData *>(work->data);
1792 if (fd && !fd->metadata_in_idle_loaded)
1796 read_exif_time_data(fd);
1798 if (!fd->exifdate_digitized)
1800 read_exif_time_digitized_data(fd);
1802 if (fd->rating == STAR_RATING_NOT_READ)
1804 read_rating_data(fd);
1806 fd->metadata_in_idle_loaded = TRUE;
1807 return G_SOURCE_CONTINUE;
1812 vf_thumb_status(vf, 0.0, nullptr);
1813 vf->read_metadata_in_idle_id = 0;
1815 return G_SOURCE_REMOVE;
1818 static void vf_read_metadata_in_idle_finished_cb(gpointer data)
1820 auto vf = static_cast<ViewFile *>(data);
1822 vf_thumb_status(vf, 0.0, "Loading meta...");
1823 vf->read_metadata_in_idle_id = 0;
1826 void vf_read_metadata_in_idle(ViewFile *vf)
1830 if (vf->read_metadata_in_idle_id)
1832 g_idle_remove_by_data(vf);
1834 vf->read_metadata_in_idle_id = 0;
1838 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);
1842 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */