4 * Copyright (C) 2008 The Geeqie Team
8 * This software is released under the GNU General Public License (GNU GPL).
9 * Please read the included file COPYING for more information.
10 * This software comes with no warranty of any kind, use at your own risk!
14 #include "view_file_list.h"
16 #include "cache_maint.h"
22 #include "layout_image.h"
26 #include "ui_bookmark.h"
27 #include "ui_fileops.h"
29 #include "ui_tree_edit.h"
32 #include <gdk/gdkkeysyms.h> /* for keyboard values */
36 FILE_COLUMN_POINTER = 0,
44 FILE_COLUMN_MARKS_LAST = FILE_COLUMN_MARKS + FILEDATA_MARKS_SIZE - 1,
49 static gint vflist_row_is_selected(ViewFileList *vfl, FileData *fd);
50 static gint vflist_row_rename_cb(TreeEditData *td, const gchar *old, const gchar *new, gpointer data);
51 static void vflist_populate_view(ViewFileList *vfl);
54 *-----------------------------------------------------------------------------
56 *-----------------------------------------------------------------------------
59 static void vflist_send_update(ViewFileList *vfl)
61 if (vfl->func_status) vfl->func_status(vfl, vfl->data_status);
65 *-----------------------------------------------------------------------------
67 *-----------------------------------------------------------------------------
74 } ViewFileListFindRowData;
76 static gboolean vflist_find_row_cb(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data)
78 ViewFileListFindRowData *find = data;
80 gtk_tree_model_get(model, iter, FILE_COLUMN_POINTER, &fd, -1);
91 static gint vflist_find_row(ViewFileList *vfl, FileData *fd, GtkTreeIter *iter)
94 ViewFileListFindRowData data = {fd, iter, 0, 0};
96 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
97 gtk_tree_model_foreach(store, vflist_find_row_cb, &data);
109 static gint vflist_find_sidecar_list_idx(GList *work, FileData *fd)
114 FileData *fd_p = work->data;
115 if (fd == fd_p) return i;
119 GList *work2 = fd_p->sidecar_files;
123 if (fd == fd_p) return i;
134 static gint vflist_sidecar_list_count(GList *work)
139 FileData *fd = work->data;
142 GList *work2 = fd->sidecar_files;
154 static void vflist_color_set(ViewFileList *vfl, FileData *fd, gint color_set)
159 if (vflist_find_row(vfl, fd, &iter) < 0) return;
160 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
161 gtk_tree_store_set(GTK_TREE_STORE(store), &iter, FILE_COLUMN_COLOR, color_set, -1);
164 static void vflist_move_cursor(ViewFileList *vfl, GtkTreeIter *iter)
169 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
171 tpath = gtk_tree_model_get_path(store, iter);
172 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vfl->listview), tpath, NULL, FALSE);
173 gtk_tree_path_free(tpath);
177 static gint vflist_column_idx(ViewFileList *vfl, gint store_idx)
179 GList *columns, *work;
182 columns = gtk_tree_view_get_columns(GTK_TREE_VIEW(vfl->listview));
186 GtkTreeViewColumn *column = work->data;
187 if (store_idx == GPOINTER_TO_INT(g_object_get_data (G_OBJECT(column), "column_store_idx")))
193 g_list_free(columns);
199 *-----------------------------------------------------------------------------
201 *-----------------------------------------------------------------------------
204 static void vflist_dnd_get(GtkWidget *widget, GdkDragContext *context,
205 GtkSelectionData *selection_data, guint info,
206 guint time, gpointer data)
208 ViewFileList *vfl = data;
210 gchar *uri_text = NULL;
213 if (!vfl->click_fd) return;
215 if (vflist_row_is_selected(vfl, vfl->click_fd))
217 list = vflist_selection_get_list(vfl);
221 list = g_list_append(NULL, file_data_ref(vfl->click_fd));
226 uri_text = uri_text_from_filelist(list, &total, (info == TARGET_TEXT_PLAIN));
231 gtk_selection_data_set(selection_data, selection_data->target,
232 8, (guchar *)uri_text, total);
236 static void vflist_dnd_begin(GtkWidget *widget, GdkDragContext *context, gpointer data)
238 ViewFileList *vfl = data;
240 vflist_color_set(vfl, vfl->click_fd, TRUE);
242 if (vfl->thumbs_enabled &&
243 vfl->click_fd && vfl->click_fd->pixbuf)
247 if (vflist_row_is_selected(vfl, vfl->click_fd))
248 items = vflist_selection_count(vfl, NULL);
252 dnd_set_drag_icon(widget, context, vfl->click_fd->pixbuf, items);
256 static void vflist_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
258 ViewFileList *vfl = data;
260 vflist_color_set(vfl, vfl->click_fd, FALSE);
262 if (context->action == GDK_ACTION_MOVE)
268 static void vflist_dnd_init(ViewFileList *vfl)
270 gtk_drag_source_set(vfl->listview, GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
271 dnd_file_drag_types, dnd_file_drag_types_count,
272 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
273 g_signal_connect(G_OBJECT(vfl->listview), "drag_data_get",
274 G_CALLBACK(vflist_dnd_get), vfl);
275 g_signal_connect(G_OBJECT(vfl->listview), "drag_begin",
276 G_CALLBACK(vflist_dnd_begin), vfl);
277 g_signal_connect(G_OBJECT(vfl->listview), "drag_end",
278 G_CALLBACK(vflist_dnd_end), vfl);
282 *-----------------------------------------------------------------------------
284 *-----------------------------------------------------------------------------
287 static GList *vflist_pop_menu_file_list(ViewFileList *vfl)
289 if (!vfl->click_fd) return NULL;
291 if (vflist_row_is_selected(vfl, vfl->click_fd))
293 return vflist_selection_get_list(vfl);
296 return g_list_append(NULL, file_data_ref(vfl->click_fd));
299 static void vflist_pop_menu_edit_cb(GtkWidget *widget, gpointer data)
305 vfl = submenu_item_get_data(widget);
306 n = GPOINTER_TO_INT(data);
310 list = vflist_pop_menu_file_list(vfl);
311 start_editor_from_filelist(n, list);
315 static void vflist_pop_menu_info_cb(GtkWidget *widget, gpointer data)
317 ViewFileList *vfl = data;
319 info_window_new(NULL, vflist_pop_menu_file_list(vfl), NULL);
322 static void vflist_pop_menu_view_cb(GtkWidget *widget, gpointer data)
324 ViewFileList *vfl = data;
326 if (vflist_row_is_selected(vfl, vfl->click_fd))
330 list = vflist_selection_get_list(vfl);
331 view_window_new_from_list(list);
336 view_window_new(vfl->click_fd);
340 static void vflist_pop_menu_copy_cb(GtkWidget *widget, gpointer data)
342 ViewFileList *vfl = data;
344 file_util_copy(NULL, vflist_pop_menu_file_list(vfl), NULL, vfl->listview);
347 static void vflist_pop_menu_copy_path_cb(GtkWidget *widget, gpointer data)
349 ViewFileList *vfl = data;
351 file_util_copy_path_list_to_clipboard(vflist_pop_menu_file_list(vfl));
354 static void vflist_pop_menu_move_cb(GtkWidget *widget, gpointer data)
356 ViewFileList *vfl = data;
358 file_util_move(NULL, vflist_pop_menu_file_list(vfl), NULL, vfl->listview);
361 static void vflist_pop_menu_rename_cb(GtkWidget *widget, gpointer data)
363 ViewFileList *vfl = data;
366 list = vflist_pop_menu_file_list(vfl);
367 if (options->file_ops.enable_in_place_rename &&
368 list && !list->next && vfl->click_fd)
375 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
376 if (vflist_find_row(vfl, vfl->click_fd, &iter) >= 0)
380 tpath = gtk_tree_model_get_path(store, &iter);
381 tree_edit_by_path(GTK_TREE_VIEW(vfl->listview), tpath,
382 vflist_column_idx(vfl, FILE_COLUMN_NAME), vfl->click_fd->name,
383 vflist_row_rename_cb, vfl);
384 gtk_tree_path_free(tpath);
389 file_util_rename(NULL, list, vfl->listview);
392 static void vflist_pop_menu_delete_cb(GtkWidget *widget, gpointer data)
394 ViewFileList *vfl = data;
396 file_util_delete(NULL, vflist_pop_menu_file_list(vfl), vfl->listview);
399 static void vflist_pop_menu_sort_cb(GtkWidget *widget, gpointer data)
404 if (!gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget))) return;
406 vfl = submenu_item_get_data(widget);
409 type = (SortType)GPOINTER_TO_INT(data);
413 layout_sort_set(vfl->layout, type, vfl->sort_ascend);
417 vflist_sort_set(vfl, type, vfl->sort_ascend);
421 static void vflist_pop_menu_sort_ascend_cb(GtkWidget *widget, gpointer data)
423 ViewFileList *vfl = data;
427 layout_sort_set(vfl->layout, vfl->sort_method, !vfl->sort_ascend);
431 vflist_sort_set(vfl, vfl->sort_method, !vfl->sort_ascend);
435 static void vflist_pop_menu_icons_cb(GtkWidget *widget, gpointer data)
437 ViewFileList *vfl = data;
439 if (vfl->layout) layout_views_set(vfl->layout, vfl->layout->dir_view_type, TRUE);
442 static void vflist_pop_menu_thumbs_cb(GtkWidget *widget, gpointer data)
444 ViewFileList *vfl = data;
446 vflist_color_set(vfl, vfl->click_fd, FALSE);
449 layout_thumb_set(vfl->layout, !vfl->thumbs_enabled);
453 vflist_thumb_set(vfl, !vfl->thumbs_enabled);
457 static void vflist_pop_menu_refresh_cb(GtkWidget *widget, gpointer data)
459 ViewFileList *vfl = data;
461 vflist_color_set(vfl, vfl->click_fd, FALSE);
465 static void vflist_pop_menu_sel_mark_cb(GtkWidget *widget, gpointer data)
467 ViewFileList *vfl = data;
468 vflist_mark_to_selection(vfl, vfl->active_mark, MTS_MODE_SET);
471 static void vflist_pop_menu_sel_mark_and_cb(GtkWidget *widget, gpointer data)
473 ViewFileList *vfl = data;
474 vflist_mark_to_selection(vfl, vfl->active_mark, MTS_MODE_AND);
477 static void vflist_pop_menu_sel_mark_or_cb(GtkWidget *widget, gpointer data)
479 ViewFileList *vfl = data;
480 vflist_mark_to_selection(vfl, vfl->active_mark, MTS_MODE_OR);
483 static void vflist_pop_menu_sel_mark_minus_cb(GtkWidget *widget, gpointer data)
485 ViewFileList *vfl = data;
486 vflist_mark_to_selection(vfl, vfl->active_mark, MTS_MODE_MINUS);
489 static void vflist_pop_menu_set_mark_sel_cb(GtkWidget *widget, gpointer data)
491 ViewFileList *vfl = data;
492 vflist_selection_to_mark(vfl, vfl->active_mark, STM_MODE_SET);
495 static void vflist_pop_menu_res_mark_sel_cb(GtkWidget *widget, gpointer data)
497 ViewFileList *vfl = data;
498 vflist_selection_to_mark(vfl, vfl->active_mark, STM_MODE_RESET);
501 static void vflist_pop_menu_toggle_mark_sel_cb(GtkWidget *widget, gpointer data)
503 ViewFileList *vfl = data;
504 vflist_selection_to_mark(vfl, vfl->active_mark, STM_MODE_TOGGLE);
508 static void vflist_popup_destroy_cb(GtkWidget *widget, gpointer data)
510 ViewFileList *vfl = data;
511 vflist_color_set(vfl, vfl->click_fd, FALSE);
512 vfl->click_fd = NULL;
517 static GtkWidget *vflist_pop_menu(ViewFileList *vfl, FileData *fd, gint col_idx)
524 vflist_color_set(vfl, fd, TRUE);
525 active = (fd != NULL);
527 menu = popup_menu_short_lived();
528 g_signal_connect(G_OBJECT(menu), "destroy",
529 G_CALLBACK(vflist_popup_destroy_cb), vfl);
531 if (col_idx >= FILE_COLUMN_MARKS && col_idx <= FILE_COLUMN_MARKS_LAST)
533 gint mark = col_idx - FILE_COLUMN_MARKS;
534 gchar *str_set_mark = g_strdup_printf(_("_Set mark %d"), mark + 1);
535 gchar *str_res_mark = g_strdup_printf(_("_Reset mark %d"), mark + 1);
536 gchar *str_toggle_mark = g_strdup_printf(_("_Toggle mark %d"), mark + 1);
537 gchar *str_sel_mark = g_strdup_printf(_("_Select mark %d"), mark + 1);
538 gchar *str_sel_mark_or = g_strdup_printf(_("_Add mark %d"), mark + 1);
539 gchar *str_sel_mark_and = g_strdup_printf(_("_Intersection with mark %d"), mark + 1);
540 gchar *str_sel_mark_minus = g_strdup_printf(_("_Unselect mark %d"), mark + 1);
543 vfl->active_mark = mark;
544 menu_item_add_sensitive(menu, str_set_mark, active,
545 G_CALLBACK(vflist_pop_menu_set_mark_sel_cb), vfl);
547 menu_item_add_sensitive(menu, str_res_mark, active,
548 G_CALLBACK(vflist_pop_menu_res_mark_sel_cb), vfl);
550 menu_item_add_sensitive(menu, str_toggle_mark, active,
551 G_CALLBACK(vflist_pop_menu_toggle_mark_sel_cb), vfl);
553 menu_item_add_divider(menu);
555 menu_item_add_sensitive(menu, str_sel_mark, active,
556 G_CALLBACK(vflist_pop_menu_sel_mark_cb), vfl);
557 menu_item_add_sensitive(menu, str_sel_mark_or, active,
558 G_CALLBACK(vflist_pop_menu_sel_mark_or_cb), vfl);
559 menu_item_add_sensitive(menu, str_sel_mark_and, active,
560 G_CALLBACK(vflist_pop_menu_sel_mark_and_cb), vfl);
561 menu_item_add_sensitive(menu, str_sel_mark_minus, active,
562 G_CALLBACK(vflist_pop_menu_sel_mark_minus_cb), vfl);
564 menu_item_add_divider(menu);
566 g_free(str_set_mark);
567 g_free(str_res_mark);
568 g_free(str_toggle_mark);
569 g_free(str_sel_mark);
570 g_free(str_sel_mark_and);
571 g_free(str_sel_mark_or);
572 g_free(str_sel_mark_minus);
575 submenu_add_edit(menu, &item, G_CALLBACK(vflist_pop_menu_edit_cb), vfl);
576 gtk_widget_set_sensitive(item, active);
578 menu_item_add_stock_sensitive(menu, _("_Properties"), GTK_STOCK_PROPERTIES, active,
579 G_CALLBACK(vflist_pop_menu_info_cb), vfl);
580 menu_item_add_stock_sensitive(menu, _("View in _new window"), GTK_STOCK_NEW, active,
581 G_CALLBACK(vflist_pop_menu_view_cb), vfl);
583 menu_item_add_divider(menu);
584 menu_item_add_stock_sensitive(menu, _("_Copy..."), GTK_STOCK_COPY, active,
585 G_CALLBACK(vflist_pop_menu_copy_cb), vfl);
586 menu_item_add_sensitive(menu, _("_Move..."), active,
587 G_CALLBACK(vflist_pop_menu_move_cb), vfl);
588 menu_item_add_sensitive(menu, _("_Rename..."), active,
589 G_CALLBACK(vflist_pop_menu_rename_cb), vfl);
590 menu_item_add_stock_sensitive(menu, _("_Delete..."), GTK_STOCK_DELETE, active,
591 G_CALLBACK(vflist_pop_menu_delete_cb), vfl);
592 if (options->show_copy_path)
593 menu_item_add_sensitive(menu, _("_Copy path"), active,
594 G_CALLBACK(vflist_pop_menu_copy_path_cb), vfl);
596 menu_item_add_divider(menu);
598 submenu = submenu_add_sort(NULL, G_CALLBACK(vflist_pop_menu_sort_cb), vfl,
599 FALSE, FALSE, TRUE, vfl->sort_method);
600 menu_item_add_divider(submenu);
601 menu_item_add_check(submenu, _("Ascending"), vfl->sort_ascend,
602 G_CALLBACK(vflist_pop_menu_sort_ascend_cb), vfl);
604 item = menu_item_add(menu, _("_Sort"), NULL, NULL);
605 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu);
607 menu_item_add_check(menu, _("View as _icons"), FALSE,
608 G_CALLBACK(vflist_pop_menu_icons_cb), vfl);
609 menu_item_add_check(menu, _("Show _thumbnails"), vfl->thumbs_enabled,
610 G_CALLBACK(vflist_pop_menu_thumbs_cb), vfl);
611 menu_item_add_stock(menu, _("Re_fresh"), GTK_STOCK_REFRESH, G_CALLBACK(vflist_pop_menu_refresh_cb), vfl);
617 *-----------------------------------------------------------------------------
619 *-----------------------------------------------------------------------------
622 static gint vflist_row_rename_cb(TreeEditData *td, const gchar *old, const gchar *new, gpointer data)
624 ViewFileList *vfl = data;
628 if (strlen(new) == 0) return FALSE;
630 old_path = concat_dir_and_file(vfl->path, old);
631 new_path = concat_dir_and_file(vfl->path, new);
633 if (strchr(new, '/') != NULL)
635 gchar *text = g_strdup_printf(_("Invalid file name:\n%s"), new);
636 file_util_warning_dialog(_("Error renaming file"), text, GTK_STOCK_DIALOG_ERROR, vfl->listview);
639 else if (isfile(new_path))
641 gchar *text = g_strdup_printf(_("A file with name %s already exists."), new);
642 file_util_warning_dialog(_("Error renaming file"), text, GTK_STOCK_DIALOG_ERROR, vfl->listview);
647 gint row = vflist_index_by_path(vfl, old_path);
650 GList *work = g_list_nth(vfl->list, row);
651 FileData *fd = work->data;
653 if (!file_data_add_change_info(fd, FILEDATA_CHANGE_RENAME, old_path, new_path) || !rename_file_ext(fd))
655 gchar *text = g_strdup_printf(_("Unable to rename file:\n%s\nto:\n%s"), old, new);
656 file_util_warning_dialog(_("Error renaming file"), text, GTK_STOCK_DIALOG_ERROR, vfl->listview);
668 static void vflist_menu_position_cb(GtkMenu *menu, gint *x, gint *y, gboolean *push_in, gpointer data)
670 ViewFileList *vfl = data;
676 if (vflist_find_row(vfl, vfl->click_fd, &iter) < 0) return;
677 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
678 tpath = gtk_tree_model_get_path(store, &iter);
679 tree_view_get_cell_clamped(GTK_TREE_VIEW(vfl->listview), tpath, FILE_COLUMN_NAME - 1, TRUE, x, y, &cw, &ch);
680 gtk_tree_path_free(tpath);
682 popup_menu_position_clamp(menu, x, y, 0);
685 static gint vflist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
687 ViewFileList *vfl = data;
690 if (event->keyval != GDK_Menu) return FALSE;
692 gtk_tree_view_get_cursor(GTK_TREE_VIEW(vfl->listview), &tpath, NULL);
698 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
699 gtk_tree_model_get_iter(store, &iter, tpath);
700 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &vfl->click_fd, -1);
701 gtk_tree_path_free(tpath);
705 vfl->click_fd = NULL;
708 vfl->popup = vflist_pop_menu(vfl, vfl->click_fd, 0);
709 gtk_menu_popup(GTK_MENU(vfl->popup), NULL, NULL, vflist_menu_position_cb, vfl, 0, GDK_CURRENT_TIME);
714 static gint vflist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
716 ViewFileList *vfl = data;
720 GtkTreeViewColumn *column;
723 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
724 &tpath, &column, NULL, NULL))
727 col_idx = GPOINTER_TO_INT(g_object_get_data (G_OBJECT(column), "column_store_idx"));
729 if (bevent->button == MOUSE_BUTTON_LEFT &&
730 col_idx >= FILE_COLUMN_MARKS && col_idx <= FILE_COLUMN_MARKS_LAST)
733 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
735 gtk_tree_model_get_iter(store, &iter, tpath);
736 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
738 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE);
740 gtk_tree_path_free(tpath);
745 if (bevent->button == MOUSE_BUTTON_RIGHT)
747 vfl->popup = vflist_pop_menu(vfl, vfl->click_fd, col_idx);
748 gtk_menu_popup(GTK_MENU(vfl->popup), NULL, NULL, NULL, NULL,
749 bevent->button, bevent->time);
753 if (!fd) return FALSE;
755 if (bevent->button == MOUSE_BUTTON_MIDDLE)
757 if (!vflist_row_is_selected(vfl, fd))
759 vflist_color_set(vfl, fd, TRUE);
765 if (bevent->button == MOUSE_BUTTON_LEFT && bevent->type == GDK_BUTTON_PRESS &&
766 !(bevent->state & GDK_SHIFT_MASK ) &&
767 !(bevent->state & GDK_CONTROL_MASK ) &&
768 vflist_row_is_selected(vfl, fd))
770 gtk_widget_grab_focus(widget);
771 // return TRUE; // FIXME - expand
775 if (bevent->button == MOUSE_BUTTON_LEFT && bevent->type == GDK_2BUTTON_PRESS)
777 if (vfl->layout) layout_image_full_screen_start(vfl->layout);
784 static gint vflist_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
786 ViewFileList *vfl = data;
791 if (bevent->button == MOUSE_BUTTON_MIDDLE)
793 vflist_color_set(vfl, vfl->click_fd, FALSE);
796 if (bevent->button != MOUSE_BUTTON_LEFT && bevent->button != MOUSE_BUTTON_MIDDLE)
801 if ((bevent->x != 0 || bevent->y != 0) &&
802 gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
803 &tpath, NULL, NULL, NULL))
807 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
808 gtk_tree_model_get_iter(store, &iter, tpath);
809 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
810 gtk_tree_path_free(tpath);
813 if (bevent->button == MOUSE_BUTTON_MIDDLE)
815 if (fd && vfl->click_fd == fd)
817 GtkTreeSelection *selection;
819 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget));
820 if (vflist_row_is_selected(vfl, fd))
822 gtk_tree_selection_unselect_iter(selection, &iter);
826 gtk_tree_selection_select_iter(selection, &iter);
832 if (fd && vfl->click_fd == fd &&
833 !(bevent->state & GDK_SHIFT_MASK ) &&
834 !(bevent->state & GDK_CONTROL_MASK ) &&
835 vflist_row_is_selected(vfl, fd))
837 GtkTreeSelection *selection;
839 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget));
840 gtk_tree_selection_unselect_all(selection);
841 gtk_tree_selection_select_iter(selection, &iter);
842 vflist_move_cursor(vfl, &iter);
843 // return TRUE;// FIXME - expand
849 static void vflist_select_image(ViewFileList *vfl, FileData *sel_fd)
851 FileData *read_ahead_fd = NULL;
856 cur_fd = layout_image_get_fd(vfl->layout);
857 if (sel_fd == cur_fd) return; /* no change */
859 row = g_list_index(vfl->list, sel_fd);
860 // FIXME sidecar data
862 if (sel_fd && options->image.enable_read_ahead && row >= 0)
864 if (row > g_list_index(vfl->list, cur_fd) &&
865 row + 1 < vflist_count(vfl, NULL))
867 read_ahead_fd = vflist_index_get_data(vfl, row + 1);
871 read_ahead_fd = vflist_index_get_data(vfl, row - 1);
875 layout_image_set_with_ahead(vfl->layout, sel_fd, read_ahead_fd);
878 static gint vflist_select_idle_cb(gpointer data)
880 ViewFileList *vfl = data;
884 vfl->select_idle_id = -1;
888 vflist_send_update(vfl);
892 vflist_select_image(vfl, vfl->select_fd);
893 vfl->select_fd = NULL;
896 vfl->select_idle_id = -1;
900 static void vflist_select_idle_cancel(ViewFileList *vfl)
902 if (vfl->select_idle_id != -1) g_source_remove(vfl->select_idle_id);
903 vfl->select_idle_id = -1;
906 static gboolean vflist_select_cb(GtkTreeSelection *selection, GtkTreeModel *store, GtkTreePath *tpath,
907 gboolean path_currently_selected, gpointer data)
909 ViewFileList *vfl = data;
912 if (!path_currently_selected &&
913 gtk_tree_model_get_iter(store, &iter, tpath))
915 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &vfl->select_fd, -1);
919 vfl->select_fd = NULL;
923 vfl->select_idle_id == -1)
925 vfl->select_idle_id = g_idle_add(vflist_select_idle_cb, vfl);
932 *-----------------------------------------------------------------------------
934 *-----------------------------------------------------------------------------
938 static gboolean vflist_dummy_select_cb(GtkTreeSelection *selection, GtkTreeModel *store, GtkTreePath *tpath,
939 gboolean path_currently_selected, gpointer data)
946 static void vflist_setup_iter(ViewFileList *vfl, GtkTreeStore *store, GtkTreeIter *iter, FileData *fd)
950 gchar *sidecars = NULL;
952 if (fd->sidecar_files)
953 sidecars = sidecar_file_data_list_to_string(fd);
954 size = text_from_size(fd->size);
956 gtk_tree_store_set(store, iter, FILE_COLUMN_POINTER, fd,
957 FILE_COLUMN_THUMB, (vfl->thumbs_enabled) ? fd->pixbuf : NULL,
958 FILE_COLUMN_NAME, fd->name,
959 FILE_COLUMN_SIDECARS, sidecars,
960 FILE_COLUMN_SIZE, size,
961 FILE_COLUMN_DATE, text_from_time(fd->date),
962 FILE_COLUMN_COLOR, FALSE, -1);
963 for (i = 0; i < FILEDATA_MARKS_SIZE; i++)
964 gtk_tree_store_set(store, iter, FILE_COLUMN_MARKS + i, fd->marks[i], -1);
971 static void vflist_setup_iter_with_sidecars(ViewFileList *vfl, GtkTreeStore *store, GtkTreeIter *iter, FileData *fd)
977 vflist_setup_iter(vfl, store, iter, fd);
980 /* this is almost the same code as in vflist_populate_view
981 maybe it should be made more generic and used in both places */
984 valid = gtk_tree_model_iter_children(GTK_TREE_MODEL(store), &s_iter, iter);
986 work = fd->sidecar_files;
990 FileData *sfd = work->data;
995 FileData *old_sfd = NULL;
999 gtk_tree_model_get(GTK_TREE_MODEL(store), &s_iter, FILE_COLUMN_POINTER, &old_sfd, -1);
1007 match = filelist_sort_compare_filedata_full(sfd, old_sfd, SORT_NAME, TRUE);
1022 gtk_tree_store_insert_before(store, &new, iter, &s_iter);
1026 gtk_tree_store_append(store, &new, iter);
1029 vflist_setup_iter(vfl, store, &new, sfd);
1035 valid = gtk_tree_store_remove(store, &s_iter);
1039 vflist_setup_iter(vfl, store, &s_iter, sfd);
1041 if (valid) valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &s_iter);
1051 valid = gtk_tree_store_remove(store, &s_iter);
1055 void vflist_sort_set(ViewFileList *vfl, SortType type, gint ascend)
1058 GHashTable *fd_idx_hash = g_hash_table_new(NULL, NULL);
1060 GtkTreeStore *store;
1063 if (vfl->sort_method == type && vfl->sort_ascend == ascend) return;
1064 if (!vfl->list) return;
1070 FileData *fd = work->data;
1071 g_hash_table_insert(fd_idx_hash, fd, GINT_TO_POINTER(i));
1076 vfl->sort_method = type;
1077 vfl->sort_ascend = ascend;
1079 vfl->list = filelist_sort(vfl->list, vfl->sort_method, vfl->sort_ascend);
1081 new_order = g_malloc(i * sizeof(gint));
1087 FileData *fd = work->data;
1088 new_order[i] = GPOINTER_TO_INT(g_hash_table_lookup(fd_idx_hash, fd));
1093 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
1094 gtk_tree_store_reorder(store, NULL, new_order);
1097 g_hash_table_destroy(fd_idx_hash);
1101 *-----------------------------------------------------------------------------
1103 *-----------------------------------------------------------------------------
1106 static gint vflist_thumb_next(ViewFileList *vfl);
1108 static void vflist_thumb_status(ViewFileList *vfl, gdouble val, const gchar *text)
1110 if (vfl->func_thumb_status)
1112 vfl->func_thumb_status(vfl, val, text, vfl->data_thumb_status);
1116 static void vflist_thumb_cleanup(ViewFileList *vfl)
1118 vflist_thumb_status(vfl, 0.0, NULL);
1120 vfl->thumbs_count = 0;
1121 vfl->thumbs_running = FALSE;
1123 thumb_loader_free(vfl->thumbs_loader);
1124 vfl->thumbs_loader = NULL;
1126 vfl->thumbs_filedata = NULL;
1129 static void vflist_thumb_stop(ViewFileList *vfl)
1131 if (vfl->thumbs_running) vflist_thumb_cleanup(vfl);
1134 static void vflist_thumb_do(ViewFileList *vfl, ThumbLoader *tl, FileData *fd)
1136 GtkTreeStore *store;
1139 if (!fd || vflist_find_row(vfl, fd, &iter) < 0) return;
1141 if (fd->pixbuf) g_object_unref(fd->pixbuf);
1142 fd->pixbuf = thumb_loader_get_pixbuf(tl, TRUE);
1144 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
1145 gtk_tree_store_set(store, &iter, FILE_COLUMN_THUMB, fd->pixbuf, -1);
1147 vflist_thumb_status(vfl, (gdouble)(vfl->thumbs_count) / vflist_sidecar_list_count(vfl->list), _("Loading thumbs..."));
1150 static void vflist_thumb_error_cb(ThumbLoader *tl, gpointer data)
1152 ViewFileList *vfl = data;
1154 if (vfl->thumbs_filedata && vfl->thumbs_loader == tl)
1156 vflist_thumb_do(vfl, tl, vfl->thumbs_filedata);
1159 while (vflist_thumb_next(vfl));
1162 static void vflist_thumb_done_cb(ThumbLoader *tl, gpointer data)
1164 ViewFileList *vfl = data;
1166 if (vfl->thumbs_filedata && vfl->thumbs_loader == tl)
1168 vflist_thumb_do(vfl, tl, vfl->thumbs_filedata);
1171 while (vflist_thumb_next(vfl));
1174 static gint vflist_thumb_next(ViewFileList *vfl)
1177 FileData *fd = NULL;
1179 /* first check the visible files */
1181 if (GTK_WIDGET_REALIZED(vfl->listview) &&
1182 gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(vfl->listview), 0, 0, &tpath, NULL, NULL, NULL))
1184 GtkTreeModel *store;
1188 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
1189 gtk_tree_model_get_iter(store, &iter, tpath);
1190 gtk_tree_path_free(tpath);
1192 while (!fd && valid && tree_view_row_get_visibility(GTK_TREE_VIEW(vfl->listview), &iter, FALSE) == 0)
1194 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
1195 if (fd->pixbuf) fd = NULL;
1197 valid = gtk_tree_model_iter_next(store, &iter);
1201 /* then find first undone */
1205 GList *work = vfl->list;
1208 FileData *fd_p = work->data;
1213 GList *work2 = fd_p->sidecar_files;
1215 while (work2 && !fd)
1218 if (!fd_p->pixbuf) fd = fd_p;
1219 work2 = work2->next;
1229 vflist_thumb_cleanup(vfl);
1233 vfl->thumbs_count++;
1235 vfl->thumbs_filedata = fd;
1237 thumb_loader_free(vfl->thumbs_loader);
1239 vfl->thumbs_loader = thumb_loader_new(options->thumbnails.max_width, options->thumbnails.max_height);
1240 thumb_loader_set_callbacks(vfl->thumbs_loader,
1241 vflist_thumb_done_cb,
1242 vflist_thumb_error_cb,
1246 if (!thumb_loader_start(vfl->thumbs_loader, fd->path))
1248 /* set icon to unknown, continue */
1249 DEBUG_1("thumb loader start failed %s\n", vfl->thumbs_loader->path);
1250 vflist_thumb_do(vfl, vfl->thumbs_loader, fd);
1258 static void vflist_thumb_update(ViewFileList *vfl)
1260 vflist_thumb_stop(vfl);
1261 if (!vfl->thumbs_enabled) return;
1263 vflist_thumb_status(vfl, 0.0, _("Loading thumbs..."));
1264 vfl->thumbs_running = TRUE;
1266 while (vflist_thumb_next(vfl));
1270 *-----------------------------------------------------------------------------
1272 *-----------------------------------------------------------------------------
1275 FileData *vflist_index_get_data(ViewFileList *vfl, gint row)
1277 return g_list_nth_data(vfl->list, row);
1280 gchar *vflist_index_get_path(ViewFileList *vfl, gint row)
1284 fd = g_list_nth_data(vfl->list, row);
1286 return (fd ? fd->path : NULL);
1289 static gint vflist_row_by_path(ViewFileList *vfl, const gchar *path, FileData **fd)
1294 if (!path) return -1;
1299 FileData *fd_n = work->data;
1300 if (strcmp(path, fd_n->path) == 0)
1313 gint vflist_index_by_path(ViewFileList *vfl, const gchar *path)
1315 return vflist_row_by_path(vfl, path, NULL);
1318 gint vflist_count(ViewFileList *vfl, gint64 *bytes)
1328 FileData *fd = work->data;
1336 return g_list_length(vfl->list);
1339 GList *vflist_get_list(ViewFileList *vfl)
1347 FileData *fd = work->data;
1350 list = g_list_prepend(list, file_data_ref(fd));
1353 return g_list_reverse(list);
1357 *-----------------------------------------------------------------------------
1359 *-----------------------------------------------------------------------------
1362 static gint vflist_row_is_selected(ViewFileList *vfl, FileData *fd)
1364 GtkTreeModel *store;
1365 GtkTreeSelection *selection;
1370 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1371 slist = gtk_tree_selection_get_selected_rows(selection, &store);
1373 while (!found && work)
1375 GtkTreePath *tpath = work->data;
1379 gtk_tree_model_get_iter(store, &iter, tpath);
1380 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd_n, -1);
1381 if (fd_n == fd) found = TRUE;
1384 g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL);
1390 gint vflist_index_is_selected(ViewFileList *vfl, gint row)
1394 fd = vflist_index_get_data(vfl, row);
1395 return vflist_row_is_selected(vfl, fd);
1398 gint vflist_selection_count(ViewFileList *vfl, gint64 *bytes)
1400 GtkTreeModel *store;
1401 GtkTreeSelection *selection;
1405 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1406 slist = gtk_tree_selection_get_selected_rows(selection, &store);
1416 GtkTreePath *tpath = work->data;
1420 gtk_tree_model_get_iter(store, &iter, tpath);
1421 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
1430 count = g_list_length(slist);
1431 g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL);
1437 GList *vflist_selection_get_list(ViewFileList *vfl)
1439 GtkTreeModel *store;
1440 GtkTreeSelection *selection;
1445 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1446 slist = gtk_tree_selection_get_selected_rows(selection, &store);
1450 GtkTreePath *tpath = work->data;
1454 gtk_tree_model_get_iter(store, &iter, tpath);
1455 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
1457 list = g_list_prepend(list, file_data_ref(fd));
1461 g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL);
1464 return g_list_reverse(list);
1467 GList *vflist_selection_get_list_by_index(ViewFileList *vfl)
1469 GtkTreeModel *store;
1470 GtkTreeSelection *selection;
1475 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1476 slist = gtk_tree_selection_get_selected_rows(selection, &store);
1480 GtkTreePath *tpath = work->data;
1484 gtk_tree_model_get_iter(store, &iter, tpath);
1485 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
1487 list = g_list_prepend(list, GINT_TO_POINTER(g_list_index(vfl->list, fd)));
1491 g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL);
1494 return g_list_reverse(list);
1497 void vflist_select_all(ViewFileList *vfl)
1499 GtkTreeSelection *selection;
1501 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1502 gtk_tree_selection_select_all(selection);
1504 vfl->select_fd = NULL;
1507 void vflist_select_none(ViewFileList *vfl)
1509 GtkTreeSelection *selection;
1511 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1512 gtk_tree_selection_unselect_all(selection);
1515 void vflist_select_by_path(ViewFileList *vfl, const gchar *path)
1519 if (vflist_row_by_path(vfl, path, &fd) < 0) return;
1521 vflist_select_by_fd(vfl, fd);
1524 void vflist_select_by_fd(ViewFileList *vfl, FileData *fd)
1528 if (vflist_find_row(vfl, fd, &iter) < 0) return;
1530 tree_view_row_make_visible(GTK_TREE_VIEW(vfl->listview), &iter, TRUE);
1532 if (!vflist_row_is_selected(vfl, fd))
1534 GtkTreeSelection *selection;
1535 GtkTreeModel *store;
1538 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1539 gtk_tree_selection_unselect_all(selection);
1540 gtk_tree_selection_select_iter(selection, &iter);
1541 vflist_move_cursor(vfl, &iter);
1543 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
1544 tpath = gtk_tree_model_get_path(store, &iter);
1545 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vfl->listview), tpath, NULL, FALSE);
1546 gtk_tree_path_free(tpath);
1550 void vflist_mark_to_selection(ViewFileList *vfl, gint mark, MarkToSelectionMode mode)
1552 GtkTreeModel *store;
1554 GtkTreeSelection *selection;
1557 g_assert(mark >= 0 && mark < FILEDATA_MARKS_SIZE);
1559 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview));
1560 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1562 valid = gtk_tree_model_get_iter_first(store, &iter);
1566 gboolean mark_val, selected;
1567 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, FILE_COLUMN_POINTER, &fd, -1);
1569 mark_val = fd->marks[mark];
1570 selected = gtk_tree_selection_iter_is_selected(selection, &iter);
1574 case MTS_MODE_SET: selected = mark_val;
1576 case MTS_MODE_OR: selected = mark_val | selected;
1578 case MTS_MODE_AND: selected = mark_val & selected;
1580 case MTS_MODE_MINUS: selected = !mark_val & selected;
1585 gtk_tree_selection_select_iter(selection, &iter);
1587 gtk_tree_selection_unselect_iter(selection, &iter);
1589 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter);
1593 void vflist_selection_to_mark(ViewFileList *vfl, gint mark, SelectionToMarkMode mode)
1595 GtkTreeModel *store;
1596 GtkTreeSelection *selection;
1600 g_assert(mark >= 0 && mark < FILEDATA_MARKS_SIZE);
1602 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
1603 slist = gtk_tree_selection_get_selected_rows(selection, &store);
1607 GtkTreePath *tpath = work->data;
1611 gtk_tree_model_get_iter(store, &iter, tpath);
1612 gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1);
1616 case STM_MODE_SET: fd->marks[mark] = 1;
1618 case STM_MODE_RESET: fd->marks[mark] = 0;
1620 case STM_MODE_TOGGLE: fd->marks[mark] = !fd->marks[mark];
1624 gtk_tree_store_set(GTK_TREE_STORE(store), &iter, FILE_COLUMN_MARKS + mark, fd->marks[mark], -1);
1628 g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL);
1633 *-----------------------------------------------------------------------------
1635 *-----------------------------------------------------------------------------
1638 static void vflist_listview_set_height(GtkWidget *listview, gint thumb)
1640 GtkTreeViewColumn *column;
1641 GtkCellRenderer *cell;
1644 column = gtk_tree_view_get_column(GTK_TREE_VIEW(listview), FILE_COLUMN_THUMB - 1);
1645 if (!column) return;
1647 gtk_tree_view_column_set_fixed_width(column, ((thumb) ? options->thumbnails.max_width : 4) + 10);
1649 list = gtk_tree_view_column_get_cell_renderers(column);
1654 g_object_set(G_OBJECT(cell), "height", (thumb) ? options->thumbnails.max_height : -1, NULL);
1655 gtk_tree_view_columns_autosize(GTK_TREE_VIEW(listview));
1658 static void vflist_populate_view(ViewFileList *vfl)
1660 GtkTreeStore *store;
1664 GtkTreeRowReference *visible_row = NULL;
1668 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
1669 thumbs = vfl->thumbs_enabled;
1671 vflist_thumb_stop(vfl);
1675 gtk_tree_store_clear(store);
1676 vflist_send_update(vfl);
1680 if (GTK_WIDGET_REALIZED(vfl->listview) &&
1681 gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(vfl->listview), 0, 0, &tpath, NULL, NULL, NULL))
1683 visible_row = gtk_tree_row_reference_new(GTK_TREE_MODEL(store), tpath);
1684 gtk_tree_path_free(tpath);
1687 vflist_listview_set_height(vfl->listview, thumbs);
1689 valid = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter);
1695 FileData *fd = work->data;
1700 FileData *old_fd = NULL;
1704 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, FILE_COLUMN_POINTER, &old_fd, -1);
1712 match = filelist_sort_compare_filedata_full(fd, old_fd, vfl->sort_method, vfl->sort_ascend);
1714 match = -1; /* probably should not happen*/
1729 gtk_tree_store_insert_before(store, &new, NULL, &iter);
1733 gtk_tree_store_append(store, &new, NULL);
1735 vflist_setup_iter_with_sidecars(vfl, store, &new, fd);
1741 valid = gtk_tree_store_remove(store, &iter);
1745 vflist_setup_iter_with_sidecars(vfl, store, &iter, fd);
1747 if (valid) valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter);
1757 valid = gtk_tree_store_remove(store, &iter);
1762 if (gtk_tree_row_reference_valid(visible_row))
1764 tpath = gtk_tree_row_reference_get_path(visible_row);
1765 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vfl->listview), tpath, NULL, TRUE, 0.0, 0.0);
1766 gtk_tree_path_free(tpath);
1768 gtk_tree_row_reference_free(visible_row);
1771 vflist_send_update(vfl);
1772 vflist_thumb_update(vfl);
1775 gint vflist_refresh(ViewFileList *vfl)
1780 old_list = vfl->list;
1785 ret = filelist_read(vfl->path, &vfl->list, NULL);
1788 vfl->list = filelist_sort(vfl->list, vfl->sort_method, vfl->sort_ascend);
1789 vflist_populate_view(vfl);
1791 filelist_free(old_list);
1796 /* this overrides the low default of a GtkCellRenderer from 100 to CELL_HEIGHT_OVERRIDE, something sane for our purposes */
1798 #define CELL_HEIGHT_OVERRIDE 512
1800 static void cell_renderer_height_override(GtkCellRenderer *renderer)
1804 spec = g_object_class_find_property(G_OBJECT_GET_CLASS(G_OBJECT(renderer)), "height");
1805 if (spec && G_IS_PARAM_SPEC_INT(spec))
1807 GParamSpecInt *spec_int;
1809 spec_int = G_PARAM_SPEC_INT(spec);
1810 if (spec_int->maximum < CELL_HEIGHT_OVERRIDE) spec_int->maximum = CELL_HEIGHT_OVERRIDE;
1814 static GdkColor *vflist_listview_color_shifted(GtkWidget *widget)
1816 static GdkColor color;
1817 static GtkWidget *done = NULL;
1823 style = gtk_widget_get_style(widget);
1824 memcpy(&color, &style->base[GTK_STATE_NORMAL], sizeof(color));
1825 shift_color(&color, -1, 0);
1832 static void vflist_listview_color_cb(GtkTreeViewColumn *tree_column, GtkCellRenderer *cell,
1833 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer data)
1835 ViewFileList *vfl = data;
1838 gtk_tree_model_get(tree_model, iter, FILE_COLUMN_COLOR, &set, -1);
1839 g_object_set(G_OBJECT(cell),
1840 "cell-background-gdk", vflist_listview_color_shifted(vfl->listview),
1841 "cell-background-set", set, NULL);
1844 static void vflist_listview_add_column(ViewFileList *vfl, gint n, const gchar *title, gint image, gint right_justify, gint expand)
1846 GtkTreeViewColumn *column;
1847 GtkCellRenderer *renderer;
1849 column = gtk_tree_view_column_new();
1850 gtk_tree_view_column_set_title(column, title);
1851 gtk_tree_view_column_set_min_width(column, 4);
1855 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY);
1856 renderer = gtk_cell_renderer_text_new();
1859 g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL);
1861 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1862 gtk_tree_view_column_add_attribute(column, renderer, "text", n);
1864 gtk_tree_view_column_set_expand(column, TRUE);
1868 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_FIXED);
1869 renderer = gtk_cell_renderer_pixbuf_new();
1870 cell_renderer_height_override(renderer);
1871 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1872 gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", n);
1875 gtk_tree_view_column_set_cell_data_func(column, renderer, vflist_listview_color_cb, vfl, NULL);
1876 g_object_set_data (G_OBJECT (column), "column_store_idx", GUINT_TO_POINTER(n));
1877 g_object_set_data (G_OBJECT (renderer), "column_store_idx", GUINT_TO_POINTER(n));
1879 gtk_tree_view_append_column(GTK_TREE_VIEW(vfl->listview), column);
1882 static void vflist_listview_mark_toggled(GtkCellRendererToggle *cell, gchar *path_str, GtkTreeStore *store)
1884 GtkTreePath *path = gtk_tree_path_new_from_string(path_str);
1890 if (!path || !gtk_tree_model_get_iter(GTK_TREE_MODEL(store), &iter, path))
1893 col_idx = GPOINTER_TO_INT(g_object_get_data (G_OBJECT(cell), "column_store_idx"));
1895 g_assert(col_idx >= FILE_COLUMN_MARKS && col_idx <= FILE_COLUMN_MARKS_LAST);
1897 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, FILE_COLUMN_POINTER, &fd, col_idx, &mark, -1);
1899 fd->marks[col_idx - FILE_COLUMN_MARKS] = mark;
1901 gtk_tree_store_set(store, &iter, col_idx, mark, -1);
1902 gtk_tree_path_free(path);
1905 static void vflist_listview_add_column_toggle(ViewFileList *vfl, gint n, const gchar *title)
1907 GtkTreeViewColumn *column;
1908 GtkCellRenderer *renderer;
1909 GtkTreeStore *store;
1912 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
1914 renderer = gtk_cell_renderer_toggle_new();
1915 column = gtk_tree_view_column_new_with_attributes(title, renderer, "active", n, NULL);
1917 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_FIXED);
1918 g_object_set_data (G_OBJECT (column), "column_store_idx", GUINT_TO_POINTER(n));
1919 g_object_set_data (G_OBJECT (renderer), "column_store_idx", GUINT_TO_POINTER(n));
1921 index = gtk_tree_view_append_column(GTK_TREE_VIEW(vfl->listview), column);
1922 gtk_tree_view_column_set_fixed_width(column, 16);
1923 gtk_tree_view_column_set_visible(column, vfl->marks_enabled);
1926 g_signal_connect(G_OBJECT(renderer), "toggled", G_CALLBACK(vflist_listview_mark_toggled), store);
1930 *-----------------------------------------------------------------------------
1932 *-----------------------------------------------------------------------------
1935 gint vflist_set_path(ViewFileList *vfl, const gchar *path)
1937 GtkTreeStore *store;
1939 if (!path) return FALSE;
1940 if (vfl->path && strcmp(path, vfl->path) == 0) return TRUE;
1943 vfl->path = g_strdup(path);
1945 /* force complete reload */
1946 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
1947 gtk_tree_store_clear(store);
1949 filelist_free(vfl->list);
1952 return vflist_refresh(vfl);
1955 static void vflist_destroy_cb(GtkWidget *widget, gpointer data)
1957 ViewFileList *vfl = data;
1961 g_signal_handlers_disconnect_matched(G_OBJECT(vfl->popup), G_SIGNAL_MATCH_DATA,
1962 0, 0, 0, NULL, vfl);
1963 gtk_widget_destroy(vfl->popup);
1966 vflist_select_idle_cancel(vfl);
1967 vflist_thumb_stop(vfl);
1970 filelist_free(vfl->list);
1974 ViewFileList *vflist_new(const gchar *path, gint thumbs)
1977 GtkTreeStore *store;
1978 GtkTreeSelection *selection;
1980 GType flist_types[FILE_COLUMN_COUNT];
1983 vfl = g_new0(ViewFileList, 1);
1987 vfl->click_fd = NULL;
1988 vfl->select_fd = NULL;
1989 vfl->sort_method = SORT_NAME;
1990 vfl->sort_ascend = TRUE;
1991 vfl->thumbs_enabled = thumbs;
1993 vfl->thumbs_running = FALSE;
1994 vfl->thumbs_count = 0;
1995 vfl->thumbs_loader = NULL;
1996 vfl->thumbs_filedata = NULL;
1998 vfl->select_idle_id = -1;
2002 vfl->widget = gtk_scrolled_window_new(NULL, NULL);
2003 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vfl->widget), GTK_SHADOW_IN);
2004 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vfl->widget),
2005 GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
2006 g_signal_connect(G_OBJECT(vfl->widget), "destroy",
2007 G_CALLBACK(vflist_destroy_cb), vfl);
2009 flist_types[FILE_COLUMN_POINTER] = G_TYPE_POINTER;
2010 flist_types[FILE_COLUMN_THUMB] = GDK_TYPE_PIXBUF;
2011 flist_types[FILE_COLUMN_NAME] = G_TYPE_STRING;
2012 flist_types[FILE_COLUMN_SIDECARS] = G_TYPE_STRING;
2013 flist_types[FILE_COLUMN_SIZE] = G_TYPE_STRING;
2014 flist_types[FILE_COLUMN_DATE] = G_TYPE_STRING;
2015 flist_types[FILE_COLUMN_COLOR] = G_TYPE_BOOLEAN;
2016 for (i = FILE_COLUMN_MARKS; i < FILE_COLUMN_MARKS + FILEDATA_MARKS_SIZE; i++)
2017 flist_types[i] = G_TYPE_BOOLEAN;
2019 store = gtk_tree_store_newv(FILE_COLUMN_COUNT, flist_types);
2021 vfl->listview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
2022 g_object_unref(store);
2024 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
2025 gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_MULTIPLE);
2026 gtk_tree_selection_set_select_function(selection, vflist_select_cb, vfl, NULL);
2028 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vfl->listview), FALSE);
2029 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vfl->listview), FALSE);
2031 vflist_listview_add_column(vfl, FILE_COLUMN_THUMB, "", TRUE, FALSE, FALSE);
2033 for(i = 0; i < FILEDATA_MARKS_SIZE;i++)
2034 vflist_listview_add_column_toggle(vfl, i + FILE_COLUMN_MARKS, "");
2036 vflist_listview_add_column(vfl, FILE_COLUMN_NAME, _("Name"), FALSE, FALSE, FALSE);
2037 vflist_listview_add_column(vfl, FILE_COLUMN_SIDECARS, _("SC"), FALSE, FALSE, FALSE);
2039 vflist_listview_add_column(vfl, FILE_COLUMN_SIZE, _("Size"), FALSE, TRUE, FALSE);
2040 vflist_listview_add_column(vfl, FILE_COLUMN_DATE, _("Date"), FALSE, TRUE, FALSE);
2043 g_signal_connect(G_OBJECT(vfl->listview), "key_press_event",
2044 G_CALLBACK(vflist_press_key_cb), vfl);
2046 gtk_container_add (GTK_CONTAINER(vfl->widget), vfl->listview);
2047 gtk_widget_show(vfl->listview);
2049 vflist_dnd_init(vfl);
2051 g_signal_connect(G_OBJECT(vfl->listview), "button_press_event",
2052 G_CALLBACK(vflist_press_cb), vfl);
2053 g_signal_connect(G_OBJECT(vfl->listview), "button_release_event",
2054 G_CALLBACK(vflist_release_cb), vfl);
2057 if (path) vflist_set_path(vfl, path);
2062 void vflist_set_status_func(ViewFileList *vfl,
2063 void (*func)(ViewFileList *vfl, gpointer data), gpointer data)
2065 vfl->func_status = func;
2066 vfl->data_status = data;
2069 void vflist_set_thumb_status_func(ViewFileList *vfl,
2070 void (*func)(ViewFileList *vfl, gdouble val, const gchar *text, gpointer data),
2073 vfl->func_thumb_status = func;
2074 vfl->data_thumb_status = data;
2077 void vflist_thumb_set(ViewFileList *vfl, gint enable)
2079 if (vfl->thumbs_enabled == enable) return;
2081 vfl->thumbs_enabled = enable;
2082 vflist_refresh(vfl);
2085 void vflist_marks_set(ViewFileList *vfl, gint enable)
2087 GList *columns, *work;
2089 if (vfl->marks_enabled == enable) return;
2091 vfl->marks_enabled = enable;
2093 columns = gtk_tree_view_get_columns(GTK_TREE_VIEW(vfl->listview));
2098 GtkTreeViewColumn *column = work->data;
2099 gint col_idx = GPOINTER_TO_INT(g_object_get_data (G_OBJECT(column), "column_store_idx"));
2102 if (col_idx <= FILE_COLUMN_MARKS_LAST && col_idx >= FILE_COLUMN_MARKS)
2103 gtk_tree_view_column_set_visible(column, enable);
2106 g_list_free(columns);
2107 //vflist_refresh(vfl);
2110 void vflist_set_layout(ViewFileList *vfl, LayoutWindow *layout)
2112 vfl->layout = layout;
2116 *-----------------------------------------------------------------------------
2117 * maintenance (for rename, move, remove)
2118 *-----------------------------------------------------------------------------
2121 static gint vflist_maint_find_closest(ViewFileList *vfl, gint row, gint count, GList *ignore_list)
2131 gint f = vflist_index_by_path(vfl, work->data);
2132 if (f >= 0) list = g_list_prepend(list, GINT_TO_POINTER(f));
2142 gpointer p = work->data;
2144 if (row == GPOINTER_TO_INT(p))
2149 if (rev == GPOINTER_TO_INT(p))
2154 if (!c) list = g_list_remove(list, p);
2162 if (row > count - 1)
2175 gint vflist_maint_renamed(ViewFileList *vfl, FileData *fd)
2181 if (g_list_index(vfl->list, fd) < 0) return FALSE;
2183 source_base = remove_level_from_path(fd->change->source);
2184 dest_base = remove_level_from_path(fd->change->dest);
2187 if (strcmp(source_base, dest_base) == 0)
2189 GtkTreeStore *store;
2191 GtkTreeIter position;
2195 old_row = g_list_index(vfl->list, fd);
2197 vfl->list = g_list_remove(vfl->list, fd);
2199 vfl->list = filelist_insert_sort(vfl->list, fd, vfl->sort_method, vfl->sort_ascend);
2200 n = g_list_index(vfl->list, fd);
2202 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
2203 if (vflist_find_row(vfl, fd, &iter) >= 0 &&
2204 gtk_tree_model_iter_nth_child(GTK_TREE_MODEL(store), &position, NULL, n))
2208 gtk_tree_store_move_before(store, &iter, &position);
2212 gtk_tree_store_move_after(store, &iter, &position);
2215 gtk_tree_store_set(store, &iter, FILE_COLUMN_NAME, fd->name, -1);
2221 ret = vflist_maint_removed(vfl, fd, NULL);
2224 g_free(source_base);
2230 gint vflist_maint_removed(ViewFileList *vfl, FileData *fd, GList *ignore_list)
2237 row = g_list_index(vfl->list, fd);
2238 if (row < 0) return FALSE;
2240 if (vflist_index_is_selected(vfl, row) &&
2241 layout_image_get_collection(vfl->layout, NULL) == NULL)
2245 n = vflist_count(vfl, NULL);
2248 new_row = vflist_maint_find_closest(vfl, row, n, ignore_list);
2249 DEBUG_1("row = %d, closest is %d\n", row, new_row);
2262 vflist_select_none(vfl);
2265 fd = vflist_index_get_data(vfl, new_row);
2266 if (vflist_find_row(vfl, fd, &iter) >= 0)
2268 GtkTreeSelection *selection;
2270 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview));
2271 gtk_tree_selection_select_iter(selection, &iter);
2272 vflist_move_cursor(vfl, &iter);
2277 fd = vflist_index_get_data(vfl, row);
2278 if (vflist_find_row(vfl, fd, &iter) >= 0)
2280 GtkTreeStore *store;
2281 store = GTK_TREE_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)));
2282 gtk_tree_store_remove(store, &iter);
2284 list = g_list_nth(vfl->list, row);
2287 /* thumbnail loader check */
2288 if (fd == vfl->thumbs_filedata) vfl->thumbs_filedata = NULL;
2289 if (vfl->thumbs_count > 0) vfl->thumbs_count--;
2291 vfl->list = g_list_remove(vfl->list, fd);
2292 file_data_unref(fd);
2294 vflist_send_update(vfl);
2299 gint vflist_maint_moved(ViewFileList *vfl, FileData *fd, GList *ignore_list)
2304 if (!fd->change->source || !vfl->path) return FALSE;
2306 buf = remove_level_from_path(fd->change->source);
2308 if (strcmp(buf, vfl->path) == 0)
2310 ret = vflist_maint_removed(vfl, fd, ignore_list);