7 * This software is released under the GNU General Public License (GNU GPL).
8 * Please read the included file COPYING for more information.
9 * This software comes with no warranty of any kind, use at your own risk!
13 #include "view_dir_list.h"
19 #include "layout_image.h"
20 #include "layout_util.h"
22 #include "ui_bookmark.h"
23 #include "ui_fileops.h"
25 #include "ui_tree_edit.h"
28 #include <gdk/gdkkeysyms.h> /* for keyboard values */
33 #define VDLIST_INFO(_vd_, _part_) (((ViewDirInfoList *)(_vd_->info))->_part_)
36 static gint vdlist_auto_scroll_notify_cb(GtkWidget *widget, gint x, gint y, gpointer data);
39 *-----------------------------------------------------------------------------
41 *-----------------------------------------------------------------------------
44 gint vdlist_find_row(ViewDir *vd, FileData *fd, GtkTreeIter *iter)
50 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
51 valid = gtk_tree_model_get_iter_first(store, iter);
55 gtk_tree_model_get(GTK_TREE_MODEL(store), iter, DIR_COLUMN_POINTER, &fd_n, -1);
56 if (fd_n == fd) return row;
58 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter);
66 FileData *vdlist_row_by_path(ViewDir *vd, const gchar *path, gint *row)
78 work = VDLIST_INFO(vd, list);
81 FileData *fd = work->data;
82 if (strcmp(fd->path, path) == 0)
96 *-----------------------------------------------------------------------------
98 *-----------------------------------------------------------------------------
101 static GtkTargetEntry vdlist_dnd_drop_types[] = {
102 { "text/uri-list", 0, TARGET_URI_LIST }
104 static gint vdlist_dnd_drop_types_count = 1;
106 static void vdlist_dest_set(ViewDir *vd, gint enable)
110 gtk_drag_dest_set(vd->view,
111 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_DROP,
112 vdlist_dnd_drop_types, vdlist_dnd_drop_types_count,
113 GDK_ACTION_MOVE | GDK_ACTION_COPY);
117 gtk_drag_dest_unset(vd->view);
121 static void vdlist_dnd_get(GtkWidget *widget, GdkDragContext *context,
122 GtkSelectionData *selection_data, guint info,
123 guint time, gpointer data)
130 if (!vd->click_fd) return;
134 case TARGET_URI_LIST:
135 case TARGET_TEXT_PLAIN:
136 list = g_list_prepend(NULL, vd->click_fd);
137 text = uri_text_from_filelist(list, &length, (info == TARGET_TEXT_PLAIN));
143 gtk_selection_data_set (selection_data, selection_data->target,
144 8, (guchar *)text, length);
149 static void vdlist_dnd_begin(GtkWidget *widget, GdkDragContext *context, gpointer data)
153 vd_color_set(vd, vd->click_fd, TRUE);
154 vdlist_dest_set(vd, FALSE);
157 static void vdlist_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
161 vd_color_set(vd, vd->click_fd, FALSE);
163 if (context->action == GDK_ACTION_MOVE)
167 vdlist_dest_set(vd, TRUE);
170 static void vdlist_dnd_drop_receive(GtkWidget *widget,
171 GdkDragContext *context, gint x, gint y,
172 GtkSelectionData *selection_data, guint info,
173 guint time, gpointer data)
182 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), x, y,
183 &tpath, NULL, NULL, NULL))
187 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
188 gtk_tree_model_get_iter(store, &iter, tpath);
189 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
190 gtk_tree_path_free(tpath);
195 if (info == TARGET_URI_LIST)
200 list = uri_filelist_from_text((gchar *)selection_data->data, TRUE);
203 active = access_file(fd->path, W_OK | X_OK);
205 vd_color_set(vd, fd, TRUE);
206 vd->popup = vd_drop_menu(vd, active);
207 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, NULL, NULL, 0, time);
210 vd->drop_list = list;
215 static gint vdlist_get_row_visibility(ViewDir *vd, FileData *fd)
218 GtkTreeViewColumn *column;
225 if (!fd || vd_find_row(vd, fd, &iter) < 0) return 0;
227 column = gtk_tree_view_get_column(GTK_TREE_VIEW(vd->view), 0);
228 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
229 tpath = gtk_tree_model_get_path(store, &iter);
231 gtk_tree_view_get_visible_rect(GTK_TREE_VIEW(vd->view), &vrect);
232 gtk_tree_view_get_cell_area(GTK_TREE_VIEW(vd->view), tpath, column, &crect);
233 printf("window: %d + %d; cell: %d + %d\n", vrect.y, vrect.height, crect.y, crect.height);
234 gtk_tree_path_free(tpath);
236 if (crect.y + crect.height < vrect.y) return -1;
237 if (crect.y > vrect.y + vrect.height) return 1;
242 static void vdlist_scroll_to_row(ViewDir *vd, FileData *fd, gfloat y_align)
246 if (GTK_WIDGET_REALIZED(vd->view) &&
247 vd_find_row(vd, fd, &iter) >= 0)
252 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
253 tpath = gtk_tree_model_get_path(store, &iter);
254 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vd->view), tpath, NULL, TRUE, y_align, 0.0);
255 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vd->view), tpath, NULL, FALSE);
256 gtk_tree_path_free(tpath);
258 if (!GTK_WIDGET_HAS_FOCUS(vd->view)) gtk_widget_grab_focus(vd->view);
262 static void vdlist_drop_update(ViewDir *vd, gint x, gint y)
268 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(vd->view), x, y,
269 &tpath, NULL, NULL, NULL))
273 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
274 gtk_tree_model_get_iter(store, &iter, tpath);
275 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
276 gtk_tree_path_free(tpath);
279 if (fd != vd->drop_fd)
281 vd_color_set(vd, vd->drop_fd, FALSE);
282 vd_color_set(vd, fd, TRUE);
288 static void vdlist_dnd_drop_scroll_cancel(ViewDir *vd)
290 if (vd->drop_scroll_id != -1) g_source_remove(vd->drop_scroll_id);
291 vd->drop_scroll_id = -1;
294 static gint vdlist_auto_scroll_idle_cb(gpointer data)
304 window = vd->view->window;
305 gdk_window_get_pointer(window, &x, &y, NULL);
306 gdk_drawable_get_size(window, &w, &h);
307 if (x >= 0 && x < w && y >= 0 && y < h)
309 vdlist_drop_update(vd, x, y);
313 vd->drop_scroll_id = -1;
317 static gint vdlist_auto_scroll_notify_cb(GtkWidget *widget, gint x, gint y, gpointer data)
321 if (!vd->drop_fd || vd->drop_list) return FALSE;
323 if (vd->drop_scroll_id == -1) vd->drop_scroll_id = g_idle_add(vdlist_auto_scroll_idle_cb, vd);
328 static gint vdlist_dnd_drop_motion(GtkWidget *widget, GdkDragContext *context,
329 gint x, gint y, guint time, gpointer data)
335 if (gtk_drag_get_source_widget(context) == vd->view)
337 /* from same window */
338 gdk_drag_status(context, 0, time);
343 gdk_drag_status(context, context->suggested_action, time);
346 vdlist_drop_update(vd, x, y);
350 GtkAdjustment *adj = gtk_tree_view_get_vadjustment(GTK_TREE_VIEW(vd->view));
351 widget_auto_scroll_start(vd->view, adj, -1, -1, vdlist_auto_scroll_notify_cb, vd);
357 static void vdlist_dnd_drop_leave(GtkWidget *widget, GdkDragContext *context, guint time, gpointer data)
361 if (vd->drop_fd != vd->click_fd) vd_color_set(vd, vd->drop_fd, FALSE);
366 static void vdlist_dnd_init(ViewDir *vd)
368 gtk_drag_source_set(vd->view, GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
369 dnd_file_drag_types, dnd_file_drag_types_count,
370 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
371 g_signal_connect(G_OBJECT(vd->view), "drag_data_get",
372 G_CALLBACK(vdlist_dnd_get), vd);
373 g_signal_connect(G_OBJECT(vd->view), "drag_begin",
374 G_CALLBACK(vdlist_dnd_begin), vd);
375 g_signal_connect(G_OBJECT(vd->view), "drag_end",
376 G_CALLBACK(vdlist_dnd_end), vd);
378 vdlist_dest_set(vd, TRUE);
379 g_signal_connect(G_OBJECT(vd->view), "drag_data_received",
380 G_CALLBACK(vdlist_dnd_drop_receive), vd);
381 g_signal_connect(G_OBJECT(vd->view), "drag_motion",
382 G_CALLBACK(vdlist_dnd_drop_motion), vd);
383 g_signal_connect(G_OBJECT(vd->view), "drag_leave",
384 G_CALLBACK(vdlist_dnd_drop_leave), vd);
388 *-----------------------------------------------------------------------------
390 *-----------------------------------------------------------------------------
393 static void vdlist_select_row(ViewDir *vd, FileData *fd)
395 if (fd && vd->select_func)
399 path = g_strdup(fd->path);
400 vd->select_func(vd, path, vd->select_data);
405 const gchar *vdlist_row_get_path(ViewDir *vd, gint row)
409 fd = g_list_nth_data(VDLIST_INFO(vd, list), row);
411 if (fd) return fd->path;
416 static void vdlist_populate(ViewDir *vd)
421 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view)));
422 gtk_list_store_clear(store);
424 work = VDLIST_INFO(vd, list);
433 if (access_file(fd->path, R_OK | X_OK) && fd->name)
435 if (fd->name[0] == '.' && fd->name[1] == '\0')
437 pixbuf = vd->pf->open;
439 else if (fd->name[0] == '.' && fd->name[1] == '.' && fd->name[2] == '\0')
441 pixbuf = vd->pf->parent;
445 pixbuf = vd->pf->close;
450 pixbuf = vd->pf->deny;
453 gtk_list_store_append(store, &iter);
454 gtk_list_store_set(store, &iter,
455 DIR_COLUMN_POINTER, fd,
456 DIR_COLUMN_ICON, pixbuf,
457 DIR_COLUMN_NAME, fd->name, -1);
466 gint vdlist_set_path(ViewDir *vd, const gchar *path)
470 gchar *old_path = NULL;
473 if (!path) return FALSE;
474 if (vd->path && strcmp(path, vd->path) == 0) return TRUE;
480 base = remove_level_from_path(vd->path);
481 if (strcmp(base, path) == 0)
483 old_path = g_strdup(filename_from_path(vd->path));
489 vd->path = g_strdup(path);
491 filelist_free(VDLIST_INFO(vd, list));
492 VDLIST_INFO(vd, list) = NULL;
494 ret = filelist_read(vd->path, NULL, &VDLIST_INFO(vd, list));
496 VDLIST_INFO(vd, list) = filelist_sort(VDLIST_INFO(vd, list), SORT_NAME, TRUE);
500 if (strcmp(vd->path, "/") != 0)
502 filepath = g_strconcat(vd->path, "/", "..", NULL);
503 fd = file_data_new_simple(filepath);
504 VDLIST_INFO(vd, list) = g_list_prepend(VDLIST_INFO(vd, list), fd);
508 if (options->file_filter.show_dot_directory)
510 filepath = g_strconcat(vd->path, "/", ".", NULL);
511 fd = file_data_new_simple(filepath);
512 VDLIST_INFO(vd, list) = g_list_prepend(VDLIST_INFO(vd, list), fd);
520 /* scroll to make last path visible */
521 FileData *found = NULL;
524 work = VDLIST_INFO(vd, list);
525 while (work && !found)
527 FileData *fd = work->data;
528 if (strcmp(old_path, fd->name) == 0) found = fd;
532 if (found) vdlist_scroll_to_row(vd, found, 0.5);
538 if (GTK_WIDGET_REALIZED(vd->view))
540 gtk_tree_view_scroll_to_point(GTK_TREE_VIEW(vd->view), 0, 0);
546 void vdlist_refresh(ViewDir *vd)
550 path = g_strdup(vd->path);
552 vdlist_set_path(vd, path);
556 static void vdlist_menu_position_cb(GtkMenu *menu, gint *x, gint *y, gboolean *push_in, gpointer data)
564 if (vd_find_row(vd, vd->click_fd, &iter) < 0) return;
565 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
566 tpath = gtk_tree_model_get_path(store, &iter);
567 tree_view_get_cell_clamped(GTK_TREE_VIEW(vd->view), tpath, 0, TRUE, x, y, &cw, &ch);
568 gtk_tree_path_free(tpath);
570 popup_menu_position_clamp(menu, x, y, 0);
573 static gint vdlist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
578 if (event->keyval != GDK_Menu) return FALSE;
580 gtk_tree_view_get_cursor(GTK_TREE_VIEW(vd->view), &tpath, NULL);
586 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
587 gtk_tree_model_get_iter(store, &iter, tpath);
588 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &vd->click_fd, -1);
590 gtk_tree_path_free(tpath);
597 vd_color_set(vd, vd->click_fd, TRUE);
599 vd->popup = vd_pop_menu(vd, vd->click_fd);
601 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, vdlist_menu_position_cb, vd, 0, GDK_CURRENT_TIME);
606 static gint vdlist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
613 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
614 &tpath, NULL, NULL, NULL))
618 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
619 gtk_tree_model_get_iter(store, &iter, tpath);
620 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
621 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE);
622 gtk_tree_path_free(tpath);
626 vd_color_set(vd, vd->click_fd, TRUE);
628 if (bevent->button == 3)
630 vd->popup = vd_pop_menu(vd, vd->click_fd);
631 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, NULL, NULL,
632 bevent->button, bevent->time);
638 static gint vdlist_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
645 vd_color_set(vd, vd->click_fd, FALSE);
647 if (bevent->button != 1) return TRUE;
649 if ((bevent->x != 0 || bevent->y != 0) &&
650 gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
651 &tpath, NULL, NULL, NULL))
655 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
656 gtk_tree_model_get_iter(store, &iter, tpath);
657 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
658 gtk_tree_path_free(tpath);
661 if (fd && vd->click_fd == fd)
663 vdlist_select_row(vd, vd->click_fd);
669 static void vdlist_select_cb(GtkTreeView *tview, GtkTreePath *tpath, GtkTreeViewColumn *column, gpointer data)
676 store = gtk_tree_view_get_model(tview);
677 gtk_tree_model_get_iter(store, &iter, tpath);
678 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
680 vdlist_select_row(vd, fd);
683 static GdkColor *vdlist_color_shifted(GtkWidget *widget)
685 static GdkColor color;
686 static GtkWidget *done = NULL;
692 style = gtk_widget_get_style(widget);
693 memcpy(&color, &style->base[GTK_STATE_NORMAL], sizeof(color));
694 shift_color(&color, -1, 0);
701 static void vdlist_color_cb(GtkTreeViewColumn *tree_column, GtkCellRenderer *cell,
702 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer data)
707 gtk_tree_model_get(tree_model, iter, DIR_COLUMN_COLOR, &set, -1);
708 g_object_set(G_OBJECT(cell),
709 "cell-background-gdk", vdlist_color_shifted(vd->view),
710 "cell-background-set", set, NULL);
713 static void vdlist_destroy_cb(GtkWidget *widget, gpointer data)
717 vdlist_dnd_drop_scroll_cancel(vd);
718 widget_auto_scroll_stop(vd->view);
720 filelist_free(VDLIST_INFO(vd, list));
723 ViewDir *vdlist_new(ViewDir *vd, const gchar *path)
726 GtkTreeSelection *selection;
727 GtkTreeViewColumn *column;
728 GtkCellRenderer *renderer;
730 vd->info = g_new0(ViewDirInfoList, 1);
731 vd->type = DIRVIEW_LIST;
732 vd->widget_destroy_cb = vdlist_destroy_cb;
734 VDLIST_INFO(vd, list) = NULL;
736 store = gtk_list_store_new(4, G_TYPE_POINTER, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_BOOLEAN);
737 vd->view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
738 g_object_unref(store);
740 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vd->view), FALSE);
741 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vd->view), FALSE);
742 g_signal_connect(G_OBJECT(vd->view), "row_activated",
744 G_CALLBACK(vdlist_select_cb), vd);
746 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vd->view));
747 gtk_tree_selection_set_mode(selection, GTK_SELECTION_NONE);
749 column = gtk_tree_view_column_new();
750 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
752 renderer = gtk_cell_renderer_pixbuf_new();
753 gtk_tree_view_column_pack_start(column, renderer, FALSE);
754 gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", DIR_COLUMN_ICON);
755 gtk_tree_view_column_set_cell_data_func(column, renderer, vdlist_color_cb, vd, NULL);
757 renderer = gtk_cell_renderer_text_new();
758 gtk_tree_view_column_pack_start(column, renderer, TRUE);
759 gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_NAME);
760 gtk_tree_view_column_set_cell_data_func(column, renderer, vdlist_color_cb, vd, NULL);
762 gtk_tree_view_append_column(GTK_TREE_VIEW(vd->view), column);
764 g_signal_connect(G_OBJECT(vd->view), "key_press_event",
765 G_CALLBACK(vdlist_press_key_cb), vd);
766 gtk_container_add(GTK_CONTAINER(vd->widget), vd->view);
767 gtk_widget_show(vd->view);
771 g_signal_connect(G_OBJECT(vd->view), "button_press_event",
772 G_CALLBACK(vdlist_press_cb), vd);
773 g_signal_connect(G_OBJECT(vd->view), "button_release_event",
774 G_CALLBACK(vdlist_release_cb), vd);
776 if (path) vdlist_set_path(vd, path);