Mercurial > geeqie
view src/view_file_list.c @ 90:dc3c77d027e6
Tue Oct 31 18:06:42 2006 John Ellis <johne@verizon.net>
* info.c: Increase default info window size to 600x400.
* po/be.po: Update Belarusian translation,
submitted by Pavel Piatruk <berserker@neolocation.com>.
* gqview.desktop: Add additional formats to MimeType list.
author | gqview |
---|---|
date | Tue, 31 Oct 2006 23:20:48 +0000 |
parents | 04ff0df3ad2f |
children | f1c8f8632e23 |
line wrap: on
line source
/* * GQview * (C) 2004 John Ellis * * Author: John Ellis * * This software is released under the GNU General Public License (GNU GPL). * Please read the included file COPYING for more information. * This software comes with no warranty of any kind, use at your own risk! */ #include "gqview.h" #include "view_file_list.h" #include "cache_maint.h" #include "dnd.h" #include "editors.h" #include "img-view.h" #include "info.h" #include "layout.h" #include "layout_image.h" #include "menu.h" #include "thumb.h" #include "utilops.h" #include "ui_bookmark.h" #include "ui_fileops.h" #include "ui_menu.h" #include "ui_tree_edit.h" #include <gdk/gdkkeysyms.h> /* for keyboard values */ enum { FILE_COLUMN_POINTER = 0, FILE_COLUMN_THUMB, FILE_COLUMN_NAME, FILE_COLUMN_SIZE, FILE_COLUMN_DATE, FILE_COLUMN_COLOR, FILE_COLUMN_COUNT }; static gint vflist_row_is_selected(ViewFileList *vfl, FileData *fd); static gint vflist_row_rename_cb(TreeEditData *td, const gchar *old, const gchar *new, gpointer data); static void vflist_populate_view(ViewFileList *vfl); /* *----------------------------------------------------------------------------- * signals *----------------------------------------------------------------------------- */ static void vflist_send_update(ViewFileList *vfl) { if (vfl->func_status) vfl->func_status(vfl, vfl->data_status); } /* *----------------------------------------------------------------------------- * misc *----------------------------------------------------------------------------- */ static gint vflist_find_row(ViewFileList *vfl, FileData *fd, GtkTreeIter *iter) { GtkTreeModel *store; gint valid; gint row = 0; store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); valid = gtk_tree_model_get_iter_first(store, iter); while (valid) { FileData *fd_n; gtk_tree_model_get(GTK_TREE_MODEL(store), iter, FILE_COLUMN_POINTER, &fd_n, -1); if (fd_n == fd) return row; valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter); row++; } return -1; } static void vflist_color_set(ViewFileList *vfl, FileData *fd, gint color_set) { GtkTreeModel *store; GtkTreeIter iter; if (vflist_find_row(vfl, fd, &iter) < 0) return; store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); gtk_list_store_set(GTK_LIST_STORE(store), &iter, FILE_COLUMN_COLOR, color_set, -1); } static void vflist_move_cursor(ViewFileList *vfl, GtkTreeIter *iter) { GtkTreeModel *store; GtkTreePath *tpath; store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); tpath = gtk_tree_model_get_path(store, iter); gtk_tree_view_set_cursor(GTK_TREE_VIEW(vfl->listview), tpath, NULL, FALSE); gtk_tree_path_free(tpath); } /* *----------------------------------------------------------------------------- * dnd *----------------------------------------------------------------------------- */ static void vflist_dnd_get(GtkWidget *widget, GdkDragContext *context, GtkSelectionData *selection_data, guint info, guint time, gpointer data) { ViewFileList *vfl = data; GList *list = NULL; gchar *uri_text = NULL; gint total; if (!vfl->click_fd) return; if (vflist_row_is_selected(vfl, vfl->click_fd)) { list = vflist_selection_get_list(vfl); } else { list = g_list_append(NULL, g_strdup(vfl->click_fd->path)); } if (!list) return; uri_text = uri_text_from_list(list, &total, (info == TARGET_TEXT_PLAIN)); path_list_free(list); if (debug) printf(uri_text); gtk_selection_data_set(selection_data, selection_data->target, 8, (guchar *)uri_text, total); g_free(uri_text); } static void vflist_dnd_begin(GtkWidget *widget, GdkDragContext *context, gpointer data) { ViewFileList *vfl = data; vflist_color_set(vfl, vfl->click_fd, TRUE); if (vfl->thumbs_enabled && vfl->click_fd && vfl->click_fd->pixbuf) { gint items; if (vflist_row_is_selected(vfl, vfl->click_fd)) items = vflist_selection_count(vfl, NULL); else items = 1; dnd_set_drag_icon(widget, context, vfl->click_fd->pixbuf, items); } } static void vflist_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data) { ViewFileList *vfl = data; vflist_color_set(vfl, vfl->click_fd, FALSE); if (context->action == GDK_ACTION_MOVE) { vflist_refresh(vfl); } } static void vflist_dnd_init(ViewFileList *vfl) { gtk_drag_source_set(vfl->listview, GDK_BUTTON1_MASK | GDK_BUTTON2_MASK, dnd_file_drag_types, dnd_file_drag_types_count, GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK); g_signal_connect(G_OBJECT(vfl->listview), "drag_data_get", G_CALLBACK(vflist_dnd_get), vfl); g_signal_connect(G_OBJECT(vfl->listview), "drag_begin", G_CALLBACK(vflist_dnd_begin), vfl); g_signal_connect(G_OBJECT(vfl->listview), "drag_end", G_CALLBACK(vflist_dnd_end), vfl); } /* *----------------------------------------------------------------------------- * pop-up menu *----------------------------------------------------------------------------- */ static GList *vflist_pop_menu_file_list(ViewFileList *vfl) { if (!vfl->click_fd) return NULL; if (vflist_row_is_selected(vfl, vfl->click_fd)) { return vflist_selection_get_list(vfl); } return g_list_append(NULL, g_strdup(vfl->click_fd->path)); } static void vflist_pop_menu_edit_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl; gint n; GList *list; vfl = submenu_item_get_data(widget); n = GPOINTER_TO_INT(data); if (!vfl) return; list = vflist_pop_menu_file_list(vfl); start_editor_from_path_list(n, list); path_list_free(list); } static void vflist_pop_menu_info_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; info_window_new(NULL, vflist_pop_menu_file_list(vfl)); } static void vflist_pop_menu_view_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; if (vflist_row_is_selected(vfl, vfl->click_fd)) { GList *list; list = vflist_selection_get_list(vfl); view_window_new_from_list(list); path_list_free(list); } else { const gchar *path; path = vfl->click_fd->path; view_window_new(path); } } static void vflist_pop_menu_copy_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; file_util_copy(NULL, vflist_pop_menu_file_list(vfl), NULL, vfl->listview); } static void vflist_pop_menu_move_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; file_util_move(NULL, vflist_pop_menu_file_list(vfl), NULL, vfl->listview); } static void vflist_pop_menu_rename_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; GList *list; list = vflist_pop_menu_file_list(vfl); if (enable_in_place_rename && list && !list->next && vfl->click_fd) { GtkTreeModel *store; GtkTreeIter iter; path_list_free(list); store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); if (vflist_find_row(vfl, vfl->click_fd, &iter) >= 0) { GtkTreePath *tpath; tpath = gtk_tree_model_get_path(store, &iter); tree_edit_by_path(GTK_TREE_VIEW(vfl->listview), tpath, FILE_COLUMN_NAME -1, vfl->click_fd->name, vflist_row_rename_cb, vfl); gtk_tree_path_free(tpath); } return; } file_util_rename(NULL, list, vfl->listview); } static void vflist_pop_menu_delete_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; file_util_delete(NULL, vflist_pop_menu_file_list(vfl), vfl->listview); } static void vflist_pop_menu_sort_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl; SortType type; vfl = submenu_item_get_data(widget); if (!vfl) return; type = (SortType)GPOINTER_TO_INT(data); if (vfl->layout) { layout_sort_set(vfl->layout, type, vfl->sort_ascend); } else { vflist_sort_set(vfl, type, vfl->sort_ascend); } } static void vflist_pop_menu_sort_ascend_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; if (vfl->layout) { layout_sort_set(vfl->layout, vfl->sort_method, !vfl->sort_ascend); } else { vflist_sort_set(vfl, vfl->sort_method, !vfl->sort_ascend); } } static void vflist_pop_menu_icons_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; if (vfl->layout) layout_views_set(vfl->layout, vfl->layout->tree_view, TRUE); } static void vflist_pop_menu_thumbs_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; vflist_color_set(vfl, vfl->click_fd, FALSE); if (vfl->layout) { layout_thumb_set(vfl->layout, !vfl->thumbs_enabled); } else { vflist_thumb_set(vfl, !vfl->thumbs_enabled); } } static void vflist_pop_menu_refresh_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; vflist_color_set(vfl, vfl->click_fd, FALSE); vflist_refresh(vfl); } static void vflist_popup_destroy_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; vflist_color_set(vfl, vfl->click_fd, FALSE); vfl->click_fd = NULL; vfl->popup = NULL; } static GtkWidget *vflist_pop_menu(ViewFileList *vfl, FileData *fd) { GtkWidget *menu; GtkWidget *item; GtkWidget *submenu; gint active; vflist_color_set(vfl, fd, TRUE); active = (fd != NULL); menu = popup_menu_short_lived(); g_signal_connect(G_OBJECT(menu), "destroy", G_CALLBACK(vflist_popup_destroy_cb), vfl); submenu_add_edit(menu, &item, G_CALLBACK(vflist_pop_menu_edit_cb), vfl); gtk_widget_set_sensitive(item, active); menu_item_add_stock_sensitive(menu, _("_Properties"), GTK_STOCK_PROPERTIES, active, G_CALLBACK(vflist_pop_menu_info_cb), vfl); menu_item_add_stock_sensitive(menu, _("View in _new window"), GTK_STOCK_NEW, active, G_CALLBACK(vflist_pop_menu_view_cb), vfl); menu_item_add_divider(menu); menu_item_add_stock_sensitive(menu, _("_Copy..."), GTK_STOCK_COPY, active, G_CALLBACK(vflist_pop_menu_copy_cb), vfl); menu_item_add_sensitive(menu, _("_Move..."), active, G_CALLBACK(vflist_pop_menu_move_cb), vfl); menu_item_add_sensitive(menu, _("_Rename..."), active, G_CALLBACK(vflist_pop_menu_rename_cb), vfl); menu_item_add_stock_sensitive(menu, _("_Delete..."), GTK_STOCK_DELETE, active, G_CALLBACK(vflist_pop_menu_delete_cb), vfl); menu_item_add_divider(menu); submenu = submenu_add_sort(NULL, G_CALLBACK(vflist_pop_menu_sort_cb), vfl, FALSE, FALSE, TRUE, vfl->sort_method); menu_item_add_divider(submenu); menu_item_add_check(submenu, _("Ascending"), vfl->sort_ascend, G_CALLBACK(vflist_pop_menu_sort_ascend_cb), vfl); item = menu_item_add(menu, _("_Sort"), NULL, NULL); gtk_menu_item_set_submenu(GTK_MENU_ITEM(item), submenu); menu_item_add_check(menu, _("View as _icons"), FALSE, G_CALLBACK(vflist_pop_menu_icons_cb), vfl); menu_item_add_check(menu, _("Show _thumbnails"), vfl->thumbs_enabled, G_CALLBACK(vflist_pop_menu_thumbs_cb), vfl); menu_item_add_stock(menu, _("Re_fresh"), GTK_STOCK_REFRESH, G_CALLBACK(vflist_pop_menu_refresh_cb), vfl); return menu; } /* *----------------------------------------------------------------------------- * callbacks *----------------------------------------------------------------------------- */ static gint vflist_row_rename_cb(TreeEditData *td, const gchar *old, const gchar *new, gpointer data) { ViewFileList *vfl = data; gchar *old_path; gchar *new_path; if (strlen(new) == 0) return FALSE; old_path = concat_dir_and_file(vfl->path, old); new_path = concat_dir_and_file(vfl->path, new); if (strchr(new, '/') != NULL) { gchar *text = g_strdup_printf(_("Invalid file name:\n%s"), new); file_util_warning_dialog(_("Error renaming file"), text, GTK_STOCK_DIALOG_ERROR, vfl->listview); g_free(text); } else if (isfile(new_path)) { gchar *text = g_strdup_printf(_("A file with name %s already exists."), new); file_util_warning_dialog(_("Error renaming file"), text, GTK_STOCK_DIALOG_ERROR, vfl->listview); g_free(text); } else if (!rename_file(old_path, new_path)) { gchar *text = g_strdup_printf(_("Unable to rename file:\n%s\nto:\n%s"), old, new); file_util_warning_dialog(_("Error renaming file"), text, GTK_STOCK_DIALOG_ERROR, vfl->listview); g_free(text); } else { file_maint_renamed(old_path, new_path); } g_free(old_path); g_free(new_path); return FALSE; } static void vflist_menu_position_cb(GtkMenu *menu, gint *x, gint *y, gboolean *push_in, gpointer data) { ViewFileList *vfl = data; GtkTreeModel *store; GtkTreeIter iter; GtkTreePath *tpath; gint cw, ch; if (vflist_find_row(vfl, vfl->click_fd, &iter) < 0) return; store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); tpath = gtk_tree_model_get_path(store, &iter); tree_view_get_cell_clamped(GTK_TREE_VIEW(vfl->listview), tpath, FILE_COLUMN_NAME - 1, TRUE, x, y, &cw, &ch); gtk_tree_path_free(tpath); *y += ch; popup_menu_position_clamp(menu, x, y, 0); } static gint vflist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data) { ViewFileList *vfl = data; GtkTreePath *tpath; if (event->keyval != GDK_Menu) return FALSE; gtk_tree_view_get_cursor(GTK_TREE_VIEW(vfl->listview), &tpath, NULL); if (tpath) { GtkTreeModel *store; GtkTreeIter iter; store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget)); gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &vfl->click_fd, -1); gtk_tree_path_free(tpath); } else { vfl->click_fd = NULL; } vfl->popup = vflist_pop_menu(vfl, vfl->click_fd); gtk_menu_popup(GTK_MENU(vfl->popup), NULL, NULL, vflist_menu_position_cb, vfl, 0, GDK_CURRENT_TIME); return TRUE; } static gint vflist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data) { ViewFileList *vfl = data; GtkTreePath *tpath; GtkTreeIter iter; FileData *fd = NULL; if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y, &tpath, NULL, NULL, NULL)) { GtkTreeModel *store; store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget)); gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1); #if 0 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE); #endif gtk_tree_path_free(tpath); } vfl->click_fd = fd; if (bevent->button == 3) { vfl->popup = vflist_pop_menu(vfl, vfl->click_fd); gtk_menu_popup(GTK_MENU(vfl->popup), NULL, NULL, NULL, NULL, bevent->button, bevent->time); return TRUE; } if (!fd) return FALSE; if (bevent->button == 2) { if (!vflist_row_is_selected(vfl, fd)) { vflist_color_set(vfl, fd, TRUE); } return TRUE; } if (bevent->button == 1 && bevent->type == GDK_BUTTON_PRESS && !(bevent->state & GDK_SHIFT_MASK ) && !(bevent->state & GDK_CONTROL_MASK ) && vflist_row_is_selected(vfl, fd)) { gtk_widget_grab_focus(widget); return TRUE; } #if 0 if (bevent->button == 1 && bevent->type == GDK_2BUTTON_PRESS) { if (vfl->layout) layout_image_full_screen_start(vfl->layout); } #endif return FALSE; } static gint vflist_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data) { ViewFileList *vfl = data; GtkTreePath *tpath; GtkTreeIter iter; FileData *fd = NULL; if (bevent->button == 2) { vflist_color_set(vfl, vfl->click_fd, FALSE); } if (bevent->button != 1 && bevent->button != 2) { return TRUE; } if ((bevent->x != 0 || bevent->y != 0) && gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y, &tpath, NULL, NULL, NULL)) { GtkTreeModel *store; store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget)); gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1); gtk_tree_path_free(tpath); } if (bevent->button == 2) { if (fd && vfl->click_fd == fd) { GtkTreeSelection *selection; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget)); if (vflist_row_is_selected(vfl, fd)) { gtk_tree_selection_unselect_iter(selection, &iter); } else { gtk_tree_selection_select_iter(selection, &iter); } } return TRUE; } if (fd && vfl->click_fd == fd && !(bevent->state & GDK_SHIFT_MASK ) && !(bevent->state & GDK_CONTROL_MASK ) && vflist_row_is_selected(vfl, fd)) { GtkTreeSelection *selection; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget)); gtk_tree_selection_unselect_all(selection); gtk_tree_selection_select_iter(selection, &iter); vflist_move_cursor(vfl, &iter); return TRUE; } return FALSE; } static void vflist_select_image(ViewFileList *vfl, gint row) { const gchar *path; const gchar *read_ahead_path = NULL; path = vflist_index_get_path(vfl, row); if (!path) return; if (path && enable_read_ahead) { FileData *fd; if (row > vflist_index_by_path(vfl, layout_image_get_path(vfl->layout)) && row + 1 < vflist_count(vfl, NULL)) { fd = vflist_index_get_data(vfl, row + 1); } else if (row > 0) { fd = vflist_index_get_data(vfl, row - 1); } else { fd = NULL; } if (fd) read_ahead_path = fd->path; } layout_image_set_with_ahead(vfl->layout, path, read_ahead_path); } static gint vflist_select_idle_cb(gpointer data) { ViewFileList *vfl = data; if (!vfl->layout) { vfl->select_idle_id = -1; return FALSE; } vflist_send_update(vfl); if (vfl->select_fd) { vflist_select_image(vfl, g_list_index(vfl->list, vfl->select_fd)); vfl->select_fd = NULL; } vfl->select_idle_id = -1; return FALSE; } static void vflist_select_idle_cancel(ViewFileList *vfl) { if (vfl->select_idle_id != -1) g_source_remove(vfl->select_idle_id); vfl->select_idle_id = -1; } static gboolean vflist_select_cb(GtkTreeSelection *selection, GtkTreeModel *store, GtkTreePath *tpath, gboolean path_currently_selected, gpointer data) { ViewFileList *vfl = data; GtkTreeIter iter; if (!path_currently_selected && gtk_tree_model_get_iter(store, &iter, tpath)) { gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &vfl->select_fd, -1); } else { vfl->select_fd = NULL; } if (vfl->layout && vfl->select_idle_id == -1) { vfl->select_idle_id = g_idle_add(vflist_select_idle_cb, vfl); } return TRUE; } /* *----------------------------------------------------------------------------- * misc *----------------------------------------------------------------------------- */ void vflist_sort_set(ViewFileList *vfl, SortType type, gint ascend) { GtkListStore *store; GList *work; if (vfl->sort_method == type && vfl->sort_ascend == ascend) return; vfl->sort_method = type; vfl->sort_ascend = ascend; if (!vfl->list) return; vfl->list = filelist_sort(vfl->list, vfl->sort_method, vfl->sort_ascend); store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview))); /* reorder the treeview, maintaining current selection */ work = g_list_last(vfl->list); while (work) { FileData *fd; GtkTreeIter iter; fd = work->data; if (vflist_find_row(vfl, fd, &iter) >= 0) { gtk_list_store_move_after(store, &iter, NULL); } work = work->prev; } } /* *----------------------------------------------------------------------------- * thumb updates *----------------------------------------------------------------------------- */ static gint vflist_thumb_next(ViewFileList *vfl); static void vflist_thumb_status(ViewFileList *vfl, gdouble val, const gchar *text) { if (vfl->func_thumb_status) { vfl->func_thumb_status(vfl, val, text, vfl->data_thumb_status); } } static void vflist_thumb_cleanup(ViewFileList *vfl) { vflist_thumb_status(vfl, 0.0, NULL); vfl->thumbs_count = 0; vfl->thumbs_running = FALSE; thumb_loader_free(vfl->thumbs_loader); vfl->thumbs_loader = NULL; vfl->thumbs_filedata = NULL; } static void vflist_thumb_stop(ViewFileList *vfl) { if (vfl->thumbs_running) vflist_thumb_cleanup(vfl); } static void vflist_thumb_do(ViewFileList *vfl, ThumbLoader *tl, FileData *fd) { GtkListStore *store; GtkTreeIter iter; if (!fd || vflist_find_row(vfl, fd, &iter) < 0) return; if (fd->pixbuf) g_object_unref(fd->pixbuf); fd->pixbuf = thumb_loader_get_pixbuf(tl, TRUE); store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview))); gtk_list_store_set(store, &iter, FILE_COLUMN_THUMB, fd->pixbuf, -1); vflist_thumb_status(vfl, (gdouble)(vfl->thumbs_count) / g_list_length(vfl->list), _("Loading thumbs...")); } static void vflist_thumb_error_cb(ThumbLoader *tl, gpointer data) { ViewFileList *vfl = data; if (vfl->thumbs_filedata && vfl->thumbs_loader == tl) { vflist_thumb_do(vfl, tl, vfl->thumbs_filedata); } while (vflist_thumb_next(vfl)); } static void vflist_thumb_done_cb(ThumbLoader *tl, gpointer data) { ViewFileList *vfl = data; if (vfl->thumbs_filedata && vfl->thumbs_loader == tl) { vflist_thumb_do(vfl, tl, vfl->thumbs_filedata); } while (vflist_thumb_next(vfl)); } static gint vflist_thumb_next(ViewFileList *vfl) { GtkTreePath *tpath; FileData *fd = NULL; /* first check the visible files */ if (GTK_WIDGET_REALIZED(vfl->listview) && gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(vfl->listview), 0, 0, &tpath, NULL, NULL, NULL)) { GtkTreeModel *store; GtkTreeIter iter; gint valid = TRUE; store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_path_free(tpath); while (!fd && valid && tree_view_row_get_visibility(GTK_TREE_VIEW(vfl->listview), &iter, FALSE) == 0) { gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1); if (fd->pixbuf) fd = NULL; valid = gtk_tree_model_iter_next(store, &iter); } } /* then find first undone */ if (!fd) { GList *work = vfl->list; while (work && !fd) { FileData *fd_p = work->data; work = work->next; if (!fd_p->pixbuf) fd = fd_p; } } if (!fd) { /* done */ vflist_thumb_cleanup(vfl); return FALSE; } vfl->thumbs_count++; vfl->thumbs_filedata = fd; thumb_loader_free(vfl->thumbs_loader); vfl->thumbs_loader = thumb_loader_new(thumb_max_width, thumb_max_height); thumb_loader_set_callbacks(vfl->thumbs_loader, vflist_thumb_done_cb, vflist_thumb_error_cb, NULL, vfl); if (!thumb_loader_start(vfl->thumbs_loader, fd->path)) { /* set icon to unknown, continue */ if (debug) printf("thumb loader start failed %s\n", vfl->thumbs_loader->path); vflist_thumb_do(vfl, vfl->thumbs_loader, fd); return TRUE; } return FALSE; } static void vflist_thumb_update(ViewFileList *vfl) { vflist_thumb_stop(vfl); if (!vfl->thumbs_enabled) return; vflist_thumb_status(vfl, 0.0, _("Loading thumbs...")); vfl->thumbs_running = TRUE; while (vflist_thumb_next(vfl)); } /* *----------------------------------------------------------------------------- * row stuff *----------------------------------------------------------------------------- */ FileData *vflist_index_get_data(ViewFileList *vfl, gint row) { return g_list_nth_data(vfl->list, row); } gchar *vflist_index_get_path(ViewFileList *vfl, gint row) { FileData *fd; fd = g_list_nth_data(vfl->list, row); return (fd ? fd->path : NULL); } static gint vflist_row_by_path(ViewFileList *vfl, const gchar *path, FileData **fd) { gint p = 0; GList *work; if (!path) return -1; work = vfl->list; while (work) { FileData *fd_n = work->data; if (strcmp(path, fd_n->path) == 0) { if (fd) *fd = fd_n; return p; } work = work->next; p++; } if (fd) *fd = NULL; return -1; } gint vflist_index_by_path(ViewFileList *vfl, const gchar *path) { return vflist_row_by_path(vfl, path, NULL); } gint vflist_count(ViewFileList *vfl, gint64 *bytes) { if (bytes) { gint64 b = 0; GList *work; work = vfl->list; while (work) { FileData *fd = work->data; work = work->next; b += fd->size; } *bytes = b; } return g_list_length(vfl->list); } GList *vflist_get_list(ViewFileList *vfl) { GList *list = NULL; GList *work; work = vfl->list; while (work) { FileData *fd = work->data; work = work->next; list = g_list_prepend(list, g_strdup(fd->path)); } return g_list_reverse(list); } /* *----------------------------------------------------------------------------- * selections *----------------------------------------------------------------------------- */ static gint vflist_row_is_selected(ViewFileList *vfl, FileData *fd) { GtkTreeModel *store; GtkTreeSelection *selection; GList *slist; GList *work; gint found = FALSE; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); slist = gtk_tree_selection_get_selected_rows(selection, &store); work = slist; while (!found && work) { GtkTreePath *tpath = work->data; FileData *fd_n; GtkTreeIter iter; gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd_n, -1); if (fd_n == fd) found = TRUE; work = work->next; } g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL); g_list_free(slist); return found; } gint vflist_index_is_selected(ViewFileList *vfl, gint row) { FileData *fd; fd = vflist_index_get_data(vfl, row); return vflist_row_is_selected(vfl, fd); } gint vflist_selection_count(ViewFileList *vfl, gint64 *bytes) { GtkTreeModel *store; GtkTreeSelection *selection; GList *slist; gint count; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); slist = gtk_tree_selection_get_selected_rows(selection, &store); if (bytes) { gint64 b = 0; GList *work; work = slist; while (work) { GtkTreePath *tpath = work->data; GtkTreeIter iter; FileData *fd; gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1); b += fd->size; work = work->next; } *bytes = b; } count = g_list_length(slist); g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL); g_list_free(slist); return count; } GList *vflist_selection_get_list(ViewFileList *vfl) { GtkTreeModel *store; GtkTreeSelection *selection; GList *slist; GList *list = NULL; GList *work; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); slist = gtk_tree_selection_get_selected_rows(selection, &store); work = slist; while (work) { GtkTreePath *tpath = work->data; FileData *fd; GtkTreeIter iter; gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1); list = g_list_prepend(list, g_strdup(fd->path)); work = work->next; } g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL); g_list_free(slist); return g_list_reverse(list); } GList *vflist_selection_get_list_by_index(ViewFileList *vfl) { GtkTreeModel *store; GtkTreeSelection *selection; GList *slist; GList *list = NULL; GList *work; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); slist = gtk_tree_selection_get_selected_rows(selection, &store); work = slist; while (work) { GtkTreePath *tpath = work->data; FileData *fd; GtkTreeIter iter; gtk_tree_model_get_iter(store, &iter, tpath); gtk_tree_model_get(store, &iter, FILE_COLUMN_POINTER, &fd, -1); list = g_list_prepend(list, GINT_TO_POINTER(g_list_index(vfl->list, fd))); work = work->next; } g_list_foreach(slist, (GFunc)gtk_tree_path_free, NULL); g_list_free(slist); return g_list_reverse(list); } void vflist_select_all(ViewFileList *vfl) { GtkTreeSelection *selection; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); gtk_tree_selection_select_all(selection); vfl->select_fd = NULL; } void vflist_select_none(ViewFileList *vfl) { GtkTreeSelection *selection; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); gtk_tree_selection_unselect_all(selection); } void vflist_select_by_path(ViewFileList *vfl, const gchar *path) { FileData *fd; GtkTreeIter iter; if (vflist_row_by_path(vfl, path, &fd) < 0) return; if (vflist_find_row(vfl, fd, &iter) < 0) return; tree_view_row_make_visible(GTK_TREE_VIEW(vfl->listview), &iter, TRUE); if (!vflist_row_is_selected(vfl, fd)) { GtkTreeSelection *selection; GtkTreeModel *store; GtkTreePath *tpath; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); gtk_tree_selection_unselect_all(selection); gtk_tree_selection_select_iter(selection, &iter); vflist_move_cursor(vfl, &iter); store = gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview)); tpath = gtk_tree_model_get_path(store, &iter); gtk_tree_view_set_cursor(GTK_TREE_VIEW(vfl->listview), tpath, NULL, FALSE); gtk_tree_path_free(tpath); } } /* *----------------------------------------------------------------------------- * core (population) *----------------------------------------------------------------------------- */ static void vflist_listview_set_height(GtkWidget *listview, gint thumb) { GtkTreeViewColumn *column; GtkCellRenderer *cell; GList *list; column = gtk_tree_view_get_column(GTK_TREE_VIEW(listview), FILE_COLUMN_THUMB - 1); if (!column) return; gtk_tree_view_column_set_fixed_width(column, (thumb) ? thumb_max_width : 4); list = gtk_tree_view_column_get_cell_renderers(column); if (!list) return; cell = list->data; g_list_free(list); g_object_set(G_OBJECT(cell), "height", (thumb) ? thumb_max_height : -1, NULL); gtk_tree_view_columns_autosize(GTK_TREE_VIEW(listview)); } static void vflist_populate_view(ViewFileList *vfl) { GtkListStore *store; GtkTreeIter iter; gint thumbs; GList *work; GtkTreeRowReference *visible_row = NULL; GtkTreePath *tpath; gint valid; store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview))); thumbs = vfl->thumbs_enabled; vflist_thumb_stop(vfl); if (!vfl->list) { gtk_list_store_clear(store); vflist_send_update(vfl); return; } if (GTK_WIDGET_REALIZED(vfl->listview) && gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(vfl->listview), 0, 0, &tpath, NULL, NULL, NULL)) { visible_row = gtk_tree_row_reference_new(GTK_TREE_MODEL(store), tpath); gtk_tree_path_free(tpath); } vflist_listview_set_height(vfl->listview, thumbs); valid = gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store), &iter); work = vfl->list; while (work) { gint match; FileData *fd = work->data; gint done = FALSE; while (!done) { FileData *old_fd = NULL; if (valid) { gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, FILE_COLUMN_POINTER, &old_fd, -1); match = CASE_SORT(fd->name, old_fd->name); } else { match = -1; } if (match < 0) { GtkTreeIter new; gchar *size; size = text_from_size(fd->size); if (valid) { gtk_list_store_insert_before(store, &new, &iter); } else { gtk_list_store_append(store, &new); } gtk_list_store_set(store, &new, FILE_COLUMN_POINTER, fd, FILE_COLUMN_THUMB, (thumbs) ? fd->pixbuf : NULL, FILE_COLUMN_NAME, fd->name, FILE_COLUMN_SIZE, size, FILE_COLUMN_DATE, text_from_time(fd->date), FILE_COLUMN_COLOR, FALSE, -1); g_free(size); done = TRUE; } else if (match > 0) { valid = gtk_list_store_remove(store, &iter); } else { gtk_list_store_set(store, &iter, FILE_COLUMN_POINTER, fd, -1); if (fd->date != old_fd->date) { gchar *size; /* update, file changed */ size = text_from_size(fd->size); gtk_list_store_set(store, &iter, FILE_COLUMN_SIZE, size, FILE_COLUMN_DATE, text_from_time(fd->date), -1); g_free(size); } else if (fd != old_fd) { /* preserve thumbnail */ if (fd->pixbuf) g_object_unref(fd->pixbuf); fd->pixbuf = old_fd->pixbuf; if (fd->pixbuf) g_object_ref(fd->pixbuf); } gtk_list_store_set(store, &iter, FILE_COLUMN_THUMB, (thumbs) ? fd->pixbuf : NULL, -1); if (vfl->select_fd == old_fd) vfl->select_fd = fd; if (valid) valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter); done = TRUE; } } work = work->next; } while (valid) { valid = gtk_list_store_remove(store, &iter); } if (visible_row) { if (gtk_tree_row_reference_valid(visible_row)) { tpath = gtk_tree_row_reference_get_path(visible_row); gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vfl->listview), tpath, NULL, TRUE, 0.0, 0.0); gtk_tree_path_free(tpath); } gtk_tree_row_reference_free(visible_row); } vflist_send_update(vfl); vflist_thumb_update(vfl); } gint vflist_refresh(ViewFileList *vfl) { GList *old_list; gint ret = TRUE; old_list = vfl->list; vfl->list = NULL; if (vfl->path) { ret = filelist_read(vfl->path, &vfl->list, NULL); } vfl->list = filelist_sort(vfl->list, vfl->sort_method, vfl->sort_ascend); vflist_populate_view(vfl); filelist_free(old_list); return ret; } /* this overrides the low default of a GtkCellRenderer from 100 to CELL_HEIGHT_OVERRIDE, something sane for our purposes */ #define CELL_HEIGHT_OVERRIDE 512 static void cell_renderer_height_override(GtkCellRenderer *renderer) { GParamSpec *spec; spec = g_object_class_find_property(G_OBJECT_GET_CLASS(G_OBJECT(renderer)), "height"); if (spec && G_IS_PARAM_SPEC_INT(spec)) { GParamSpecInt *spec_int; spec_int = G_PARAM_SPEC_INT(spec); if (spec_int->maximum < CELL_HEIGHT_OVERRIDE) spec_int->maximum = CELL_HEIGHT_OVERRIDE; } } static GdkColor *vflist_listview_color_shifted(GtkWidget *widget) { static GdkColor color; static GtkWidget *done = NULL; if (done != widget) { GtkStyle *style; style = gtk_widget_get_style(widget); memcpy(&color, &style->base[GTK_STATE_NORMAL], sizeof(color)); shift_color(&color, -1, 0); done = widget; } return &color; } static void vflist_listview_color_cb(GtkTreeViewColumn *tree_column, GtkCellRenderer *cell, GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer data) { ViewFileList *vfl = data; gboolean set; gtk_tree_model_get(tree_model, iter, FILE_COLUMN_COLOR, &set, -1); g_object_set(G_OBJECT(cell), "cell-background-gdk", vflist_listview_color_shifted(vfl->listview), "cell-background-set", set, NULL); } static void vflist_listview_add_column(ViewFileList *vfl, gint n, const gchar *title, gint image, gint right_justify) { GtkTreeViewColumn *column; GtkCellRenderer *renderer; column = gtk_tree_view_column_new(); gtk_tree_view_column_set_title(column, title); gtk_tree_view_column_set_min_width(column, 4); if (!image) { gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_GROW_ONLY); renderer = gtk_cell_renderer_text_new(); if (right_justify) { g_object_set(G_OBJECT(renderer), "xalign", 1.0, NULL); } gtk_tree_view_column_pack_start(column, renderer, TRUE); gtk_tree_view_column_add_attribute(column, renderer, "text", n); } else { gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_FIXED); renderer = gtk_cell_renderer_pixbuf_new(); cell_renderer_height_override(renderer); gtk_tree_view_column_pack_start(column, renderer, TRUE); gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", n); } gtk_tree_view_column_set_cell_data_func(column, renderer, vflist_listview_color_cb, vfl, NULL); gtk_tree_view_append_column(GTK_TREE_VIEW(vfl->listview), column); } /* *----------------------------------------------------------------------------- * base *----------------------------------------------------------------------------- */ gint vflist_set_path(ViewFileList *vfl, const gchar *path) { GtkListStore *store; if (!path) return FALSE; if (vfl->path && strcmp(path, vfl->path) == 0) return TRUE; g_free(vfl->path); vfl->path = g_strdup(path); /* force complete reload */ store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview))); gtk_list_store_clear(store); filelist_free(vfl->list); vfl->list = NULL; return vflist_refresh(vfl); } static void vflist_destroy_cb(GtkWidget *widget, gpointer data) { ViewFileList *vfl = data; if (vfl->popup) { g_signal_handlers_disconnect_matched(G_OBJECT(vfl->popup), G_SIGNAL_MATCH_DATA, 0, 0, 0, NULL, vfl); gtk_widget_destroy(vfl->popup); } vflist_select_idle_cancel(vfl); vflist_thumb_stop(vfl); g_free(vfl->path); filelist_free(vfl->list); g_free(vfl); } ViewFileList *vflist_new(const gchar *path, gint thumbs) { ViewFileList *vfl; GtkListStore *store; GtkTreeSelection *selection; vfl = g_new0(ViewFileList, 1); vfl->path = NULL; vfl->list = NULL; vfl->click_fd = NULL; vfl->select_fd = NULL; vfl->sort_method = SORT_NAME; vfl->sort_ascend = TRUE; vfl->thumbs_enabled = thumbs; vfl->thumbs_running = FALSE; vfl->thumbs_count = 0; vfl->thumbs_loader = NULL; vfl->thumbs_filedata = NULL; vfl->select_idle_id = -1; vfl->popup = NULL; vfl->widget = gtk_scrolled_window_new(NULL, NULL); gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vfl->widget), GTK_SHADOW_IN); gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vfl->widget), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS); g_signal_connect(G_OBJECT(vfl->widget), "destroy", G_CALLBACK(vflist_destroy_cb), vfl); store = gtk_list_store_new(6, G_TYPE_POINTER, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN); vfl->listview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store)); g_object_unref(store); selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); gtk_tree_selection_set_mode(GTK_TREE_SELECTION(selection), GTK_SELECTION_MULTIPLE); gtk_tree_selection_set_select_function(selection, vflist_select_cb, vfl, NULL); gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vfl->listview), FALSE); gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vfl->listview), FALSE); vflist_listview_add_column(vfl, FILE_COLUMN_THUMB, "", TRUE, FALSE); vflist_listview_add_column(vfl, FILE_COLUMN_NAME, _("Name"), FALSE, FALSE); vflist_listview_add_column(vfl, FILE_COLUMN_SIZE, _("Size"), FALSE, TRUE); vflist_listview_add_column(vfl, FILE_COLUMN_DATE, _("Date"), FALSE, TRUE); g_signal_connect(G_OBJECT(vfl->listview), "key_press_event", G_CALLBACK(vflist_press_key_cb), vfl); gtk_container_add (GTK_CONTAINER(vfl->widget), vfl->listview); gtk_widget_show(vfl->listview); vflist_dnd_init(vfl); g_signal_connect(G_OBJECT(vfl->listview), "button_press_event", G_CALLBACK(vflist_press_cb), vfl); g_signal_connect(G_OBJECT(vfl->listview), "button_release_event", G_CALLBACK(vflist_release_cb), vfl); if (path) vflist_set_path(vfl, path); return vfl; } void vflist_set_status_func(ViewFileList *vfl, void (*func)(ViewFileList *vfl, gpointer data), gpointer data) { vfl->func_status = func; vfl->data_status = data; } void vflist_set_thumb_status_func(ViewFileList *vfl, void (*func)(ViewFileList *vfl, gdouble val, const gchar *text, gpointer data), gpointer data) { vfl->func_thumb_status = func; vfl->data_thumb_status = data; } void vflist_thumb_set(ViewFileList *vfl, gint enable) { if (vfl->thumbs_enabled == enable) return; vfl->thumbs_enabled = enable; vflist_refresh(vfl); } void vflist_set_layout(ViewFileList *vfl, LayoutWindow *layout) { vfl->layout = layout; } /* *----------------------------------------------------------------------------- * maintenance (for rename, move, remove) *----------------------------------------------------------------------------- */ static gint vflist_maint_find_closest(ViewFileList *vfl, gint row, gint count, GList *ignore_list) { GList *list = NULL; GList *work; gint rev = row - 1; row ++; work = ignore_list; while (work) { gint f = vflist_index_by_path(vfl, work->data); if (f >= 0) list = g_list_prepend(list, GINT_TO_POINTER(f)); work = work->next; } while (list) { gint c = TRUE; work = list; while (work && c) { gpointer p = work->data; work = work->next; if (row == GPOINTER_TO_INT(p)) { row++; c = FALSE; } if (rev == GPOINTER_TO_INT(p)) { rev--; c = FALSE; } if (!c) list = g_list_remove(list, p); } if (c && list) { g_list_free(list); list = NULL; } } if (row > count - 1) { if (rev < 0) return -1; else return rev; } else { return row; } } gint vflist_maint_renamed(ViewFileList *vfl, const gchar *source, const gchar *dest) { gint ret = FALSE; gint row; gchar *source_base; gchar *dest_base; GList *work; FileData *fd; row = vflist_index_by_path(vfl, source); if (row < 0) return FALSE; source_base = remove_level_from_path(source); dest_base = remove_level_from_path(dest); work = g_list_nth(vfl->list, row); fd = work->data; if (strcmp(source_base, dest_base) == 0) { GtkListStore *store; GtkTreeIter iter; GtkTreeIter position; gint old_row; gint n; old_row = g_list_index(vfl->list, fd); vfl->list = g_list_remove(vfl->list, fd); g_free(fd->path); fd->path = g_strdup(dest); fd->name = filename_from_path(fd->path); vfl->list = filelist_insert_sort(vfl->list, fd, vfl->sort_method, vfl->sort_ascend); n = g_list_index(vfl->list, fd); store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview))); row = vflist_find_row(vfl, fd, &iter); if (vflist_find_row(vfl, fd, &iter) >= 0 && gtk_tree_model_iter_nth_child(GTK_TREE_MODEL(store), &position, NULL, n)) { if (old_row >= n) { gtk_list_store_move_before(store, &iter, &position); } else { gtk_list_store_move_after(store, &iter, &position); } } gtk_list_store_set(store, &iter, FILE_COLUMN_NAME, fd->name, -1); ret = TRUE; } else { ret = vflist_maint_removed(vfl, source, NULL); } g_free(source_base); g_free(dest_base); return ret; } gint vflist_maint_removed(ViewFileList *vfl, const gchar *path, GList *ignore_list) { GtkTreeIter iter; GList *list; FileData *fd; gint row; gint new_row = -1; row = vflist_index_by_path(vfl, path); if (row < 0) return FALSE; if (vflist_index_is_selected(vfl, row) && layout_image_get_collection(vfl->layout, NULL) == NULL) { gint n; n = vflist_count(vfl, NULL); if (ignore_list) { new_row = vflist_maint_find_closest(vfl, row, n, ignore_list); if (debug) printf("row = %d, closest is %d\n", row, new_row); } else { if (row + 1 < n) { new_row = row + 1; } else if (row > 0) { new_row = row - 1; } } vflist_select_none(vfl); if (new_row >= 0) { fd = vflist_index_get_data(vfl, new_row); if (vflist_find_row(vfl, fd, &iter) >= 0) { GtkTreeSelection *selection; selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vfl->listview)); gtk_tree_selection_select_iter(selection, &iter); vflist_move_cursor(vfl, &iter); } } } fd = vflist_index_get_data(vfl, row); if (vflist_find_row(vfl, fd, &iter) >= 0) { GtkListStore *store; store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vfl->listview))); gtk_list_store_remove(store, &iter); } list = g_list_nth(vfl->list, row); fd = list->data; /* thumbnail loader check */ if (fd == vfl->thumbs_filedata) vfl->thumbs_filedata = NULL; vfl->list = g_list_remove(vfl->list, fd); file_data_free(fd); vflist_send_update(vfl); return TRUE; } gint vflist_maint_moved(ViewFileList *vfl, const gchar *source, const gchar *dest, GList *ignore_list) { gint ret = FALSE; gchar *buf; if (!source || !vfl->path) return FALSE; buf = remove_level_from_path(source); if (strcmp(buf, vfl->path) == 0) { ret = vflist_maint_removed(vfl, source, ignore_list); } g_free(buf); return ret; }