2 * Copyright (C) 2004 John Ellis
3 * Copyright (C) 2008 - 2016 The Geeqie Team
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
23 #include "view-dir-list.h"
26 #include "ui-fileops.h"
27 #include "ui-tree-edit.h"
30 #define VDLIST(_vd_) ((ViewDirInfoList *)(_vd_->info))
34 *-----------------------------------------------------------------------------
36 *-----------------------------------------------------------------------------
39 gboolean vdlist_find_row(ViewDir *vd, FileData *fd, GtkTreeIter *iter)
44 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
45 valid = gtk_tree_model_get_iter_first(store, iter);
49 gtk_tree_model_get(GTK_TREE_MODEL(store), iter, DIR_COLUMN_POINTER, &fd_n, -1);
50 if (fd_n == fd) return TRUE;
52 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter);
59 FileData *vdlist_row_by_path(ViewDir *vd, const gchar *path, gint *row)
71 work = VDLIST(vd)->list;
74 FileData *fd = static_cast<FileData *>(work->data);
75 if (strcmp(fd->path, path) == 0)
89 *-----------------------------------------------------------------------------
91 *-----------------------------------------------------------------------------
94 static void vdlist_scroll_to_row(ViewDir *vd, FileData *fd, gfloat y_align)
98 if (gtk_widget_get_realized(vd->view) && vd_find_row(vd, fd, &iter))
103 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
104 tpath = gtk_tree_model_get_path(store, &iter);
105 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vd->view), tpath, NULL, TRUE, y_align, 0.0);
106 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vd->view), tpath, NULL, FALSE);
107 gtk_tree_path_free(tpath);
109 if (!gtk_widget_has_focus(vd->view)) gtk_widget_grab_focus(vd->view);
114 *-----------------------------------------------------------------------------
116 *-----------------------------------------------------------------------------
119 const gchar *vdlist_row_get_path(ViewDir *vd, gint row)
123 fd = static_cast<FileData *>(g_list_nth_data(VDLIST(vd)->list, row));
125 if (fd) return fd->path;
130 static gboolean vdlist_populate(ViewDir *vd, gboolean clear)
140 SortType sort_type = SORT_NAME;
141 gboolean sort_ascend = TRUE;
146 sort_type = vd->layout->options.dir_view_list_sort.method;
147 sort_ascend = vd->layout->options.dir_view_list_sort.ascend;
150 old_list = VDLIST(vd)->list;
152 ret = filelist_read(vd->dir_fd, NULL, &VDLIST(vd)->list);
153 VDLIST(vd)->list = filelist_sort(VDLIST(vd)->list, sort_type, sort_ascend);
157 if (options->file_filter.show_parent_directory && strcmp(vd->dir_fd->path, G_DIR_SEPARATOR_S) != 0)
159 filepath = g_build_filename(vd->dir_fd->path, "..", NULL);
160 fd = file_data_new_dir(filepath);
161 VDLIST(vd)->list = g_list_prepend(VDLIST(vd)->list, fd);
165 if (options->file_filter.show_dot_directory)
167 filepath = g_build_filename(vd->dir_fd->path, ".", NULL);
168 fd = file_data_new_dir(filepath);
169 VDLIST(vd)->list = g_list_prepend(VDLIST(vd)->list, fd);
173 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view)));
174 if (clear) gtk_list_store_clear(store);
176 valid = gtk_tree_model_iter_children(GTK_TREE_MODEL(store), &iter, NULL);
178 work = VDLIST(vd)->list;
183 const gchar *date = "";
184 gboolean done = FALSE;
186 fd = static_cast<FileData *>(work->data);
188 if (access_file(fd->path, R_OK | X_OK) && fd->name)
190 if (islink(fd->path))
192 pixbuf = vd->pf->link;
194 else if (fd->name[0] == '.' && fd->name[1] == '\0')
196 pixbuf = vd->pf->open;
198 else if (fd->name[0] == '.' && fd->name[1] == '.' && fd->name[2] == '\0')
200 pixbuf = vd->pf->parent;
202 else if (!access_file(fd->path, W_OK) )
204 pixbuf = vd->pf->read_only;
208 pixbuf = vd->pf->close;
209 if (vd->layout && vd->layout->options.show_directory_date)
210 date = text_from_time(fd->date);
215 pixbuf = vd->pf->deny;
220 FileData *old_fd = NULL;
224 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter,
225 DIR_COLUMN_POINTER, &old_fd,
234 match = filelist_sort_compare_filedata_full(fd, old_fd, sort_type, sort_ascend);
236 if (match == 0) g_warning("multiple fd for the same path");
245 if (islink(fd->path))
247 link = realpath(fd->path, NULL);
256 GtkTreeIter new_iter;
260 gtk_list_store_insert_before(store, &new_iter, &iter);
264 gtk_list_store_append(store, &new_iter);
267 gtk_list_store_set(store, &new_iter,
268 DIR_COLUMN_POINTER, fd,
269 DIR_COLUMN_ICON, pixbuf,
270 DIR_COLUMN_NAME, fd->name,
271 DIR_COLUMN_LINK, link,
272 DIR_COLUMN_DATE, date,
279 valid = gtk_list_store_remove(store, &iter);
283 gtk_list_store_set(store, &iter,
284 DIR_COLUMN_ICON, pixbuf,
285 DIR_COLUMN_NAME, fd->name,
286 DIR_COLUMN_LINK, link,
287 DIR_COLUMN_DATE, date,
290 if (valid) valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter);
301 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, DIR_COLUMN_POINTER, &old_fd, -1);
303 valid = gtk_list_store_remove(store, &iter);
310 filelist_free(old_list);
315 gboolean vdlist_set_fd(ViewDir *vd, FileData *dir_fd)
318 gchar *old_path = NULL; /* Used to store directory for walking up */
320 if (!dir_fd) return FALSE;
321 if (vd->dir_fd == dir_fd) return TRUE;
327 base = remove_level_from_path(vd->dir_fd->path);
328 if (strcmp(base, dir_fd->path) == 0)
330 old_path = g_strdup(filename_from_path(vd->dir_fd->path));
335 file_data_unref(vd->dir_fd);
336 vd->dir_fd = file_data_ref(dir_fd);
338 ret = vdlist_populate(vd, TRUE);
340 /* scroll to make last path visible */
341 FileData *found = NULL;
344 work = VDLIST(vd)->list;
345 while (work && !found)
347 FileData *fd = static_cast<FileData *>(work->data);
348 if (!old_path || strcmp(old_path, fd->name) == 0) found = fd;
352 if (found) vdlist_scroll_to_row(vd, found, 0.5);
354 if (old_path) g_free(old_path);
359 void vdlist_refresh(ViewDir *vd)
361 vdlist_populate(vd, FALSE);
364 gboolean vdlist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
366 ViewDir *vd = static_cast<ViewDir *>(data);
369 if (event->keyval != GDK_KEY_Menu) return FALSE;
371 gtk_tree_view_get_cursor(GTK_TREE_VIEW(vd->view), &tpath, NULL);
377 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
378 gtk_tree_model_get_iter(store, &iter, tpath);
379 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &vd->click_fd, -1);
381 gtk_tree_path_free(tpath);
388 vd_color_set(vd, vd->click_fd, TRUE);
390 vd->popup = vd_pop_menu(vd, vd->click_fd);
392 gtk_menu_popup_at_pointer(GTK_MENU(vd->popup), NULL);
397 gboolean vdlist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
399 ViewDir *vd = static_cast<ViewDir *>(data);
404 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
405 &tpath, NULL, NULL, NULL))
409 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
410 gtk_tree_model_get_iter(store, &iter, tpath);
411 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
412 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE);
413 gtk_tree_path_free(tpath);
418 if (options->view_dir_list_single_click_enter)
419 vd_color_set(vd, vd->click_fd, TRUE);
421 if (bevent->button == MOUSE_BUTTON_RIGHT)
423 vd->popup = vd_pop_menu(vd, vd->click_fd);
424 gtk_menu_popup_at_pointer(GTK_MENU(vd->popup), NULL);
428 return options->view_dir_list_single_click_enter;
431 void vdlist_destroy_cb(GtkWidget *UNUSED(widget), gpointer data)
433 ViewDir *vd = static_cast<ViewDir *>(data);
435 vd_dnd_drop_scroll_cancel(vd);
436 widget_auto_scroll_stop(vd->view);
438 filelist_free(VDLIST(vd)->list);
441 ViewDir *vdlist_new(ViewDir *vd, FileData *UNUSED(dir_fd))
444 GtkTreeSelection *selection;
445 GtkTreeViewColumn *column;
446 GtkCellRenderer *renderer;
448 vd->info = g_new0(ViewDirInfoList, 1);
450 vd->type = DIRVIEW_LIST;
452 store = gtk_list_store_new(6, G_TYPE_POINTER, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING);
453 vd->view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
454 g_object_unref(store);
456 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vd->view), FALSE);
457 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vd->view), FALSE);
459 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vd->view));
460 gtk_tree_selection_set_mode(selection, GTK_SELECTION_NONE);
462 column = gtk_tree_view_column_new();
463 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
465 renderer = gtk_cell_renderer_pixbuf_new();
466 gtk_tree_view_column_pack_start(column, renderer, FALSE);
467 gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", DIR_COLUMN_ICON);
468 gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL);
470 renderer = gtk_cell_renderer_text_new();
471 gtk_tree_view_column_pack_start(column, renderer, TRUE);
472 gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_NAME);
473 gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL);
475 renderer = gtk_cell_renderer_text_new();
476 gtk_tree_view_column_pack_start(column, renderer, TRUE);
477 gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_DATE);
478 gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL);
480 gtk_tree_view_append_column(GTK_TREE_VIEW(vd->view), column);
482 gtk_tree_view_set_tooltip_column(GTK_TREE_VIEW(vd->view), DIR_COLUMN_LINK);
486 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */