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 void vdlist_popup_destroy_cb(GtkWidget *widget, gpointer data);
37 static gint vdlist_auto_scroll_notify_cb(GtkWidget *widget, gint x, gint y, gpointer data);
40 *-----------------------------------------------------------------------------
42 *-----------------------------------------------------------------------------
45 static gint vdlist_find_row(ViewDir *vd, FileData *fd, GtkTreeIter *iter)
51 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
52 valid = gtk_tree_model_get_iter_first(store, iter);
56 gtk_tree_model_get(GTK_TREE_MODEL(store), iter, DIR_COLUMN_POINTER, &fd_n, -1);
57 if (fd_n == fd) return row;
59 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter);
66 static gint vdlist_rename_row_cb(TreeEditData *td, const gchar *old, const gchar *new, gpointer data)
76 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
77 if (!gtk_tree_model_get_iter(store, &iter, td->path)) return FALSE;
78 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
79 if (!fd) return FALSE;
81 old_path = g_strdup(fd->path);
83 base = remove_level_from_path(old_path);
84 new_path = concat_dir_and_file(base, new);
87 if (file_util_rename_dir(fd, new_path, vd->view))
89 if (vd->layout && strcmp(vd->path, old_path) == 0)
91 layout_set_path(vd->layout, new_path);
104 static void vdlist_rename_by_row(ViewDir *vd, FileData *fd)
110 if (vdlist_find_row(vd, fd, &iter) < 0) return;
111 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
112 tpath = gtk_tree_model_get_path(store, &iter);
114 tree_edit_by_path(GTK_TREE_VIEW(vd->view), tpath, 0, fd->name,
115 vdlist_rename_row_cb, vd);
116 gtk_tree_path_free(tpath);
119 static FileData *vdlist_row_by_path(ViewDir *vd, const gchar *path, gint *row)
131 work = VDLIST_INFO(vd, list);
134 FileData *fd = work->data;
135 if (strcmp(fd->path, path) == 0)
148 static void vdlist_color_set(ViewDir *vd, FileData *fd, gint color_set)
153 if (vdlist_find_row(vd, fd, &iter) < 0) return;
154 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
155 gtk_list_store_set(GTK_LIST_STORE(store), &iter, DIR_COLUMN_COLOR, color_set, -1);
159 *-----------------------------------------------------------------------------
160 * drop menu (from dnd)
161 *-----------------------------------------------------------------------------
164 static void vdlist_drop_menu_copy_cb(GtkWidget *widget, gpointer data)
170 if (!vd->drop_fd) return;
172 path = vd->drop_fd->path;
173 list = vd->drop_list;
174 vd->drop_list = NULL;
176 file_util_copy_simple(list, path);
179 static void vdlist_drop_menu_move_cb(GtkWidget *widget, gpointer data)
185 if (!vd->drop_fd) return;
187 path = vd->drop_fd->path;
188 list = vd->drop_list;
190 vd->drop_list = NULL;
192 file_util_move_simple(list, path);
195 static GtkWidget *vdlist_drop_menu(ViewDir *vd, gint active)
199 menu = popup_menu_short_lived();
200 g_signal_connect(G_OBJECT(menu), "destroy",
201 G_CALLBACK(vdlist_popup_destroy_cb), vd);
203 menu_item_add_stock_sensitive(menu, _("_Copy"), GTK_STOCK_COPY, active,
204 G_CALLBACK(vdlist_drop_menu_copy_cb), vd);
205 menu_item_add_sensitive(menu, _("_Move"), active, G_CALLBACK(vdlist_drop_menu_move_cb), vd);
207 menu_item_add_divider(menu);
208 menu_item_add_stock(menu, _("Cancel"), GTK_STOCK_CANCEL, NULL, vd);
214 *-----------------------------------------------------------------------------
216 *-----------------------------------------------------------------------------
219 static void vdlist_pop_menu_up_cb(GtkWidget *widget, gpointer data)
224 if (!vd->path || strcmp(vd->path, "/") == 0) return;
225 path = remove_level_from_path(vd->path);
229 vd->select_func(vd, path, vd->select_data);
235 static void vdlist_pop_menu_slide_cb(GtkWidget *widget, gpointer data)
240 if (!vd->layout || !vd->click_fd) return;
242 path = g_strdup(vd->click_fd->path);
244 layout_set_path(vd->layout, path);
245 layout_select_none(vd->layout);
246 layout_image_slideshow_stop(vd->layout);
247 layout_image_slideshow_start(vd->layout);
252 static void vdlist_pop_menu_slide_rec_cb(GtkWidget *widget, gpointer data)
258 if (!vd->layout || !vd->click_fd) return;
260 path = g_strdup(vd->click_fd->path);
262 list = filelist_recursive(path);
264 layout_image_slideshow_stop(vd->layout);
265 layout_image_slideshow_start_from_list(vd->layout, list);
270 static void vdlist_pop_menu_dupe(ViewDir *vd, gint recursive)
275 if (!vd->click_fd) return;
279 list = g_list_append(list, file_data_ref(vd->click_fd));
283 filelist_read(vd->click_fd->path, &list, NULL);
284 list = filelist_filter(list, FALSE);
287 dw = dupe_window_new(DUPE_MATCH_NAME);
288 dupe_window_add_files(dw, list, recursive);
293 static void vdlist_pop_menu_dupe_cb(GtkWidget *widget, gpointer data)
296 vdlist_pop_menu_dupe(vd, FALSE);
299 static void vdlist_pop_menu_dupe_rec_cb(GtkWidget *widget, gpointer data)
302 vdlist_pop_menu_dupe(vd, TRUE);
305 static void vdlist_pop_menu_new_cb(GtkWidget *widget, gpointer data)
311 if (!vd->path) return;
313 buf = concat_dir_and_file(vd->path, _("new_folder"));
314 new_path = unique_filename(buf, NULL, NULL, FALSE);
316 if (!new_path) return;
318 if (!mkdir_utf8(new_path, 0755))
322 text = g_strdup_printf(_("Unable to create folder:\n%s"), new_path);
323 file_util_warning_dialog(_("Error creating folder"), text, GTK_STOCK_DIALOG_ERROR, vd->view);
331 fd = vdlist_row_by_path(vd, new_path, NULL);
333 vdlist_rename_by_row(vd, fd);
339 static void vdlist_pop_menu_rename_cb(GtkWidget *widget, gpointer data)
343 vdlist_rename_by_row(vd, vd->click_fd);
346 static void vdlist_pop_menu_delete_cb(GtkWidget *widget, gpointer data)
350 if (!vd->click_fd) return;
351 file_util_delete_dir(vd->click_fd, vd->widget);
354 static void vdlist_pop_menu_dir_view_as_cb(GtkWidget *widget, gpointer data)
358 if (vd->layout) layout_views_set(vd->layout, DIRVIEW_TREE, vd->layout->icon_view);
361 static void vdlist_pop_menu_refresh_cb(GtkWidget *widget, gpointer data)
365 if (vd->layout) layout_refresh(vd->layout);
368 static void vdlist_toggle_show_hidden_files_cb(GtkWidget *widget, gpointer data)
372 options->file_filter.show_hidden_files = !options->file_filter.show_hidden_files;
373 if (vd->layout) layout_refresh(vd->layout);
376 static GtkWidget *vdlist_pop_menu(ViewDir *vd, FileData *fd)
381 active = (fd != NULL);
383 menu = popup_menu_short_lived();
384 g_signal_connect(G_OBJECT(menu), "destroy",
385 G_CALLBACK(vdlist_popup_destroy_cb), vd);
387 menu_item_add_stock_sensitive(menu, _("_Up to parent"), GTK_STOCK_GO_UP,
388 (vd->path && strcmp(vd->path, "/") != 0),
389 G_CALLBACK(vdlist_pop_menu_up_cb), vd);
391 menu_item_add_divider(menu);
392 menu_item_add_sensitive(menu, _("_Slideshow"), active,
393 G_CALLBACK(vdlist_pop_menu_slide_cb), vd);
394 menu_item_add_sensitive(menu, _("Slideshow recursive"), active,
395 G_CALLBACK(vdlist_pop_menu_slide_rec_cb), vd);
397 menu_item_add_divider(menu);
398 menu_item_add_stock_sensitive(menu, _("Find _duplicates..."), GTK_STOCK_FIND, active,
399 G_CALLBACK(vdlist_pop_menu_dupe_cb), vd);
400 menu_item_add_stock_sensitive(menu, _("Find duplicates recursive..."), GTK_STOCK_FIND, active,
401 G_CALLBACK(vdlist_pop_menu_dupe_rec_cb), vd);
403 menu_item_add_divider(menu);
405 /* check using . (always row 0) */
406 active = (vd->path && access_file(vd->path , W_OK | X_OK));
407 menu_item_add_sensitive(menu, _("_New folder..."), active,
408 G_CALLBACK(vdlist_pop_menu_new_cb), vd);
410 /* ignore .. and . */
411 active = (active && fd &&
412 strcmp(fd->name, ".") != 0 &&
413 strcmp(fd->name, "..") != 0 &&
414 access_file(fd->path, W_OK | X_OK));
415 menu_item_add_sensitive(menu, _("_Rename..."), active,
416 G_CALLBACK(vdlist_pop_menu_rename_cb), vd);
417 menu_item_add_stock_sensitive(menu, _("_Delete..."), GTK_STOCK_DELETE, active,
418 G_CALLBACK(vdlist_pop_menu_delete_cb), vd);
420 menu_item_add_divider(menu);
421 menu_item_add_check(menu, _("View as _tree"), FALSE,
422 G_CALLBACK(vdlist_pop_menu_dir_view_as_cb), vd);
423 menu_item_add_check(menu, _("Show _hidden files"), options->file_filter.show_hidden_files,
424 G_CALLBACK(vdlist_toggle_show_hidden_files_cb), vd);
426 menu_item_add_stock(menu, _("Re_fresh"), GTK_STOCK_REFRESH,
427 G_CALLBACK(vdlist_pop_menu_refresh_cb), vd);
432 static void vdlist_popup_destroy_cb(GtkWidget *widget, gpointer data)
436 vdlist_color_set(vd, vd->click_fd, FALSE);
440 vdlist_color_set(vd, vd->drop_fd, FALSE);
441 filelist_free(vd->drop_list);
442 vd->drop_list = NULL;
447 *-----------------------------------------------------------------------------
449 *-----------------------------------------------------------------------------
452 static GtkTargetEntry vdlist_dnd_drop_types[] = {
453 { "text/uri-list", 0, TARGET_URI_LIST }
455 static gint vdlist_dnd_drop_types_count = 1;
457 static void vdlist_dest_set(ViewDir *vd, gint enable)
461 gtk_drag_dest_set(vd->view,
462 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_DROP,
463 vdlist_dnd_drop_types, vdlist_dnd_drop_types_count,
464 GDK_ACTION_MOVE | GDK_ACTION_COPY);
468 gtk_drag_dest_unset(vd->view);
472 static void vdlist_dnd_get(GtkWidget *widget, GdkDragContext *context,
473 GtkSelectionData *selection_data, guint info,
474 guint time, gpointer data)
481 if (!vd->click_fd) return;
485 case TARGET_URI_LIST:
486 case TARGET_TEXT_PLAIN:
487 list = g_list_prepend(NULL, vd->click_fd);
488 text = uri_text_from_filelist(list, &length, (info == TARGET_TEXT_PLAIN));
494 gtk_selection_data_set (selection_data, selection_data->target,
495 8, (guchar *)text, length);
500 static void vdlist_dnd_begin(GtkWidget *widget, GdkDragContext *context, gpointer data)
504 vdlist_color_set(vd, vd->click_fd, TRUE);
505 vdlist_dest_set(vd, FALSE);
508 static void vdlist_dnd_end(GtkWidget *widget, GdkDragContext *context, gpointer data)
512 vdlist_color_set(vd, vd->click_fd, FALSE);
514 if (context->action == GDK_ACTION_MOVE)
518 vdlist_dest_set(vd, TRUE);
521 static void vdlist_dnd_drop_receive(GtkWidget *widget,
522 GdkDragContext *context, gint x, gint y,
523 GtkSelectionData *selection_data, guint info,
524 guint time, gpointer data)
533 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), x, y,
534 &tpath, NULL, NULL, NULL))
538 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
539 gtk_tree_model_get_iter(store, &iter, tpath);
540 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
541 gtk_tree_path_free(tpath);
546 if (info == TARGET_URI_LIST)
551 list = uri_filelist_from_text((gchar *)selection_data->data, TRUE);
554 active = access_file(fd->path, W_OK | X_OK);
556 vdlist_color_set(vd, fd, TRUE);
557 vd->popup = vdlist_drop_menu(vd, active);
558 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, NULL, NULL, 0, time);
561 vd->drop_list = list;
566 static gint vdlist_get_row_visibility(ViewDir *vd, FileData *fd)
569 GtkTreeViewColumn *column;
576 if (!fd || vdlist_find_row(vd, fd, &iter) < 0) return 0;
578 column = gtk_tree_view_get_column(GTK_TREE_VIEW(vd->view), 0);
579 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
580 tpath = gtk_tree_model_get_path(store, &iter);
582 gtk_tree_view_get_visible_rect(GTK_TREE_VIEW(vd->view), &vrect);
583 gtk_tree_view_get_cell_area(GTK_TREE_VIEW(vd->view), tpath, column, &crect);
584 printf("window: %d + %d; cell: %d + %d\n", vrect.y, vrect.height, crect.y, crect.height);
585 gtk_tree_path_free(tpath);
587 if (crect.y + crect.height < vrect.y) return -1;
588 if (crect.y > vrect.y + vrect.height) return 1;
593 static void vdlist_scroll_to_row(ViewDir *vd, FileData *fd, gfloat y_align)
597 if (GTK_WIDGET_REALIZED(vd->view) &&
598 vdlist_find_row(vd, fd, &iter) >= 0)
603 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
604 tpath = gtk_tree_model_get_path(store, &iter);
605 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vd->view), tpath, NULL, TRUE, y_align, 0.0);
606 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vd->view), tpath, NULL, FALSE);
607 gtk_tree_path_free(tpath);
609 if (!GTK_WIDGET_HAS_FOCUS(vd->view)) gtk_widget_grab_focus(vd->view);
613 static void vdlist_drop_update(ViewDir *vd, gint x, gint y)
619 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(vd->view), x, y,
620 &tpath, NULL, NULL, NULL))
624 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
625 gtk_tree_model_get_iter(store, &iter, tpath);
626 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
627 gtk_tree_path_free(tpath);
630 if (fd != vd->drop_fd)
632 vdlist_color_set(vd, vd->drop_fd, FALSE);
633 vdlist_color_set(vd, fd, TRUE);
639 static void vdlist_dnd_drop_scroll_cancel(ViewDir *vd)
641 if (vd->drop_scroll_id != -1) g_source_remove(vd->drop_scroll_id);
642 vd->drop_scroll_id = -1;
645 static gint vdlist_auto_scroll_idle_cb(gpointer data)
655 window = vd->view->window;
656 gdk_window_get_pointer(window, &x, &y, NULL);
657 gdk_drawable_get_size(window, &w, &h);
658 if (x >= 0 && x < w && y >= 0 && y < h)
660 vdlist_drop_update(vd, x, y);
664 vd->drop_scroll_id = -1;
668 static gint vdlist_auto_scroll_notify_cb(GtkWidget *widget, gint x, gint y, gpointer data)
672 if (!vd->drop_fd || vd->drop_list) return FALSE;
674 if (vd->drop_scroll_id == -1) vd->drop_scroll_id = g_idle_add(vdlist_auto_scroll_idle_cb, vd);
679 static gint vdlist_dnd_drop_motion(GtkWidget *widget, GdkDragContext *context,
680 gint x, gint y, guint time, gpointer data)
686 if (gtk_drag_get_source_widget(context) == vd->view)
688 /* from same window */
689 gdk_drag_status(context, 0, time);
694 gdk_drag_status(context, context->suggested_action, time);
697 vdlist_drop_update(vd, x, y);
701 GtkAdjustment *adj = gtk_tree_view_get_vadjustment(GTK_TREE_VIEW(vd->view));
702 widget_auto_scroll_start(vd->view, adj, -1, -1, vdlist_auto_scroll_notify_cb, vd);
708 static void vdlist_dnd_drop_leave(GtkWidget *widget, GdkDragContext *context, guint time, gpointer data)
712 if (vd->drop_fd != vd->click_fd) vdlist_color_set(vd, vd->drop_fd, FALSE);
717 static void vdlist_dnd_init(ViewDir *vd)
719 gtk_drag_source_set(vd->view, GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
720 dnd_file_drag_types, dnd_file_drag_types_count,
721 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK);
722 g_signal_connect(G_OBJECT(vd->view), "drag_data_get",
723 G_CALLBACK(vdlist_dnd_get), vd);
724 g_signal_connect(G_OBJECT(vd->view), "drag_begin",
725 G_CALLBACK(vdlist_dnd_begin), vd);
726 g_signal_connect(G_OBJECT(vd->view), "drag_end",
727 G_CALLBACK(vdlist_dnd_end), vd);
729 vdlist_dest_set(vd, TRUE);
730 g_signal_connect(G_OBJECT(vd->view), "drag_data_received",
731 G_CALLBACK(vdlist_dnd_drop_receive), vd);
732 g_signal_connect(G_OBJECT(vd->view), "drag_motion",
733 G_CALLBACK(vdlist_dnd_drop_motion), vd);
734 g_signal_connect(G_OBJECT(vd->view), "drag_leave",
735 G_CALLBACK(vdlist_dnd_drop_leave), vd);
739 *-----------------------------------------------------------------------------
741 *-----------------------------------------------------------------------------
744 static void vdlist_select_row(ViewDir *vd, FileData *fd)
746 if (fd && vd->select_func)
750 path = g_strdup(fd->path);
751 vd->select_func(vd, path, vd->select_data);
756 const gchar *vdlist_row_get_path(ViewDir *vd, gint row)
760 fd = g_list_nth_data(VDLIST_INFO(vd, list), row);
762 if (fd) return fd->path;
767 static void vdlist_populate(ViewDir *vd)
772 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view)));
773 gtk_list_store_clear(store);
775 work = VDLIST_INFO(vd, list);
784 if (access_file(fd->path, R_OK | X_OK) && fd->name)
786 if (fd->name[0] == '.' && fd->name[1] == '\0')
788 pixbuf = vd->pf->open;
790 else if (fd->name[0] == '.' && fd->name[1] == '.' && fd->name[2] == '\0')
792 pixbuf = vd->pf->parent;
796 pixbuf = vd->pf->close;
801 pixbuf = vd->pf->deny;
804 gtk_list_store_append(store, &iter);
805 gtk_list_store_set(store, &iter,
806 DIR_COLUMN_POINTER, fd,
807 DIR_COLUMN_ICON, pixbuf,
808 DIR_COLUMN_NAME, fd->name, -1);
817 gint vdlist_set_path(ViewDir *vd, const gchar *path)
821 gchar *old_path = NULL;
824 if (!path) return FALSE;
825 if (vd->path && strcmp(path, vd->path) == 0) return TRUE;
831 base = remove_level_from_path(vd->path);
832 if (strcmp(base, path) == 0)
834 old_path = g_strdup(filename_from_path(vd->path));
840 vd->path = g_strdup(path);
842 filelist_free(VDLIST_INFO(vd, list));
843 VDLIST_INFO(vd, list) = NULL;
845 ret = filelist_read(vd->path, NULL, &VDLIST_INFO(vd, list));
847 VDLIST_INFO(vd, list) = filelist_sort(VDLIST_INFO(vd, list), SORT_NAME, TRUE);
851 if (strcmp(vd->path, "/") != 0)
853 filepath = g_strconcat(vd->path, "/", "..", NULL);
854 fd = file_data_new_simple(filepath);
855 VDLIST_INFO(vd, list) = g_list_prepend(VDLIST_INFO(vd, list), fd);
859 if (options->file_filter.show_dot_directory)
861 filepath = g_strconcat(vd->path, "/", ".", NULL);
862 fd = file_data_new_simple(filepath);
863 VDLIST_INFO(vd, list) = g_list_prepend(VDLIST_INFO(vd, list), fd);
871 /* scroll to make last path visible */
872 FileData *found = NULL;
875 work = VDLIST_INFO(vd, list);
876 while (work && !found)
878 FileData *fd = work->data;
879 if (strcmp(old_path, fd->name) == 0) found = fd;
883 if (found) vdlist_scroll_to_row(vd, found, 0.5);
889 if (GTK_WIDGET_REALIZED(vd->view))
891 gtk_tree_view_scroll_to_point(GTK_TREE_VIEW(vd->view), 0, 0);
897 void vdlist_refresh(ViewDir *vd)
901 path = g_strdup(vd->path);
903 vdlist_set_path(vd, path);
907 static void vdlist_menu_position_cb(GtkMenu *menu, gint *x, gint *y, gboolean *push_in, gpointer data)
915 if (vdlist_find_row(vd, vd->click_fd, &iter) < 0) return;
916 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
917 tpath = gtk_tree_model_get_path(store, &iter);
918 tree_view_get_cell_clamped(GTK_TREE_VIEW(vd->view), tpath, 0, TRUE, x, y, &cw, &ch);
919 gtk_tree_path_free(tpath);
921 popup_menu_position_clamp(menu, x, y, 0);
924 static gint vdlist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
929 if (event->keyval != GDK_Menu) return FALSE;
931 gtk_tree_view_get_cursor(GTK_TREE_VIEW(vd->view), &tpath, NULL);
937 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
938 gtk_tree_model_get_iter(store, &iter, tpath);
939 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &vd->click_fd, -1);
941 gtk_tree_path_free(tpath);
948 vdlist_color_set(vd, vd->click_fd, TRUE);
950 vd->popup = vdlist_pop_menu(vd, vd->click_fd);
952 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, vdlist_menu_position_cb, vd, 0, GDK_CURRENT_TIME);
957 static gint vdlist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
964 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
965 &tpath, NULL, NULL, NULL))
969 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
970 gtk_tree_model_get_iter(store, &iter, tpath);
971 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
972 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE);
973 gtk_tree_path_free(tpath);
977 vdlist_color_set(vd, vd->click_fd, TRUE);
979 if (bevent->button == 3)
981 vd->popup = vdlist_pop_menu(vd, vd->click_fd);
982 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, NULL, NULL,
983 bevent->button, bevent->time);
989 static gint vdlist_release_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
996 vdlist_color_set(vd, vd->click_fd, FALSE);
998 if (bevent->button != 1) return TRUE;
1000 if ((bevent->x != 0 || bevent->y != 0) &&
1001 gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
1002 &tpath, NULL, NULL, NULL))
1004 GtkTreeModel *store;
1006 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
1007 gtk_tree_model_get_iter(store, &iter, tpath);
1008 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
1009 gtk_tree_path_free(tpath);
1012 if (fd && vd->click_fd == fd)
1014 vdlist_select_row(vd, vd->click_fd);
1020 static void vdlist_select_cb(GtkTreeView *tview, GtkTreePath *tpath, GtkTreeViewColumn *column, gpointer data)
1023 GtkTreeModel *store;
1027 store = gtk_tree_view_get_model(tview);
1028 gtk_tree_model_get_iter(store, &iter, tpath);
1029 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
1031 vdlist_select_row(vd, fd);
1034 static GdkColor *vdlist_color_shifted(GtkWidget *widget)
1036 static GdkColor color;
1037 static GtkWidget *done = NULL;
1043 style = gtk_widget_get_style(widget);
1044 memcpy(&color, &style->base[GTK_STATE_NORMAL], sizeof(color));
1045 shift_color(&color, -1, 0);
1052 static void vdlist_color_cb(GtkTreeViewColumn *tree_column, GtkCellRenderer *cell,
1053 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer data)
1058 gtk_tree_model_get(tree_model, iter, DIR_COLUMN_COLOR, &set, -1);
1059 g_object_set(G_OBJECT(cell),
1060 "cell-background-gdk", vdlist_color_shifted(vd->view),
1061 "cell-background-set", set, NULL);
1064 static void vdlist_destroy_cb(GtkWidget *widget, gpointer data)
1070 g_signal_handlers_disconnect_matched(G_OBJECT(vd->popup), G_SIGNAL_MATCH_DATA,
1072 gtk_widget_destroy(vd->popup);
1075 vdlist_dnd_drop_scroll_cancel(vd);
1076 widget_auto_scroll_stop(vd->view);
1078 filelist_free(vd->drop_list);
1080 folder_icons_free(vd->pf);
1083 filelist_free(VDLIST_INFO(vd, list));
1088 ViewDir *vdlist_new(const gchar *path)
1091 GtkListStore *store;
1092 GtkTreeSelection *selection;
1093 GtkTreeViewColumn *column;
1094 GtkCellRenderer *renderer;
1096 vd = g_new0(ViewDir, 1);
1097 vd->info = g_new0(ViewDirInfoList, 1);
1098 vd->type = DIRVIEW_LIST;
1101 VDLIST_INFO(vd, list) = NULL;
1102 vd->click_fd = NULL;
1105 vd->drop_list = NULL;
1107 vd->drop_scroll_id = -1;
1111 vd->widget = gtk_scrolled_window_new(NULL, NULL);
1112 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(vd->widget), GTK_SHADOW_IN);
1113 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(vd->widget),
1114 GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
1115 g_signal_connect(G_OBJECT(vd->widget), "destroy",
1116 G_CALLBACK(vdlist_destroy_cb), vd);
1118 store = gtk_list_store_new(4, G_TYPE_POINTER, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_BOOLEAN);
1119 vd->view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
1120 g_object_unref(store);
1122 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vd->view), FALSE);
1123 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vd->view), FALSE);
1124 g_signal_connect(G_OBJECT(vd->view), "row_activated",
1126 G_CALLBACK(vdlist_select_cb), vd);
1128 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vd->view));
1129 gtk_tree_selection_set_mode(selection, GTK_SELECTION_NONE);
1131 column = gtk_tree_view_column_new();
1132 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
1134 renderer = gtk_cell_renderer_pixbuf_new();
1135 gtk_tree_view_column_pack_start(column, renderer, FALSE);
1136 gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", DIR_COLUMN_ICON);
1137 gtk_tree_view_column_set_cell_data_func(column, renderer, vdlist_color_cb, vd, NULL);
1139 renderer = gtk_cell_renderer_text_new();
1140 gtk_tree_view_column_pack_start(column, renderer, TRUE);
1141 gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_NAME);
1142 gtk_tree_view_column_set_cell_data_func(column, renderer, vdlist_color_cb, vd, NULL);
1144 gtk_tree_view_append_column(GTK_TREE_VIEW(vd->view), column);
1146 g_signal_connect(G_OBJECT(vd->view), "key_press_event",
1147 G_CALLBACK(vdlist_press_key_cb), vd);
1148 gtk_container_add(GTK_CONTAINER(vd->widget), vd->view);
1149 gtk_widget_show(vd->view);
1151 vd->pf = folder_icons_new();
1153 vdlist_dnd_init(vd);
1155 g_signal_connect(G_OBJECT(vd->view), "button_press_event",
1156 G_CALLBACK(vdlist_press_cb), vd);
1157 g_signal_connect(G_OBJECT(vd->view), "button_release_event",
1158 G_CALLBACK(vdlist_release_cb), vd);
1160 if (path) vdlist_set_path(vd, path);