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"
26 #include "collect-table.h"
28 #include "history_list.h"
32 #include "pixbuf_util.h"
35 #include "ui_fileops.h"
38 #include "view_file/view_file_list.h"
39 #include "view_file/view_file_icon.h"
43 *-----------------------------------------------------------------------------
45 *-----------------------------------------------------------------------------
48 void vf_send_update(ViewFile *vf)
50 if (vf->func_status) vf->func_status(vf, vf->data_status);
54 *-----------------------------------------------------------------------------
56 *-----------------------------------------------------------------------------
59 void vf_sort_set(ViewFile *vf, SortType type, gboolean ascend)
63 case FILEVIEW_LIST: vflist_sort_set(vf, type, ascend); break;
64 case FILEVIEW_ICON: vficon_sort_set(vf, type, ascend); break;
69 *-----------------------------------------------------------------------------
71 *-----------------------------------------------------------------------------
74 FileData *vf_index_get_data(ViewFile *vf, gint row)
76 return g_list_nth_data(vf->list, row);
79 gint vf_index_by_fd(ViewFile *vf, FileData *fd)
85 case FILEVIEW_LIST: ret = vflist_index_by_fd(vf, fd); break;
86 case FILEVIEW_ICON: ret = vficon_index_by_fd(vf, fd); break;
93 guint vf_count(ViewFile *vf, gint64 *bytes)
103 FileData *fd = work->data;
112 return g_list_length(vf->list);
115 GList *vf_get_list(ViewFile *vf)
119 for (work = vf->list; work; work = work->next)
121 FileData *fd = work->data;
122 list = g_list_prepend(list, file_data_ref(fd));
125 return g_list_reverse(list);
129 *-------------------------------------------------------------------
131 *-------------------------------------------------------------------
134 static gboolean vf_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
141 case FILEVIEW_LIST: ret = vflist_press_key_cb(widget, event, data); break;
142 case FILEVIEW_ICON: ret = vficon_press_key_cb(widget, event, data); break;
143 default: ret = FALSE;
150 *-------------------------------------------------------------------
152 *-------------------------------------------------------------------
155 static gboolean vf_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
162 case FILEVIEW_LIST: ret = vflist_press_cb(widget, bevent, data); break;
163 case FILEVIEW_ICON: ret = vficon_press_cb(widget, bevent, data); break;
164 default: ret = FALSE;
170 static gboolean vf_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
177 case FILEVIEW_LIST: ret = vflist_release_cb(widget, bevent, data); break;
178 case FILEVIEW_ICON: ret = vficon_release_cb(widget, bevent, data); break;
179 default: ret = FALSE;
187 *-----------------------------------------------------------------------------
189 *-----------------------------------------------------------------------------
192 guint vf_selection_count(ViewFile *vf, gint64 *bytes)
198 case FILEVIEW_LIST: ret = vflist_selection_count(vf, bytes); break;
199 case FILEVIEW_ICON: ret = vficon_selection_count(vf, bytes); break;
206 GList *vf_selection_get_list(ViewFile *vf)
212 case FILEVIEW_LIST: ret = vflist_selection_get_list(vf); break;
213 case FILEVIEW_ICON: ret = vficon_selection_get_list(vf); break;
220 GList *vf_selection_get_list_by_index(ViewFile *vf)
226 case FILEVIEW_LIST: ret = vflist_selection_get_list_by_index(vf); break;
227 case FILEVIEW_ICON: ret = vficon_selection_get_list_by_index(vf); break;
234 void vf_select_all(ViewFile *vf)
238 case FILEVIEW_LIST: vflist_select_all(vf); break;
239 case FILEVIEW_ICON: vficon_select_all(vf); break;
243 void vf_select_none(ViewFile *vf)
247 case FILEVIEW_LIST: vflist_select_none(vf); break;
248 case FILEVIEW_ICON: vficon_select_none(vf); break;
252 void vf_select_invert(ViewFile *vf)
256 case FILEVIEW_LIST: vflist_select_invert(vf); break;
257 case FILEVIEW_ICON: vficon_select_invert(vf); break;
261 void vf_select_by_fd(ViewFile *vf, FileData *fd)
265 case FILEVIEW_LIST: vflist_select_by_fd(vf, fd); break;
266 case FILEVIEW_ICON: vficon_select_by_fd(vf, fd); break;
270 void vf_select_list(ViewFile *vf, GList *list)
274 case FILEVIEW_LIST: vflist_select_list(vf, list); break;
275 case FILEVIEW_ICON: vficon_select_list(vf, list); break;
279 void vf_mark_to_selection(ViewFile *vf, gint mark, MarkToSelectionMode mode)
283 case FILEVIEW_LIST: vflist_mark_to_selection(vf, mark, mode); break;
284 case FILEVIEW_ICON: vficon_mark_to_selection(vf, mark, mode); break;
288 void vf_selection_to_mark(ViewFile *vf, gint mark, SelectionToMarkMode mode)
292 case FILEVIEW_LIST: vflist_selection_to_mark(vf, mark, mode); break;
293 case FILEVIEW_ICON: vficon_selection_to_mark(vf, mark, mode); break;
298 *-----------------------------------------------------------------------------
300 *-----------------------------------------------------------------------------
304 static void vf_dnd_init(ViewFile *vf)
308 case FILEVIEW_LIST: vflist_dnd_init(vf); break;
309 case FILEVIEW_ICON: vficon_dnd_init(vf); break;
314 *-----------------------------------------------------------------------------
316 *-----------------------------------------------------------------------------
319 GList *vf_pop_menu_file_list(ViewFile *vf)
325 case FILEVIEW_LIST: ret = vflist_pop_menu_file_list(vf); break;
326 case FILEVIEW_ICON: ret = vficon_pop_menu_file_list(vf); break;
333 GList *vf_selection_get_one(ViewFile *vf, FileData *fd)
339 case FILEVIEW_LIST: ret = vflist_selection_get_one(vf, fd); break;
340 case FILEVIEW_ICON: ret = vficon_selection_get_one(vf, fd); break;
347 static void vf_pop_menu_edit_cb(GtkWidget *widget, gpointer data)
350 const gchar *key = data;
352 vf = submenu_item_get_data(widget);
356 file_util_start_editor_from_filelist(key, vf_pop_menu_file_list(vf), vf->dir_fd->path, vf->listview);
359 static void vf_pop_menu_view_cb(GtkWidget *widget, gpointer data)
365 case FILEVIEW_LIST: vflist_pop_menu_view_cb(widget, data); break;
366 case FILEVIEW_ICON: vficon_pop_menu_view_cb(widget, data); break;
370 static void vf_pop_menu_open_archive_cb(GtkWidget *widget, gpointer data)
373 LayoutWindow *lw_new;
380 fd = (VFLIST(vf)->click_fd);
383 fd = (VFICON(vf)->click_fd);
387 dest_dir = open_archive(fd);
390 lw_new = layout_new_from_default();
391 layout_set_path(lw_new, dest_dir);
396 warning_dialog(_("Cannot open archive file"), _("See the Log Window"), GTK_STOCK_DIALOG_WARNING, NULL);
400 static void vf_pop_menu_copy_cb(GtkWidget *widget, gpointer data)
404 file_util_copy(NULL, vf_pop_menu_file_list(vf), NULL, vf->listview);
407 static void vf_pop_menu_move_cb(GtkWidget *widget, gpointer data)
411 file_util_move(NULL, vf_pop_menu_file_list(vf), NULL, vf->listview);
414 static void vf_pop_menu_rename_cb(GtkWidget *widget, gpointer data)
420 case FILEVIEW_LIST: vflist_pop_menu_rename_cb(widget, data); break;
421 case FILEVIEW_ICON: vficon_pop_menu_rename_cb(widget, data); break;
425 static void vf_pop_menu_delete_cb(GtkWidget *widget, gpointer data)
429 options->file_ops.safe_delete_enable = FALSE;
430 file_util_delete(NULL, vf_pop_menu_file_list(vf), vf->listview);
433 static void vf_pop_menu_move_to_trash_cb(GtkWidget *widget, gpointer data)
437 options->file_ops.safe_delete_enable = TRUE;
438 file_util_delete(NULL, vf_pop_menu_file_list(vf), vf->listview);
441 static void vf_pop_menu_copy_path_cb(GtkWidget *widget, gpointer data)
445 file_util_copy_path_list_to_clipboard(vf_pop_menu_file_list(vf), TRUE);
448 static void vf_pop_menu_copy_path_unquoted_cb(GtkWidget *widget, gpointer data)
452 file_util_copy_path_list_to_clipboard(vf_pop_menu_file_list(vf), FALSE);
455 static void vf_pop_menu_enable_grouping_cb(GtkWidget *widget, gpointer data)
459 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), FALSE);
462 static void vf_pop_menu_duplicates_cb(GtkWidget *widget, gpointer data)
467 dw = dupe_window_new();
468 dupe_window_add_files(dw, vf_pop_menu_file_list(vf), FALSE);
471 static void vf_pop_menu_disable_grouping_cb(GtkWidget *widget, gpointer data)
475 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), TRUE);
478 static void vf_pop_menu_sort_cb(GtkWidget *widget, gpointer data)
483 if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget))) return;
485 vf = submenu_item_get_data(widget);
488 type = (SortType)GPOINTER_TO_INT(data);
490 if (type == SORT_EXIFTIME || type == SORT_EXIFTIMEDIGITIZED || type == SORT_RATING)
492 vf_read_metadata_in_idle(vf);
497 layout_sort_set(vf->layout, type, vf->sort_ascend);
501 vf_sort_set(vf, type, vf->sort_ascend);
505 static void vf_pop_menu_sort_ascend_cb(GtkWidget *widget, gpointer data)
511 layout_sort_set(vf->layout, vf->sort_method, !vf->sort_ascend);
515 vf_sort_set(vf, vf->sort_method, !vf->sort_ascend);
519 static void vf_pop_menu_sel_mark_cb(GtkWidget *widget, gpointer data)
522 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_SET);
525 static void vf_pop_menu_sel_mark_and_cb(GtkWidget *widget, gpointer data)
528 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_AND);
531 static void vf_pop_menu_sel_mark_or_cb(GtkWidget *widget, gpointer data)
534 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_OR);
537 static void vf_pop_menu_sel_mark_minus_cb(GtkWidget *widget, gpointer data)
540 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_MINUS);
543 static void vf_pop_menu_set_mark_sel_cb(GtkWidget *widget, gpointer data)
546 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_SET);
549 static void vf_pop_menu_res_mark_sel_cb(GtkWidget *widget, gpointer data)
552 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_RESET);
555 static void vf_pop_menu_toggle_mark_sel_cb(GtkWidget *widget, gpointer data)
558 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_TOGGLE);
561 static void vf_pop_menu_toggle_view_type_cb(GtkWidget *widget, gpointer data)
564 FileViewType new_type = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), "menu_item_radio_data"));
565 if (!vf->layout) return;
567 layout_views_set(vf->layout, vf->layout->options.dir_view_type, new_type);
570 static void vf_pop_menu_refresh_cb(GtkWidget *widget, gpointer data)
576 case FILEVIEW_LIST: vflist_pop_menu_refresh_cb(widget, data); break;
577 case FILEVIEW_ICON: vficon_pop_menu_refresh_cb(widget, data); break;
581 static void vf_popup_destroy_cb(GtkWidget *widget, gpointer data)
587 case FILEVIEW_LIST: vflist_popup_destroy_cb(widget, data); break;
588 case FILEVIEW_ICON: vficon_popup_destroy_cb(widget, data); break;
591 filelist_free(vf->editmenu_fd_list);
592 vf->editmenu_fd_list = NULL;
596 * @brief Add file selection list to a collection
598 * @param[in] data Index to the collection list menu item selected, or -1 for new collection
602 static void vf_pop_menu_collections_cb(GtkWidget *widget, gpointer data)
605 GList *selection_list;
607 vf = submenu_item_get_data(widget);
608 selection_list = vf_selection_get_list(vf);
609 pop_menu_collections(selection_list, data);
611 filelist_free(selection_list);
614 GtkWidget *vf_pop_menu(ViewFile *vf)
619 gboolean active = FALSE;
620 gboolean class_archive = FALSE;
621 GtkAccelGroup *accel_group;
626 vflist_color_set(vf, VFLIST(vf)->click_fd, TRUE);
627 active = (VFLIST(vf)->click_fd != NULL);
628 class_archive = (VFLIST(vf)->click_fd != NULL && VFLIST(vf)->click_fd->format_class == FORMAT_CLASS_ARCHIVE);
631 active = (VFICON(vf)->click_fd != NULL);
632 class_archive = (VFICON(vf)->click_fd != NULL && VFICON(vf)->click_fd->format_class == FORMAT_CLASS_ARCHIVE);
636 menu = popup_menu_short_lived();
638 accel_group = gtk_accel_group_new();
639 gtk_menu_set_accel_group(GTK_MENU(menu), accel_group);
641 g_object_set_data(G_OBJECT(menu), "window_keys", NULL);
642 g_object_set_data(G_OBJECT(menu), "accel_group", accel_group);
644 g_signal_connect(G_OBJECT(menu), "destroy",
645 G_CALLBACK(vf_popup_destroy_cb), vf);
647 if (vf->clicked_mark > 0)
649 gint mark = vf->clicked_mark;
650 gchar *str_set_mark = g_strdup_printf(_("_Set mark %d"), mark);
651 gchar *str_res_mark = g_strdup_printf(_("_Reset mark %d"), mark);
652 gchar *str_toggle_mark = g_strdup_printf(_("_Toggle mark %d"), mark);
653 gchar *str_sel_mark = g_strdup_printf(_("_Select mark %d"), mark);
654 gchar *str_sel_mark_or = g_strdup_printf(_("_Add mark %d"), mark);
655 gchar *str_sel_mark_and = g_strdup_printf(_("_Intersection with mark %d"), mark);
656 gchar *str_sel_mark_minus = g_strdup_printf(_("_Unselect mark %d"), mark);
658 g_assert(mark >= 1 && mark <= FILEDATA_MARKS_SIZE);
660 vf->active_mark = mark;
661 vf->clicked_mark = 0;
663 menu_item_add_sensitive(menu, str_set_mark, active,
664 G_CALLBACK(vf_pop_menu_set_mark_sel_cb), vf);
666 menu_item_add_sensitive(menu, str_res_mark, active,
667 G_CALLBACK(vf_pop_menu_res_mark_sel_cb), vf);
669 menu_item_add_sensitive(menu, str_toggle_mark, active,
670 G_CALLBACK(vf_pop_menu_toggle_mark_sel_cb), vf);
672 menu_item_add_divider(menu);
674 menu_item_add_sensitive(menu, str_sel_mark, active,
675 G_CALLBACK(vf_pop_menu_sel_mark_cb), vf);
676 menu_item_add_sensitive(menu, str_sel_mark_or, active,
677 G_CALLBACK(vf_pop_menu_sel_mark_or_cb), vf);
678 menu_item_add_sensitive(menu, str_sel_mark_and, active,
679 G_CALLBACK(vf_pop_menu_sel_mark_and_cb), vf);
680 menu_item_add_sensitive(menu, str_sel_mark_minus, active,
681 G_CALLBACK(vf_pop_menu_sel_mark_minus_cb), vf);
683 menu_item_add_divider(menu);
685 g_free(str_set_mark);
686 g_free(str_res_mark);
687 g_free(str_toggle_mark);
688 g_free(str_sel_mark);
689 g_free(str_sel_mark_and);
690 g_free(str_sel_mark_or);
691 g_free(str_sel_mark_minus);
694 vf->editmenu_fd_list = vf_pop_menu_file_list(vf);
695 submenu_add_edit(menu, &item, G_CALLBACK(vf_pop_menu_edit_cb), vf, vf->editmenu_fd_list);
696 gtk_widget_set_sensitive(item, active);
698 menu_item_add_stock_sensitive(menu, _("View in _new window"), GTK_STOCK_NEW, active,
699 G_CALLBACK(vf_pop_menu_view_cb), vf);
701 menu_item_add_stock_sensitive(menu, _("Open archive"), GTK_STOCK_OPEN, active & class_archive, G_CALLBACK(vf_pop_menu_open_archive_cb), vf);
703 menu_item_add_divider(menu);
704 menu_item_add_stock_sensitive(menu, _("_Copy..."), GTK_STOCK_COPY, active,
705 G_CALLBACK(vf_pop_menu_copy_cb), vf);
706 menu_item_add_sensitive(menu, _("_Move..."), active,
707 G_CALLBACK(vf_pop_menu_move_cb), vf);
708 menu_item_add_sensitive(menu, _("_Rename..."), active,
709 G_CALLBACK(vf_pop_menu_rename_cb), vf);
710 menu_item_add_sensitive(menu, _("_Copy path to clipboard"), active,
711 G_CALLBACK(vf_pop_menu_copy_path_cb), vf);
712 menu_item_add_sensitive(menu, _("_Copy path unquoted to clipboard"), active,
713 G_CALLBACK(vf_pop_menu_copy_path_unquoted_cb), vf);
714 menu_item_add_divider(menu);
715 menu_item_add_stock_sensitive(menu,
716 options->file_ops.confirm_move_to_trash ? _("Move to Trash...") :
717 _("Move to Trash"), PIXBUF_INLINE_ICON_TRASH, active,
718 G_CALLBACK(vf_pop_menu_move_to_trash_cb), vf);
719 menu_item_add_stock_sensitive(menu,
720 options->file_ops.confirm_delete ? _("_Delete...") :
721 _("_Delete"), GTK_STOCK_DELETE, active,
722 G_CALLBACK(vf_pop_menu_delete_cb), vf);
723 menu_item_add_divider(menu);
725 menu_item_add_sensitive(menu, _("Enable file _grouping"), active,
726 G_CALLBACK(vf_pop_menu_enable_grouping_cb), vf);
727 menu_item_add_sensitive(menu, _("Disable file groupi_ng"), active,
728 G_CALLBACK(vf_pop_menu_disable_grouping_cb), vf);
730 menu_item_add_divider(menu);
731 menu_item_add_stock_sensitive(menu, _("_Find duplicates..."), GTK_STOCK_FIND, active,
732 G_CALLBACK(vf_pop_menu_duplicates_cb), vf);
733 menu_item_add_divider(menu);
735 submenu = submenu_add_collections(menu, &item,
736 G_CALLBACK(vf_pop_menu_collections_cb), vf);
737 gtk_widget_set_sensitive(item, active);
738 menu_item_add_divider(menu);
740 submenu = submenu_add_sort(NULL, G_CALLBACK(vf_pop_menu_sort_cb), vf,
741 FALSE, FALSE, TRUE, vf->sort_method);
742 menu_item_add_divider(submenu);
743 menu_item_add_check(submenu, _("Ascending"), vf->sort_ascend,
744 G_CALLBACK(vf_pop_menu_sort_ascend_cb), vf);
746 item = menu_item_add(menu, _("_Sort"), NULL, NULL);
747 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
749 item = menu_item_add_radio(menu, _("Images as List"), GINT_TO_POINTER(FILEVIEW_LIST), vf->type == FILEVIEW_LIST,
750 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
752 item = menu_item_add_radio(menu, _("Images as Icons"), GINT_TO_POINTER(FILEVIEW_ICON), vf->type == FILEVIEW_ICON,
753 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
758 menu_item_add_check(menu, _("Show _thumbnails"), VFLIST(vf)->thumbs_enabled,
759 G_CALLBACK(vflist_pop_menu_thumbs_cb), vf);
762 menu_item_add_check(menu, _("Show filename _text"), VFICON(vf)->show_text,
763 G_CALLBACK(vficon_pop_menu_show_names_cb), vf);
770 menu_item_add_check(menu, _("Show star rating"), options->show_star_rating,
771 G_CALLBACK(vflist_pop_menu_show_star_rating_cb), vf);
774 menu_item_add_check(menu, _("Show star rating"), options->show_star_rating,
775 G_CALLBACK(vficon_pop_menu_show_star_rating_cb), vf);
779 menu_item_add_stock(menu, _("Re_fresh"), GTK_STOCK_REFRESH, G_CALLBACK(vf_pop_menu_refresh_cb), vf);
784 gboolean vf_refresh(ViewFile *vf)
790 case FILEVIEW_LIST: ret = vflist_refresh(vf); break;
791 case FILEVIEW_ICON: ret = vficon_refresh(vf); break;
792 default: ret = FALSE;
798 gboolean vf_set_fd(ViewFile *vf, FileData *dir_fd)
804 case FILEVIEW_LIST: ret = vflist_set_fd(vf, dir_fd); break;
805 case FILEVIEW_ICON: ret = vficon_set_fd(vf, dir_fd); break;
806 default: ret = FALSE;
812 static void vf_destroy_cb(GtkWidget *widget, gpointer data)
818 case FILEVIEW_LIST: vflist_destroy_cb(widget, data); break;
819 case FILEVIEW_ICON: vficon_destroy_cb(widget, data); break;
824 g_signal_handlers_disconnect_matched(G_OBJECT(vf->popup), G_SIGNAL_MATCH_DATA,
826 gtk_widget_destroy(vf->popup);
829 if (vf->read_metadata_in_idle_id)
831 g_idle_remove_by_data(vf);
833 file_data_unref(vf->dir_fd);
838 static void vf_marks_filter_toggle_cb(GtkWidget *widget, gpointer data)
844 typedef struct _MarksTextEntry MarksTextEntry;
845 struct _MarksTextEntry {
848 GtkWidget *edit_widget;
853 static void vf_marks_tooltip_cancel_cb(GenericDialog *gd, gpointer data)
855 MarksTextEntry *mte = data;
857 g_free(mte->text_entry);
858 generic_dialog_close(gd);
861 static void vf_marks_tooltip_ok_cb(GenericDialog *gd, gpointer data)
863 MarksTextEntry *mte = data;
865 g_free(options->marks_tooltips[mte->mark_no]);
866 options->marks_tooltips[mte->mark_no] = g_strdup(gtk_entry_get_text(GTK_ENTRY(mte->edit_widget)));
868 gtk_widget_set_tooltip_text(mte->parent, options->marks_tooltips[mte->mark_no]);
870 g_free(mte->text_entry);
871 generic_dialog_close(gd);
874 void vf_marks_filter_on_icon_press(GtkEntry *entry, GtkEntryIconPosition pos,
875 GdkEvent *event, gpointer userdata)
877 MarksTextEntry *mte = userdata;
879 g_free(mte->text_entry);
880 mte->text_entry = g_strdup("");
881 gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), "");
884 static void vf_marks_tooltip_help_cb(GenericDialog *gd, gpointer data)
886 help_window_show("GuideImageMarks.html");
889 static gboolean vf_marks_tooltip_cb(GtkWidget *widget,
890 GdkEventButton *event,
894 gint i = GPOINTER_TO_INT(user_data);
897 if (event->button == MOUSE_BUTTON_RIGHT)
899 mte = g_new0(MarksTextEntry, 1);
901 mte->text_entry = g_strdup(options->marks_tooltips[i]);
902 mte->parent = widget;
904 mte->gd = generic_dialog_new(_("Mark text"), "mark_text",
906 vf_marks_tooltip_cancel_cb, mte);
907 generic_dialog_add_message(mte->gd, GTK_STOCK_DIALOG_QUESTION, _("Set mark text"),
908 _("This will set or clear the mark text."), FALSE);
909 generic_dialog_add_button(mte->gd, GTK_STOCK_OK, NULL,
910 vf_marks_tooltip_ok_cb, TRUE);
911 generic_dialog_add_button(mte->gd, GTK_STOCK_HELP, NULL,
912 vf_marks_tooltip_help_cb, FALSE);
914 table = pref_table_new(mte->gd->vbox, 3, 1, FALSE, TRUE);
915 pref_table_label(table, 0, 0, g_strdup_printf("%s%d", _("Mark "), mte->mark_no + 1), 1.0);
916 mte->edit_widget = gtk_entry_new();
917 gtk_widget_set_size_request(mte->edit_widget, 300, -1);
920 gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), mte->text_entry);
922 gtk_table_attach_defaults(GTK_TABLE(table), mte->edit_widget, 1, 2, 0, 1);
923 generic_dialog_attach_default(mte->gd, mte->edit_widget);
925 gtk_entry_set_icon_from_stock(GTK_ENTRY(mte->edit_widget),
926 GTK_ENTRY_ICON_SECONDARY, GTK_STOCK_CLEAR);
927 gtk_entry_set_icon_tooltip_text (GTK_ENTRY(mte->edit_widget),
928 GTK_ENTRY_ICON_SECONDARY, "Clear");
929 g_signal_connect(GTK_ENTRY(mte->edit_widget), "icon-press",
930 G_CALLBACK(vf_marks_filter_on_icon_press), mte);
932 gtk_widget_show(mte->edit_widget);
933 gtk_widget_grab_focus(mte->edit_widget);
934 gtk_widget_show(GTK_WIDGET(mte->gd->dialog));
942 static void vf_file_filter_save_cb(GtkWidget *widget, gpointer data)
946 gchar *remove_text = NULL;
947 gchar *index_text = NULL;
948 gboolean text_found = FALSE;
951 entry_text = g_strdup(gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo)))));
953 if (entry_text[0] == '\0' && vf->file_filter.last_selected >= 0)
955 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), vf->file_filter.last_selected);
956 remove_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
957 history_list_item_remove("file_filter", remove_text);
958 gtk_combo_box_text_remove(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), vf->file_filter.last_selected);
961 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), -1);
962 vf->file_filter.last_selected = - 1;
963 gtk_entry_set_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo))), "");
964 vf->file_filter.count--;
968 if (entry_text[0] != '\0')
970 for (i = 0; i < vf->file_filter.count; i++)
976 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), i);
977 index_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
979 if (g_strcmp0(index_text, entry_text) == 0)
989 history_list_add_to_key("file_filter", entry_text, 10);
990 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), entry_text);
991 vf->file_filter.count++;
992 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), vf->file_filter.count - 1);
1001 static void vf_file_filter_cb(GtkWidget *widget, gpointer data)
1003 ViewFile *vf = data;
1008 static gboolean vf_file_filter_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
1010 ViewFile *vf = data;
1011 vf->file_filter.last_selected = gtk_combo_box_get_active(GTK_COMBO_BOX(vf->file_filter.combo));
1013 gtk_widget_grab_focus(widget);
1018 static GtkWidget *vf_marks_filter_init(ViewFile *vf)
1020 GtkWidget *frame = gtk_frame_new(NULL);
1021 GtkWidget *hbox = gtk_hbox_new(FALSE, 0);
1025 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1027 GtkWidget *check = gtk_check_button_new();
1028 gtk_box_pack_start(GTK_BOX(hbox), check, FALSE, FALSE, 0);
1029 g_signal_connect(G_OBJECT(check), "toggled",
1030 G_CALLBACK(vf_marks_filter_toggle_cb), vf);
1031 g_signal_connect(G_OBJECT(check), "button_press_event",
1032 G_CALLBACK(vf_marks_tooltip_cb), GINT_TO_POINTER(i));
1033 gtk_widget_set_tooltip_text(check, options->marks_tooltips[i]);
1035 gtk_widget_show(check);
1036 vf->filter_check[i] = check;
1038 gtk_container_add(GTK_CONTAINER(frame), hbox);
1039 gtk_widget_show(hbox);
1043 void vf_file_filter_set(ViewFile *vf, gboolean enable)
1047 gtk_widget_show(vf->file_filter.combo);
1048 gtk_widget_show(vf->file_filter.frame);
1052 gtk_widget_hide(vf->file_filter.combo);
1053 gtk_widget_hide(vf->file_filter.frame);
1059 static gboolean vf_file_filter_class_cb(GtkWidget *widget, gpointer data)
1061 ViewFile *vf = data;
1064 gboolean state = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
1066 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1068 if (g_strcmp0(format_class_list[i], gtk_menu_item_get_label(GTK_MENU_ITEM(widget))) == 0)
1070 options->class_filter[i] = state;
1078 static gboolean vf_file_filter_class_set_all_cb(GtkWidget *widget, gpointer data)
1080 ViewFile *vf = data;
1087 if (g_strcmp0(_("Select all"), gtk_menu_item_get_label(GTK_MENU_ITEM(widget))) == 0)
1096 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1098 options->class_filter[i] = state;
1102 parent = gtk_widget_get_parent(widget);
1103 children = gtk_container_get_children(GTK_CONTAINER(parent));
1106 child = children->data;
1107 if (i < FILE_FORMAT_CLASSES)
1109 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(child), state);
1112 children = children->next;
1114 g_list_free(children);
1120 static GtkWidget *class_filter_menu (ViewFile *vf)
1123 GtkWidget *menu_item;
1126 menu = gtk_menu_new();
1128 for (i = 0; i < FILE_FORMAT_CLASSES; i++)
1130 menu_item = gtk_check_menu_item_new_with_label(format_class_list[i]);
1131 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menu_item), options->class_filter[i]);
1132 g_signal_connect(G_OBJECT(menu_item), "toggled", G_CALLBACK(vf_file_filter_class_cb), vf);
1133 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1134 gtk_widget_show(menu_item);
1137 menu_item = gtk_menu_item_new_with_label(_("Select all"));
1138 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1139 gtk_widget_show(menu_item);
1140 g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(vf_file_filter_class_set_all_cb), vf);
1142 menu_item = gtk_menu_item_new_with_label(_("Select none"));
1143 gtk_menu_shell_append(GTK_MENU_SHELL (menu), menu_item);
1144 gtk_widget_show(menu_item);
1145 g_signal_connect(G_OBJECT(menu_item), "activate", G_CALLBACK(vf_file_filter_class_set_all_cb), vf);
1150 static void case_sensitive_cb(GtkWidget *widget, gpointer data)
1152 ViewFile *vf = data;
1154 vf->file_filter.case_sensitive = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
1158 static GtkWidget *vf_file_filter_init(ViewFile *vf)
1160 GtkWidget *frame = gtk_frame_new(NULL);
1161 GtkWidget *hbox = gtk_hbox_new(FALSE, 0);
1164 GtkWidget *combo_entry;
1166 GtkWidget *menuitem;
1167 GtkWidget *case_sensitive;
1169 vf->file_filter.combo = gtk_combo_box_text_new_with_entry();
1170 combo_entry = gtk_bin_get_child(GTK_BIN(vf->file_filter.combo));
1171 gtk_widget_show(gtk_bin_get_child(GTK_BIN(vf->file_filter.combo)));
1172 gtk_widget_show((GTK_WIDGET(vf->file_filter.combo)));
1173 gtk_widget_set_tooltip_text(GTK_WIDGET(vf->file_filter.combo), "Use regular expressions");
1175 work = history_list_get_by_key("file_filter");
1178 gtk_combo_box_text_append_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo), (gchar *)work->data);
1181 vf->file_filter.count = n;
1183 gtk_combo_box_set_active(GTK_COMBO_BOX(vf->file_filter.combo), 0);
1185 g_signal_connect(G_OBJECT(combo_entry), "activate",
1186 G_CALLBACK(vf_file_filter_save_cb), vf);
1188 g_signal_connect(G_OBJECT(vf->file_filter.combo), "changed",
1189 G_CALLBACK(vf_file_filter_cb), vf);
1191 g_signal_connect(G_OBJECT(combo_entry), "button_press_event",
1192 G_CALLBACK(vf_file_filter_press_cb), vf);
1194 gtk_box_pack_start(GTK_BOX(hbox), vf->file_filter.combo, FALSE, FALSE, 0);
1195 gtk_widget_show(vf->file_filter.combo);
1196 gtk_container_add(GTK_CONTAINER(frame), hbox);
1197 gtk_widget_show(hbox);
1199 case_sensitive = gtk_check_button_new_with_label("Case");
1200 gtk_box_pack_start(GTK_BOX(hbox), case_sensitive, FALSE, FALSE, 0);
1201 gtk_widget_set_tooltip_text(GTK_WIDGET(case_sensitive), _("Case sensitive"));
1202 g_signal_connect(G_OBJECT(case_sensitive), "clicked", G_CALLBACK(case_sensitive_cb), vf);
1203 gtk_widget_show(case_sensitive);
1205 menubar = gtk_menu_bar_new();
1206 gtk_box_pack_start(GTK_BOX(hbox), menubar, FALSE, TRUE, 0);
1207 gtk_widget_show(menubar);
1209 menuitem = gtk_menu_item_new_with_label(_("Class"));
1210 gtk_widget_set_tooltip_text(GTK_WIDGET(menuitem), _("Select Class filter"));
1211 gtk_menu_item_set_submenu(GTK_MENU_ITEM (menuitem), class_filter_menu(vf));
1212 gtk_menu_shell_append(GTK_MENU_SHELL (menubar), menuitem);
1213 gtk_widget_show(menuitem);
1215 gtk_widget_show(menuitem);
1220 void vf_mark_filter_toggle(ViewFile *vf, gint mark)
1223 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(vf->filter_check[n]),
1224 !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[n])));
1227 ViewFile *vf_new(FileViewType type, FileData *dir_fd)
1231 vf = g_new0(ViewFile, 1);
1234 vf->sort_method = SORT_NAME;
1235 vf->sort_ascend = TRUE;
1236 vf->read_metadata_in_idle_id = 0;
1238 vf->scrolled = gtk_scrolled_window_new(NULL, NULL);
1239 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vf->scrolled), GTK_SHADOW_IN);
1240 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vf->scrolled),
1241 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
1243 vf->filter = vf_marks_filter_init(vf);
1244 vf->file_filter.frame = vf_file_filter_init(vf);
1246 vf->widget = gtk_vbox_new(FALSE, 0);
1247 gtk_box_pack_start(GTK_BOX(vf->widget), vf->filter, FALSE, FALSE, 0);
1248 gtk_box_pack_start(GTK_BOX(vf->widget), vf->file_filter.frame, FALSE, FALSE, 0);
1249 gtk_box_pack_start(GTK_BOX(vf->widget), vf->scrolled, TRUE, TRUE, 0);
1250 gtk_widget_show(vf->scrolled);
1252 g_signal_connect(G_OBJECT(vf->widget), "destroy",
1253 G_CALLBACK(vf_destroy_cb), vf);
1257 case FILEVIEW_LIST: vf = vflist_new(vf, dir_fd); break;
1258 case FILEVIEW_ICON: vf = vficon_new(vf, dir_fd); break;
1263 g_signal_connect(G_OBJECT(vf->listview), "key_press_event",
1264 G_CALLBACK(vf_press_key_cb), vf);
1265 g_signal_connect(G_OBJECT(vf->listview), "button_press_event",
1266 G_CALLBACK(vf_press_cb), vf);
1267 g_signal_connect(G_OBJECT(vf->listview), "button_release_event",
1268 G_CALLBACK(vf_release_cb), vf);
1270 gtk_container_add(GTK_CONTAINER(vf->scrolled), vf->listview);
1271 gtk_widget_show(vf->listview);
1273 if (dir_fd) vf_set_fd(vf, dir_fd);
1278 void vf_set_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gpointer data), gpointer data)
1280 vf->func_status = func;
1281 vf->data_status = data;
1284 void vf_set_thumb_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gdouble val, const gchar *text, gpointer data), gpointer data)
1286 vf->func_thumb_status = func;
1287 vf->data_thumb_status = data;
1290 void vf_thumb_set(ViewFile *vf, gboolean enable)
1294 case FILEVIEW_LIST: vflist_thumb_set(vf, enable); break;
1295 case FILEVIEW_ICON: /*vficon_thumb_set(vf, enable);*/ break;
1300 static gboolean vf_thumb_next(ViewFile *vf);
1302 static gdouble vf_thumb_progress(ViewFile *vf)
1309 case FILEVIEW_LIST: vflist_thumb_progress_count(vf->list, &count, &done); break;
1310 case FILEVIEW_ICON: vficon_thumb_progress_count(vf->list, &count, &done); break;
1313 DEBUG_1("thumb progress: %d of %d", done, count);
1314 return (gdouble)done / count;
1317 static gdouble vf_read_metadata_in_idle_progress(ViewFile *vf)
1324 case FILEVIEW_LIST: vflist_read_metadata_progress_count(vf->list, &count, &done); break;
1325 case FILEVIEW_ICON: vficon_read_metadata_progress_count(vf->list, &count, &done); break;
1328 return (gdouble)done / count;
1331 static void vf_set_thumb_fd(ViewFile *vf, FileData *fd)
1335 case FILEVIEW_LIST: vflist_set_thumb_fd(vf, fd); break;
1336 case FILEVIEW_ICON: vficon_set_thumb_fd(vf, fd); break;
1340 static void vf_thumb_status(ViewFile *vf, gdouble val, const gchar *text)
1342 if (vf->func_thumb_status)
1344 vf->func_thumb_status(vf, val, text, vf->data_thumb_status);
1348 static void vf_thumb_do(ViewFile *vf, FileData *fd)
1352 vf_set_thumb_fd(vf, fd);
1353 vf_thumb_status(vf, vf_thumb_progress(vf), _("Loading thumbs..."));
1356 void vf_thumb_cleanup(ViewFile *vf)
1358 vf_thumb_status(vf, 0.0, NULL);
1360 vf->thumbs_running = FALSE;
1362 thumb_loader_free(vf->thumbs_loader);
1363 vf->thumbs_loader = NULL;
1365 vf->thumbs_filedata = NULL;
1368 void vf_thumb_stop(ViewFile *vf)
1370 if (vf->thumbs_running) vf_thumb_cleanup(vf);
1373 static void vf_thumb_common_cb(ThumbLoader *tl, gpointer data)
1375 ViewFile *vf = data;
1377 if (vf->thumbs_filedata && vf->thumbs_loader == tl)
1379 vf_thumb_do(vf, vf->thumbs_filedata);
1382 while (vf_thumb_next(vf));
1385 static void vf_thumb_error_cb(ThumbLoader *tl, gpointer data)
1387 vf_thumb_common_cb(tl, data);
1390 static void vf_thumb_done_cb(ThumbLoader *tl, gpointer data)
1392 vf_thumb_common_cb(tl, data);
1395 static gboolean vf_thumb_next(ViewFile *vf)
1397 FileData *fd = NULL;
1399 if (!gtk_widget_get_realized(vf->listview))
1401 vf_thumb_status(vf, 0.0, NULL);
1407 case FILEVIEW_LIST: fd = vflist_thumb_next_fd(vf); break;
1408 case FILEVIEW_ICON: fd = vficon_thumb_next_fd(vf); break;
1414 vf_thumb_cleanup(vf);
1418 vf->thumbs_filedata = fd;
1420 thumb_loader_free(vf->thumbs_loader);
1422 vf->thumbs_loader = thumb_loader_new(options->thumbnails.max_width, options->thumbnails.max_height);
1423 thumb_loader_set_callbacks(vf->thumbs_loader,
1429 if (!thumb_loader_start(vf->thumbs_loader, fd))
1431 /* set icon to unknown, continue */
1432 DEBUG_1("thumb loader start failed %s", fd->path);
1433 vf_thumb_do(vf, fd);
1441 static void vf_thumb_reset_all(ViewFile *vf)
1445 for (work = vf->list; work; work = work->next)
1447 FileData *fd = work->data;
1448 if (fd->thumb_pixbuf)
1450 g_object_unref(fd->thumb_pixbuf);
1451 fd->thumb_pixbuf = NULL;
1456 void vf_thumb_update(ViewFile *vf)
1460 if (vf->type == FILEVIEW_LIST && !VFLIST(vf)->thumbs_enabled) return;
1462 vf_thumb_status(vf, 0.0, _("Loading thumbs..."));
1463 vf->thumbs_running = TRUE;
1465 if (thumb_format_changed)
1467 vf_thumb_reset_all(vf);
1468 thumb_format_changed = FALSE;
1471 while (vf_thumb_next(vf));
1474 void vf_star_cleanup(ViewFile *vf)
1476 if (vf->stars_id != 0)
1478 g_source_remove(vf->stars_id);
1482 vf->stars_filedata = NULL;
1485 void vf_star_stop(ViewFile *vf)
1487 vf_star_cleanup(vf);
1490 static void vf_set_star_fd(ViewFile *vf, FileData *fd)
1494 case FILEVIEW_LIST: vflist_set_star_fd(vf, fd); break;
1495 case FILEVIEW_ICON: vficon_set_star_fd(vf, fd); break;
1500 static void vf_star_do(ViewFile *vf, FileData *fd)
1504 vf_set_star_fd(vf, fd);
1507 static gboolean vf_star_next(ViewFile *vf)
1509 FileData *fd = NULL;
1513 case FILEVIEW_LIST: fd = vflist_star_next_fd(vf); break;
1514 case FILEVIEW_ICON: fd = vficon_star_next_fd(vf); break;
1521 vf_star_cleanup(vf);
1528 gboolean vf_stars_cb(gpointer data)
1530 ViewFile *vf = data;
1531 FileData *fd = vf->stars_filedata;
1535 read_rating_data(fd);
1539 if (vf_star_next(vf))
1545 vf->stars_filedata = NULL;
1554 void vf_star_update(ViewFile *vf)
1558 if (!options->show_star_rating)
1566 void vf_marks_set(ViewFile *vf, gboolean enable)
1568 if (vf->marks_enabled == enable) return;
1570 vf->marks_enabled = enable;
1574 case FILEVIEW_LIST: vflist_marks_set(vf, enable); break;
1575 case FILEVIEW_ICON: vficon_marks_set(vf, enable); break;
1578 gtk_widget_show(vf->filter);
1580 gtk_widget_hide(vf->filter);
1582 vf_refresh_idle(vf);
1585 void vf_star_rating_set(ViewFile *vf, gboolean enable)
1587 if (options->show_star_rating == enable) return;
1588 options->show_star_rating = enable;
1592 case FILEVIEW_LIST: vflist_star_rating_set(vf, enable); break;
1593 case FILEVIEW_ICON: vficon_star_rating_set(vf, enable); break;
1595 vf_refresh_idle(vf);
1598 guint vf_marks_get_filter(ViewFile *vf)
1602 if (!vf->marks_enabled) return 0;
1604 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1606 if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[i])))
1614 GRegex *vf_file_filter_get_filter(ViewFile *vf)
1617 GError *error = NULL;
1618 gchar *file_filter_text = NULL;
1620 if (!gtk_widget_get_visible(vf->file_filter.combo))
1622 return g_regex_new("", 0, 0, NULL);
1625 file_filter_text = gtk_combo_box_text_get_active_text(GTK_COMBO_BOX_TEXT(vf->file_filter.combo));
1627 if (file_filter_text[0] != '\0')
1629 ret = g_regex_new(file_filter_text, vf->file_filter.case_sensitive ? 0 : G_REGEX_CASELESS, 0, &error);
1632 log_printf("Error: could not compile regular expression %s\n%s\n", file_filter_text, error->message);
1633 g_error_free(error);
1635 ret = g_regex_new("", 0, 0, NULL);
1637 g_free(file_filter_text);
1641 ret = g_regex_new("", 0, 0, NULL);
1647 guint vf_class_get_filter(ViewFile *vf)
1652 if (!gtk_widget_get_visible(vf->file_filter.combo))
1657 for ( i = 0; i < FILE_FORMAT_CLASSES; i++)
1659 if (options->class_filter[i])
1668 void vf_set_layout(ViewFile *vf, LayoutWindow *layout)
1670 vf->layout = layout;
1675 *-----------------------------------------------------------------------------
1676 * maintenance (for rename, move, remove)
1677 *-----------------------------------------------------------------------------
1680 static gboolean vf_refresh_idle_cb(gpointer data)
1682 ViewFile *vf = data;
1685 vf->refresh_idle_id = 0;
1689 void vf_refresh_idle_cancel(ViewFile *vf)
1691 if (vf->refresh_idle_id)
1693 g_source_remove(vf->refresh_idle_id);
1694 vf->refresh_idle_id = 0;
1699 void vf_refresh_idle(ViewFile *vf)
1701 if (!vf->refresh_idle_id)
1703 vf->time_refresh_set = time(NULL);
1704 /* file operations run with G_PRIORITY_DEFAULT_IDLE */
1705 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE + 50, vf_refresh_idle_cb, vf, NULL);
1707 else if (time(NULL) - vf->time_refresh_set > 1)
1709 /* more than 1 sec since last update - increase priority */
1710 vf_refresh_idle_cancel(vf);
1711 vf->time_refresh_set = time(NULL);
1712 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE - 50, vf_refresh_idle_cb, vf, NULL);
1716 void vf_notify_cb(FileData *fd, NotifyType type, gpointer data)
1718 ViewFile *vf = data;
1721 NotifyType interested = NOTIFY_CHANGE | NOTIFY_REREAD | NOTIFY_GROUPING;
1722 if (vf->marks_enabled) interested |= NOTIFY_MARKS | NOTIFY_METADATA;
1723 /** @FIXME NOTIFY_METADATA should be checked by the keyword-to-mark functions and converted to NOTIFY_MARKS only if there was a change */
1725 if (!(type & interested) || vf->refresh_idle_id || !vf->dir_fd) return;
1727 refresh = (fd == vf->dir_fd);
1731 gchar *base = remove_level_from_path(fd->path);
1732 refresh = (g_strcmp0(base, vf->dir_fd->path) == 0);
1736 if ((type & NOTIFY_CHANGE) && fd->change)
1738 if (!refresh && fd->change->dest)
1740 gchar *dest_base = remove_level_from_path(fd->change->dest);
1741 refresh = (g_strcmp0(dest_base, vf->dir_fd->path) == 0);
1745 if (!refresh && fd->change->source)
1747 gchar *source_base = remove_level_from_path(fd->change->source);
1748 refresh = (g_strcmp0(source_base, vf->dir_fd->path) == 0);
1749 g_free(source_base);
1755 DEBUG_1("Notify vf: %s %04x", fd->path, type);
1756 vf_refresh_idle(vf);
1760 static gboolean vf_read_metadata_in_idle_cb(gpointer data)
1763 ViewFile *vf = data;
1766 vf_thumb_status(vf, vf_read_metadata_in_idle_progress(vf), _("Loading meta..."));
1774 if (fd && !fd->metadata_in_idle_loaded)
1778 read_exif_time_data(fd);
1780 if (!fd->exifdate_digitized)
1782 read_exif_time_digitized_data(fd);
1784 if (fd->rating == STAR_RATING_NOT_READ)
1786 read_rating_data(fd);
1788 fd->metadata_in_idle_loaded = TRUE;
1794 vf_thumb_status(vf, 0.0, NULL);
1795 vf->read_metadata_in_idle_id = 0;
1800 static void vf_read_metadata_in_idle_finished_cb(gpointer data)
1802 ViewFile *vf = data;
1804 vf_thumb_status(vf, 0.0, "Loading meta...");
1805 vf->read_metadata_in_idle_id = 0;
1808 void vf_read_metadata_in_idle(ViewFile *vf)
1812 if (vf->read_metadata_in_idle_id)
1814 g_idle_remove_by_data(vf);
1816 vf->read_metadata_in_idle_id = 0;
1820 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);
1826 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */