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"
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 g_list_nth_data(vf->list, row);
76 gint vf_index_by_fd(ViewFile *vf, FileData *fd)
80 case FILEVIEW_LIST: return vflist_index_by_fd(vf, fd);
81 case FILEVIEW_ICON: return vficon_index_by_fd(vf, fd);
85 guint vf_count(ViewFile *vf, gint64 *bytes)
95 FileData *fd = work->data;
104 return g_list_length(vf->list);
107 GList *vf_get_list(ViewFile *vf)
111 for (work = vf->list; work; work = work->next)
113 FileData *fd = work->data;
114 list = g_list_prepend(list, file_data_ref(fd));
117 return g_list_reverse(list);
121 *-------------------------------------------------------------------
123 *-------------------------------------------------------------------
126 static gboolean vf_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
132 case FILEVIEW_LIST: return vflist_press_key_cb(widget, event, data);
133 case FILEVIEW_ICON: return vficon_press_key_cb(widget, event, data);
138 *-------------------------------------------------------------------
140 *-------------------------------------------------------------------
143 static gboolean vf_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
149 case FILEVIEW_LIST: return vflist_press_cb(widget, bevent, data);
150 case FILEVIEW_ICON: return vficon_press_cb(widget, bevent, data);
154 static gboolean vf_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
160 case FILEVIEW_LIST: return vflist_release_cb(widget, bevent, data);
161 case FILEVIEW_ICON: return vficon_release_cb(widget, bevent, data);
167 *-----------------------------------------------------------------------------
169 *-----------------------------------------------------------------------------
172 gboolean vf_index_is_selected(ViewFile *vf, gint row)
176 case FILEVIEW_LIST: return vflist_index_is_selected(vf, row);
177 case FILEVIEW_ICON: return vficon_index_is_selected(vf, row);
182 guint vf_selection_count(ViewFile *vf, gint64 *bytes)
186 case FILEVIEW_LIST: return vflist_selection_count(vf, bytes);
187 case FILEVIEW_ICON: return vficon_selection_count(vf, bytes);
191 GList *vf_selection_get_list(ViewFile *vf)
195 case FILEVIEW_LIST: return vflist_selection_get_list(vf);
196 case FILEVIEW_ICON: return vficon_selection_get_list(vf);
200 GList *vf_selection_get_list_by_index(ViewFile *vf)
204 case FILEVIEW_LIST: return vflist_selection_get_list_by_index(vf);
205 case FILEVIEW_ICON: return vficon_selection_get_list_by_index(vf);
209 void vf_select_all(ViewFile *vf)
213 case FILEVIEW_LIST: vflist_select_all(vf); break;
214 case FILEVIEW_ICON: vficon_select_all(vf); break;
218 void vf_select_none(ViewFile *vf)
222 case FILEVIEW_LIST: vflist_select_none(vf); break;
223 case FILEVIEW_ICON: vficon_select_none(vf); break;
227 void vf_select_invert(ViewFile *vf)
231 case FILEVIEW_LIST: vflist_select_invert(vf); break;
232 case FILEVIEW_ICON: vficon_select_invert(vf); break;
236 void vf_select_by_fd(ViewFile *vf, FileData *fd)
240 case FILEVIEW_LIST: vflist_select_by_fd(vf, fd); break;
241 case FILEVIEW_ICON: vficon_select_by_fd(vf, fd); break;
245 void vf_mark_to_selection(ViewFile *vf, gint mark, MarkToSelectionMode mode)
249 case FILEVIEW_LIST: vflist_mark_to_selection(vf, mark, mode); break;
250 case FILEVIEW_ICON: vficon_mark_to_selection(vf, mark, mode); break;
254 void vf_selection_to_mark(ViewFile *vf, gint mark, SelectionToMarkMode mode)
258 case FILEVIEW_LIST: vflist_selection_to_mark(vf, mark, mode); break;
259 case FILEVIEW_ICON: vficon_selection_to_mark(vf, mark, mode); break;
264 *-----------------------------------------------------------------------------
266 *-----------------------------------------------------------------------------
270 static void vf_dnd_init(ViewFile *vf)
274 case FILEVIEW_LIST: vflist_dnd_init(vf); break;
275 case FILEVIEW_ICON: vficon_dnd_init(vf); break;
280 *-----------------------------------------------------------------------------
282 *-----------------------------------------------------------------------------
285 GList *vf_pop_menu_file_list(ViewFile *vf)
289 case FILEVIEW_LIST: return vflist_pop_menu_file_list(vf);
290 case FILEVIEW_ICON: return vficon_pop_menu_file_list(vf);
294 GList *vf_selection_get_one(ViewFile *vf, FileData *fd)
298 case FILEVIEW_LIST: return vflist_selection_get_one(vf, fd);
299 case FILEVIEW_ICON: return vficon_selection_get_one(vf, fd);
303 static void vf_pop_menu_edit_cb(GtkWidget *widget, gpointer data)
306 const gchar *key = data;
308 vf = submenu_item_get_data(widget);
312 file_util_start_editor_from_filelist(key, vf_pop_menu_file_list(vf), vf->dir_fd->path, vf->listview);
315 static void vf_pop_menu_view_cb(GtkWidget *widget, gpointer data)
321 case FILEVIEW_LIST: vflist_pop_menu_view_cb(widget, data); break;
322 case FILEVIEW_ICON: vficon_pop_menu_view_cb(widget, data); break;
326 static void vf_pop_menu_copy_cb(GtkWidget *widget, gpointer data)
330 file_util_copy(NULL, vf_pop_menu_file_list(vf), NULL, vf->listview);
333 static void vf_pop_menu_move_cb(GtkWidget *widget, gpointer data)
337 file_util_move(NULL, vf_pop_menu_file_list(vf), NULL, vf->listview);
340 static void vf_pop_menu_rename_cb(GtkWidget *widget, gpointer data)
346 case FILEVIEW_LIST: vflist_pop_menu_rename_cb(widget, data); break;
347 case FILEVIEW_ICON: vficon_pop_menu_rename_cb(widget, data); break;
351 static void vf_pop_menu_delete_cb(GtkWidget *widget, gpointer data)
355 file_util_delete(NULL, vf_pop_menu_file_list(vf), vf->listview);
358 static void vf_pop_menu_copy_path_cb(GtkWidget *widget, gpointer data)
362 file_util_copy_path_list_to_clipboard(vf_pop_menu_file_list(vf), TRUE);
365 static void vf_pop_menu_copy_path_unquoted_cb(GtkWidget *widget, gpointer data)
369 file_util_copy_path_list_to_clipboard(vf_pop_menu_file_list(vf), FALSE);
372 static void vf_pop_menu_enable_grouping_cb(GtkWidget *widget, gpointer data)
376 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), FALSE);
379 static void vf_pop_menu_duplicates_cb(GtkWidget *widget, gpointer data)
384 dw = dupe_window_new();
385 dupe_window_add_files(dw, vf_pop_menu_file_list(vf), FALSE);
388 static void vf_pop_menu_disable_grouping_cb(GtkWidget *widget, gpointer data)
392 file_data_disable_grouping_list(vf_pop_menu_file_list(vf), TRUE);
395 static void vf_pop_menu_sort_cb(GtkWidget *widget, gpointer data)
400 if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget))) return;
402 vf = submenu_item_get_data(widget);
405 type = (SortType)GPOINTER_TO_INT(data);
409 layout_sort_set(vf->layout, type, vf->sort_ascend);
413 vf_sort_set(vf, type, vf->sort_ascend);
417 static void vf_pop_menu_sort_ascend_cb(GtkWidget *widget, gpointer data)
423 layout_sort_set(vf->layout, vf->sort_method, !vf->sort_ascend);
427 vf_sort_set(vf, vf->sort_method, !vf->sort_ascend);
431 static void vf_pop_menu_sel_mark_cb(GtkWidget *widget, gpointer data)
434 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_SET);
437 static void vf_pop_menu_sel_mark_and_cb(GtkWidget *widget, gpointer data)
440 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_AND);
443 static void vf_pop_menu_sel_mark_or_cb(GtkWidget *widget, gpointer data)
446 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_OR);
449 static void vf_pop_menu_sel_mark_minus_cb(GtkWidget *widget, gpointer data)
452 vf_mark_to_selection(vf, vf->active_mark, MTS_MODE_MINUS);
455 static void vf_pop_menu_set_mark_sel_cb(GtkWidget *widget, gpointer data)
458 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_SET);
461 static void vf_pop_menu_res_mark_sel_cb(GtkWidget *widget, gpointer data)
464 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_RESET);
467 static void vf_pop_menu_toggle_mark_sel_cb(GtkWidget *widget, gpointer data)
470 vf_selection_to_mark(vf, vf->active_mark, STM_MODE_TOGGLE);
473 static void vf_pop_menu_toggle_view_type_cb(GtkWidget *widget, gpointer data)
476 FileViewType new_type = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(widget), "menu_item_radio_data"));
477 if (!vf->layout) return;
479 layout_views_set(vf->layout, vf->layout->options.dir_view_type, new_type);
482 static void vf_pop_menu_refresh_cb(GtkWidget *widget, gpointer data)
488 case FILEVIEW_LIST: vflist_pop_menu_refresh_cb(widget, data); break;
489 case FILEVIEW_ICON: vficon_pop_menu_refresh_cb(widget, data); break;
493 static void vf_popup_destroy_cb(GtkWidget *widget, gpointer data)
499 case FILEVIEW_LIST: vflist_popup_destroy_cb(widget, data); break;
500 case FILEVIEW_ICON: vficon_popup_destroy_cb(widget, data); break;
503 filelist_free(vf->editmenu_fd_list);
504 vf->editmenu_fd_list = NULL;
508 * @brief Add file selection list to a collection
510 * @param[in] data Index to the collection list menu item selected, or -1 for new collection
514 static void vf_pop_menu_collections_cb(GtkWidget *widget, gpointer data)
517 GList *selection_list;
519 vf = submenu_item_get_data(widget);
520 selection_list = vf_selection_get_list(vf);
521 pop_menu_collections(selection_list, data);
523 filelist_free(selection_list);
526 GtkWidget *vf_pop_menu(ViewFile *vf)
531 gboolean active = FALSE;
536 vflist_color_set(vf, VFLIST(vf)->click_fd, TRUE);
537 active = (VFLIST(vf)->click_fd != NULL);
540 active = (VFICON(vf)->click_fd != NULL);
544 menu = popup_menu_short_lived();
546 g_signal_connect(G_OBJECT(menu), "destroy",
547 G_CALLBACK(vf_popup_destroy_cb), vf);
549 if (vf->clicked_mark > 0)
551 gint mark = vf->clicked_mark;
552 gchar *str_set_mark = g_strdup_printf(_("_Set mark %d"), mark);
553 gchar *str_res_mark = g_strdup_printf(_("_Reset mark %d"), mark);
554 gchar *str_toggle_mark = g_strdup_printf(_("_Toggle mark %d"), mark);
555 gchar *str_sel_mark = g_strdup_printf(_("_Select mark %d"), mark);
556 gchar *str_sel_mark_or = g_strdup_printf(_("_Add mark %d"), mark);
557 gchar *str_sel_mark_and = g_strdup_printf(_("_Intersection with mark %d"), mark);
558 gchar *str_sel_mark_minus = g_strdup_printf(_("_Unselect mark %d"), mark);
560 g_assert(mark >= 1 && mark <= FILEDATA_MARKS_SIZE);
562 vf->active_mark = mark;
563 vf->clicked_mark = 0;
565 menu_item_add_sensitive(menu, str_set_mark, active,
566 G_CALLBACK(vf_pop_menu_set_mark_sel_cb), vf);
568 menu_item_add_sensitive(menu, str_res_mark, active,
569 G_CALLBACK(vf_pop_menu_res_mark_sel_cb), vf);
571 menu_item_add_sensitive(menu, str_toggle_mark, active,
572 G_CALLBACK(vf_pop_menu_toggle_mark_sel_cb), vf);
574 menu_item_add_divider(menu);
576 menu_item_add_sensitive(menu, str_sel_mark, active,
577 G_CALLBACK(vf_pop_menu_sel_mark_cb), vf);
578 menu_item_add_sensitive(menu, str_sel_mark_or, active,
579 G_CALLBACK(vf_pop_menu_sel_mark_or_cb), vf);
580 menu_item_add_sensitive(menu, str_sel_mark_and, active,
581 G_CALLBACK(vf_pop_menu_sel_mark_and_cb), vf);
582 menu_item_add_sensitive(menu, str_sel_mark_minus, active,
583 G_CALLBACK(vf_pop_menu_sel_mark_minus_cb), vf);
585 menu_item_add_divider(menu);
587 g_free(str_set_mark);
588 g_free(str_res_mark);
589 g_free(str_toggle_mark);
590 g_free(str_sel_mark);
591 g_free(str_sel_mark_and);
592 g_free(str_sel_mark_or);
593 g_free(str_sel_mark_minus);
596 vf->editmenu_fd_list = vf_pop_menu_file_list(vf);
597 submenu_add_edit(menu, &item, G_CALLBACK(vf_pop_menu_edit_cb), vf, vf->editmenu_fd_list);
598 gtk_widget_set_sensitive(item, active);
600 menu_item_add_stock_sensitive(menu, _("View in _new window"), GTK_STOCK_NEW, active,
601 G_CALLBACK(vf_pop_menu_view_cb), vf);
603 menu_item_add_divider(menu);
604 menu_item_add_stock_sensitive(menu, _("_Copy..."), GTK_STOCK_COPY, active,
605 G_CALLBACK(vf_pop_menu_copy_cb), vf);
606 menu_item_add_sensitive(menu, _("_Move..."), active,
607 G_CALLBACK(vf_pop_menu_move_cb), vf);
608 menu_item_add_sensitive(menu, _("_Rename..."), active,
609 G_CALLBACK(vf_pop_menu_rename_cb), vf);
610 menu_item_add_sensitive(menu, _("_Copy path"), active,
611 G_CALLBACK(vf_pop_menu_copy_path_cb), vf);
612 menu_item_add_sensitive(menu, _("_Copy path unquoted"), active,
613 G_CALLBACK(vf_pop_menu_copy_path_unquoted_cb), vf);
614 menu_item_add_stock_sensitive(menu, _("_Delete..."), GTK_STOCK_DELETE, active,
615 G_CALLBACK(vf_pop_menu_delete_cb), vf);
616 menu_item_add_divider(menu);
618 menu_item_add_sensitive(menu, _("Enable file _grouping"), active,
619 G_CALLBACK(vf_pop_menu_enable_grouping_cb), vf);
620 menu_item_add_sensitive(menu, _("Disable file groupi_ng"), active,
621 G_CALLBACK(vf_pop_menu_disable_grouping_cb), vf);
623 menu_item_add_divider(menu);
624 menu_item_add_stock_sensitive(menu, _("_Find duplicates..."), GTK_STOCK_FIND, active,
625 G_CALLBACK(vf_pop_menu_duplicates_cb), vf);
626 menu_item_add_divider(menu);
628 submenu = submenu_add_collections(menu, &item,
629 G_CALLBACK(vf_pop_menu_collections_cb), vf);
630 gtk_widget_set_sensitive(item, active);
631 menu_item_add_divider(menu);
633 submenu = submenu_add_sort(NULL, G_CALLBACK(vf_pop_menu_sort_cb), vf,
634 FALSE, FALSE, TRUE, vf->sort_method);
635 menu_item_add_divider(submenu);
636 menu_item_add_check(submenu, _("Ascending"), vf->sort_ascend,
637 G_CALLBACK(vf_pop_menu_sort_ascend_cb), vf);
639 item = menu_item_add(menu, _("_Sort"), NULL, NULL);
640 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
642 item = menu_item_add_radio(menu, _("View as _List"), GINT_TO_POINTER(FILEVIEW_LIST), vf->type == FILEVIEW_LIST,
643 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
645 item = menu_item_add_radio(menu, _("View as _Icons"), GINT_TO_POINTER(FILEVIEW_ICON), vf->type == FILEVIEW_ICON,
646 G_CALLBACK(vf_pop_menu_toggle_view_type_cb), vf);
651 menu_item_add_check(menu, _("Show _thumbnails"), VFLIST(vf)->thumbs_enabled,
652 G_CALLBACK(vflist_pop_menu_thumbs_cb), vf);
655 menu_item_add_check(menu, _("Show filename _text"), VFICON(vf)->show_text,
656 G_CALLBACK(vficon_pop_menu_show_names_cb), vf);
660 menu_item_add_stock(menu, _("Re_fresh"), GTK_STOCK_REFRESH, G_CALLBACK(vf_pop_menu_refresh_cb), vf);
665 gboolean vf_refresh(ViewFile *vf)
669 case FILEVIEW_LIST: return vflist_refresh(vf);
670 case FILEVIEW_ICON: return vficon_refresh(vf);
674 gboolean vf_set_fd(ViewFile *vf, FileData *dir_fd)
678 case FILEVIEW_LIST: return vflist_set_fd(vf, dir_fd);
679 case FILEVIEW_ICON: return vficon_set_fd(vf, dir_fd);
683 static void vf_destroy_cb(GtkWidget *widget, gpointer data)
689 case FILEVIEW_LIST: vflist_destroy_cb(widget, data); break;
690 case FILEVIEW_ICON: vficon_destroy_cb(widget, data); break;
695 g_signal_handlers_disconnect_matched(G_OBJECT(vf->popup), G_SIGNAL_MATCH_DATA,
697 gtk_widget_destroy(vf->popup);
700 file_data_unref(vf->dir_fd);
705 static void vf_marks_filter_toggle_cb(GtkWidget *widget, gpointer data)
711 typedef struct _MarksTextEntry MarksTextEntry;
712 struct _MarksTextEntry {
715 GtkWidget *edit_widget;
720 static void vf_marks_tooltip_cancel_cb(GenericDialog *gd, gpointer data)
722 MarksTextEntry *mte = data;
724 g_free(mte->text_entry);
725 generic_dialog_close(gd);
728 static void vf_marks_tooltip_ok_cb(GenericDialog *gd, gpointer data)
730 MarksTextEntry *mte = data;
732 g_free(options->marks_tooltips[mte->mark_no]);
733 options->marks_tooltips[mte->mark_no] = g_strdup(gtk_entry_get_text(GTK_ENTRY(mte->edit_widget)));
735 gtk_widget_set_tooltip_text(mte->parent, options->marks_tooltips[mte->mark_no]);
737 g_free(mte->text_entry);
738 generic_dialog_close(gd);
741 void vf_marks_filter_on_icon_press(GtkEntry *entry, GtkEntryIconPosition pos,
742 GdkEvent *event, gpointer userdata)
744 MarksTextEntry *mte = userdata;
746 g_free(mte->text_entry);
747 mte->text_entry = g_strdup("");
748 gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), "");
751 static void vf_marks_tooltip_help_cb(GenericDialog *gd, gpointer data)
753 help_window_show("GuideImageMarks.html");
756 static gboolean vf_marks_tooltip_cb(GtkWidget *widget,
757 GdkEventButton *event,
761 gint i = GPOINTER_TO_INT(user_data);
764 if (event->button == MOUSE_BUTTON_RIGHT)
766 mte = g_new0(MarksTextEntry, 1);
768 mte->text_entry = g_strdup(options->marks_tooltips[i]);
769 mte->parent = widget;
771 mte->gd = generic_dialog_new(_("Mark text"), "mark_text",
773 vf_marks_tooltip_cancel_cb, mte);
774 generic_dialog_add_message(mte->gd, GTK_STOCK_DIALOG_QUESTION, _("Set mark text"),
775 _("This will set or clear the mark text."), FALSE);
776 generic_dialog_add_button(mte->gd, GTK_STOCK_OK, NULL,
777 vf_marks_tooltip_ok_cb, TRUE);
778 generic_dialog_add_button(mte->gd, GTK_STOCK_HELP, NULL,
779 vf_marks_tooltip_help_cb, FALSE);
781 table = pref_table_new(mte->gd->vbox, 3, 1, FALSE, TRUE);
782 pref_table_label(table, 0, 0, g_strdup_printf("%s%d", _("Mark "), mte->mark_no + 1), 1.0);
783 mte->edit_widget = gtk_entry_new();
784 gtk_widget_set_size_request(mte->edit_widget, 300, -1);
787 gtk_entry_set_text(GTK_ENTRY(mte->edit_widget), mte->text_entry);
789 gtk_table_attach_defaults(GTK_TABLE(table), mte->edit_widget, 1, 2, 0, 1);
790 generic_dialog_attach_default(mte->gd, mte->edit_widget);
792 gtk_entry_set_icon_from_stock(GTK_ENTRY(mte->edit_widget),
793 GTK_ENTRY_ICON_SECONDARY, GTK_STOCK_CLEAR);
794 gtk_entry_set_icon_tooltip_text (GTK_ENTRY(mte->edit_widget),
795 GTK_ENTRY_ICON_SECONDARY, "Clear");
796 g_signal_connect(GTK_ENTRY(mte->edit_widget), "icon-press",
797 G_CALLBACK(vf_marks_filter_on_icon_press), mte);
799 gtk_widget_show(mte->edit_widget);
800 gtk_widget_grab_focus(mte->edit_widget);
801 gtk_widget_show(GTK_WIDGET(mte->gd->dialog));
810 static GtkWidget *vf_marks_filter_init(ViewFile *vf)
812 GtkWidget *frame = gtk_frame_new(NULL);
813 GtkWidget *hbox = gtk_hbox_new(FALSE, 0);
817 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
819 GtkWidget *check = gtk_check_button_new();
820 gtk_box_pack_start(GTK_BOX(hbox), check, FALSE, FALSE, 0);
821 g_signal_connect(G_OBJECT(check), "toggled",
822 G_CALLBACK(vf_marks_filter_toggle_cb), vf);
823 g_signal_connect(G_OBJECT(check), "button_press_event",
824 G_CALLBACK(vf_marks_tooltip_cb), GINT_TO_POINTER(i));
825 gtk_widget_set_tooltip_text(check, options->marks_tooltips[i]);
827 gtk_widget_show(check);
828 vf->filter_check[i] = check;
830 gtk_container_add(GTK_CONTAINER(frame), hbox);
831 gtk_widget_show(hbox);
835 void vf_mark_filter_toggle(ViewFile *vf, gint mark)
838 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(vf->filter_check[n]),
839 !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[n])));
842 ViewFile *vf_new(FileViewType type, FileData *dir_fd)
846 vf = g_new0(ViewFile, 1);
849 vf->sort_method = SORT_NAME;
850 vf->sort_ascend = TRUE;
852 vf->scrolled = gtk_scrolled_window_new(NULL, NULL);
853 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vf->scrolled), GTK_SHADOW_IN);
854 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vf->scrolled),
855 GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
857 vf->filter = vf_marks_filter_init(vf);
859 vf->widget = gtk_vbox_new(FALSE, 0);
860 gtk_box_pack_start(GTK_BOX(vf->widget), vf->filter, FALSE, FALSE, 0);
861 gtk_box_pack_start(GTK_BOX(vf->widget), vf->scrolled, TRUE, TRUE, 0);
862 gtk_widget_show(vf->scrolled);
864 g_signal_connect(G_OBJECT(vf->widget), "destroy",
865 G_CALLBACK(vf_destroy_cb), vf);
869 case FILEVIEW_LIST: vf = vflist_new(vf, dir_fd); break;
870 case FILEVIEW_ICON: vf = vficon_new(vf, dir_fd); break;
875 g_signal_connect(G_OBJECT(vf->listview), "key_press_event",
876 G_CALLBACK(vf_press_key_cb), vf);
877 g_signal_connect(G_OBJECT(vf->listview), "button_press_event",
878 G_CALLBACK(vf_press_cb), vf);
879 g_signal_connect(G_OBJECT(vf->listview), "button_release_event",
880 G_CALLBACK(vf_release_cb), vf);
882 gtk_container_add(GTK_CONTAINER(vf->scrolled), vf->listview);
883 gtk_widget_show(vf->listview);
885 if (dir_fd) vf_set_fd(vf, dir_fd);
890 void vf_set_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gpointer data), gpointer data)
892 vf->func_status = func;
893 vf->data_status = data;
896 void vf_set_thumb_status_func(ViewFile *vf, void (*func)(ViewFile *vf, gdouble val, const gchar *text, gpointer data), gpointer data)
898 vf->func_thumb_status = func;
899 vf->data_thumb_status = data;
902 void vf_thumb_set(ViewFile *vf, gboolean enable)
906 case FILEVIEW_LIST: vflist_thumb_set(vf, enable); break;
907 case FILEVIEW_ICON: /*vficon_thumb_set(vf, enable);*/ break;
912 static gboolean vf_thumb_next(ViewFile *vf);
914 static gdouble vf_thumb_progress(ViewFile *vf)
921 case FILEVIEW_LIST: vflist_thumb_progress_count(vf->list, &count, &done); break;
922 case FILEVIEW_ICON: vficon_thumb_progress_count(vf->list, &count, &done); break;
925 DEBUG_1("thumb progress: %d of %d", done, count);
926 return (gdouble)done / count;
929 static void vf_set_thumb_fd(ViewFile *vf, FileData *fd)
933 case FILEVIEW_LIST: vflist_set_thumb_fd(vf, fd); break;
934 case FILEVIEW_ICON: vficon_set_thumb_fd(vf, fd); break;
938 static void vf_thumb_status(ViewFile *vf, gdouble val, const gchar *text)
940 if (vf->func_thumb_status)
942 vf->func_thumb_status(vf, val, text, vf->data_thumb_status);
946 static void vf_thumb_do(ViewFile *vf, FileData *fd)
950 vf_set_thumb_fd(vf, fd);
951 vf_thumb_status(vf, vf_thumb_progress(vf), _("Loading thumbs..."));
954 void vf_thumb_cleanup(ViewFile *vf)
956 vf_thumb_status(vf, 0.0, NULL);
958 vf->thumbs_running = FALSE;
960 thumb_loader_free(vf->thumbs_loader);
961 vf->thumbs_loader = NULL;
963 vf->thumbs_filedata = NULL;
966 void vf_thumb_stop(ViewFile *vf)
968 if (vf->thumbs_running) vf_thumb_cleanup(vf);
971 static void vf_thumb_common_cb(ThumbLoader *tl, gpointer data)
975 if (vf->thumbs_filedata && vf->thumbs_loader == tl)
977 vf_thumb_do(vf, vf->thumbs_filedata);
980 while (vf_thumb_next(vf));
983 static void vf_thumb_error_cb(ThumbLoader *tl, gpointer data)
985 vf_thumb_common_cb(tl, data);
988 static void vf_thumb_done_cb(ThumbLoader *tl, gpointer data)
990 vf_thumb_common_cb(tl, data);
993 static gboolean vf_thumb_next(ViewFile *vf)
997 if (!gtk_widget_get_realized(vf->listview))
999 vf_thumb_status(vf, 0.0, NULL);
1005 case FILEVIEW_LIST: fd = vflist_thumb_next_fd(vf); break;
1006 case FILEVIEW_ICON: fd = vficon_thumb_next_fd(vf); break;
1012 vf_thumb_cleanup(vf);
1016 vf->thumbs_filedata = fd;
1018 thumb_loader_free(vf->thumbs_loader);
1020 vf->thumbs_loader = thumb_loader_new(options->thumbnails.max_width, options->thumbnails.max_height);
1021 thumb_loader_set_callbacks(vf->thumbs_loader,
1027 if (!thumb_loader_start(vf->thumbs_loader, fd))
1029 /* set icon to unknown, continue */
1030 DEBUG_1("thumb loader start failed %s", fd->path);
1031 vf_thumb_do(vf, fd);
1039 static void vf_thumb_reset_all(ViewFile *vf)
1043 for (work = vf->list; work; work = work->next)
1045 FileData *fd = work->data;
1046 if (fd->thumb_pixbuf)
1048 g_object_unref(fd->thumb_pixbuf);
1049 fd->thumb_pixbuf = NULL;
1054 void vf_thumb_update(ViewFile *vf)
1058 if (vf->type == FILEVIEW_LIST && !VFLIST(vf)->thumbs_enabled) return;
1060 vf_thumb_status(vf, 0.0, _("Loading thumbs..."));
1061 vf->thumbs_running = TRUE;
1063 if (thumb_format_changed)
1065 vf_thumb_reset_all(vf);
1066 thumb_format_changed = FALSE;
1069 while (vf_thumb_next(vf));
1073 void vf_marks_set(ViewFile *vf, gboolean enable)
1075 if (vf->marks_enabled == enable) return;
1077 vf->marks_enabled = enable;
1081 case FILEVIEW_LIST: vflist_marks_set(vf, enable); break;
1082 case FILEVIEW_ICON: vficon_marks_set(vf, enable); break;
1085 gtk_widget_show(vf->filter);
1087 gtk_widget_hide(vf->filter);
1089 vf_refresh_idle(vf);
1092 guint vf_marks_get_filter(ViewFile *vf)
1096 if (!vf->marks_enabled) return 0;
1098 for (i = 0; i < FILEDATA_MARKS_SIZE ; i++)
1100 if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(vf->filter_check[i])))
1108 void vf_set_layout(ViewFile *vf, LayoutWindow *layout)
1110 vf->layout = layout;
1115 *-----------------------------------------------------------------------------
1116 * maintenance (for rename, move, remove)
1117 *-----------------------------------------------------------------------------
1120 static gboolean vf_refresh_idle_cb(gpointer data)
1122 ViewFile *vf = data;
1125 vf->refresh_idle_id = 0;
1129 void vf_refresh_idle_cancel(ViewFile *vf)
1131 if (vf->refresh_idle_id)
1133 g_source_remove(vf->refresh_idle_id);
1134 vf->refresh_idle_id = 0;
1139 void vf_refresh_idle(ViewFile *vf)
1141 if (!vf->refresh_idle_id)
1143 vf->time_refresh_set = time(NULL);
1144 /* file operations run with G_PRIORITY_DEFAULT_IDLE */
1145 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE + 50, vf_refresh_idle_cb, vf, NULL);
1147 else if (time(NULL) - vf->time_refresh_set > 1)
1149 /* more than 1 sec since last update - increase priority */
1150 vf_refresh_idle_cancel(vf);
1151 vf->time_refresh_set = time(NULL);
1152 vf->refresh_idle_id = g_idle_add_full(G_PRIORITY_DEFAULT_IDLE - 50, vf_refresh_idle_cb, vf, NULL);
1156 void vf_notify_cb(FileData *fd, NotifyType type, gpointer data)
1158 ViewFile *vf = data;
1161 NotifyType interested = NOTIFY_CHANGE | NOTIFY_REREAD | NOTIFY_GROUPING;
1162 if (vf->marks_enabled) interested |= NOTIFY_MARKS | NOTIFY_METADATA;
1163 /* FIXME: NOTIFY_METADATA should be checked by the keyword-to-mark functions and converted to NOTIFY_MARKS only if there was a change */
1165 if (!(type & interested) || vf->refresh_idle_id || !vf->dir_fd) return;
1167 refresh = (fd == vf->dir_fd);
1171 gchar *base = remove_level_from_path(fd->path);
1172 refresh = (g_strcmp0(base, vf->dir_fd->path) == 0);
1176 if ((type & NOTIFY_CHANGE) && fd->change)
1178 if (!refresh && fd->change->dest)
1180 gchar *dest_base = remove_level_from_path(fd->change->dest);
1181 refresh = (g_strcmp0(dest_base, vf->dir_fd->path) == 0);
1185 if (!refresh && fd->change->source)
1187 gchar *source_base = remove_level_from_path(fd->change->source);
1188 refresh = (g_strcmp0(source_base, vf->dir_fd->path) == 0);
1189 g_free(source_base);
1195 DEBUG_1("Notify vf: %s %04x", fd->path, type);
1196 vf_refresh_idle(vf);
1200 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */