always check for sidecars by readdir
[geeqie.git] / src / view_dir_list.c
1 /*
2  * Geeqie
3  * (C) 2004 John Ellis
4  * Copyright (C) 2008 - 2010 The Geeqie Team
5  *
6  * Author: John Ellis
7  *
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!
11  */
12
13 #include "main.h"
14 #include "view_dir_list.h"
15
16 #include "dnd.h"
17 #include "dupe.h"
18 #include "filedata.h"
19 #include "layout.h"
20 #include "layout_image.h"
21 #include "layout_util.h"
22 #include "utilops.h"
23 #include "ui_fileops.h"
24 #include "ui_menu.h"
25 #include "ui_tree_edit.h"
26 #include "view_dir.h"
27
28 #include <gdk/gdkkeysyms.h> /* for keyboard values */
29
30
31 #define VDLIST(_vd_) ((ViewDirInfoList *)(_vd_->info))
32
33
34 /*
35  *-----------------------------------------------------------------------------
36  * misc
37  *-----------------------------------------------------------------------------
38  */
39
40 gboolean vdlist_find_row(ViewDir *vd, FileData *fd, GtkTreeIter *iter)
41 {
42         GtkTreeModel *store;
43         gboolean valid;
44
45         store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
46         valid = gtk_tree_model_get_iter_first(store, iter);
47         while (valid)
48                 {
49                 FileData *fd_n;
50                 gtk_tree_model_get(GTK_TREE_MODEL(store), iter, DIR_COLUMN_POINTER, &fd_n, -1);
51                 if (fd_n == fd) return TRUE;
52
53                 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter);
54                 }
55
56         return FALSE;
57 }
58
59
60 FileData *vdlist_row_by_path(ViewDir *vd, const gchar *path, gint *row)
61 {
62         GList *work;
63         gint n;
64
65         if (!path)
66                 {
67                 if (row) *row = -1;
68                 return NULL;
69                 }
70
71         n = 0;
72         work = VDLIST(vd)->list;
73         while (work)
74                 {
75                 FileData *fd = work->data;
76                 if (strcmp(fd->path, path) == 0)
77                         {
78                         if (row) *row = n;
79                         return fd;
80                         }
81                 work = work->next;
82                 n++;
83                 }
84
85         if (row) *row = -1;
86         return NULL;
87 }
88
89 /*
90  *-----------------------------------------------------------------------------
91  * dnd
92  *-----------------------------------------------------------------------------
93  */
94
95 static void vdlist_scroll_to_row(ViewDir *vd, FileData *fd, gfloat y_align)
96 {
97         GtkTreeIter iter;
98
99 #if GTK_CHECK_VERSION(2,20,0)
100         if (gtk_widget_get_realized(vd->view) && vd_find_row(vd, fd, &iter))
101 #else
102         if (GTK_WIDGET_REALIZED(vd->view) && vd_find_row(vd, fd, &iter))
103 #endif
104                 {
105                 GtkTreeModel *store;
106                 GtkTreePath *tpath;
107
108                 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view));
109                 tpath = gtk_tree_model_get_path(store, &iter);
110                 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vd->view), tpath, NULL, TRUE, y_align, 0.0);
111                 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vd->view), tpath, NULL, FALSE);
112                 gtk_tree_path_free(tpath);
113
114 #if GTK_CHECK_VERSION(2,20,0)
115                 if (!gtk_widget_has_focus(vd->view)) gtk_widget_grab_focus(vd->view);
116 #else
117                 if (!GTK_WIDGET_HAS_FOCUS(vd->view)) gtk_widget_grab_focus(vd->view);
118 #endif
119                 }
120 }
121
122 /*
123  *-----------------------------------------------------------------------------
124  * main
125  *-----------------------------------------------------------------------------
126  */
127
128 const gchar *vdlist_row_get_path(ViewDir *vd, gint row)
129 {
130         FileData *fd;
131
132         fd = g_list_nth_data(VDLIST(vd)->list, row);
133
134         if (fd) return fd->path;
135
136         return NULL;
137 }
138
139 static gboolean vdlist_populate(ViewDir *vd, gboolean clear)
140 {
141         GtkListStore *store;
142         GList *work;
143         GtkTreeIter iter;
144         gboolean valid;
145         gchar *filepath;
146         GList *old_list;
147         gboolean ret;
148         FileData *fd;
149         SortType sort_type = SORT_NAME;
150         gboolean sort_ascend = TRUE;
151
152         old_list = VDLIST(vd)->list;
153
154         ret = filelist_read(vd->dir_fd, NULL, &VDLIST(vd)->list);
155         VDLIST(vd)->list = filelist_sort(VDLIST(vd)->list, sort_type, sort_ascend);
156
157         /* add . and .. */
158
159         if (strcmp(vd->dir_fd->path, G_DIR_SEPARATOR_S) != 0)
160                 {
161                 filepath = g_build_filename(vd->dir_fd->path, "..", NULL);
162                 fd = file_data_new_dir(filepath);
163                 VDLIST(vd)->list = g_list_prepend(VDLIST(vd)->list, fd);
164                 g_free(filepath);
165                 }
166
167         if (options->file_filter.show_dot_directory)
168                 {
169                 filepath = g_build_filename(vd->dir_fd->path, ".", NULL);
170                 fd = file_data_new_dir(filepath);
171                 VDLIST(vd)->list = g_list_prepend(VDLIST(vd)->list, fd);
172                 g_free(filepath);
173         }
174
175         store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view)));
176         if (clear) gtk_list_store_clear(store);
177
178         valid = gtk_tree_model_iter_children(GTK_TREE_MODEL(store), &iter, NULL);
179
180         work = VDLIST(vd)->list;
181         while (work)
182                 {
183                 gint match;
184                 GdkPixbuf *pixbuf;
185                 const gchar *date = "";
186                 gboolean done = FALSE;
187                 
188                 fd = work->data;
189
190                 if (access_file(fd->path, R_OK | X_OK) && fd->name)
191                         {
192                         if (fd->name[0] == '.' && fd->name[1] == '\0')
193                                 {
194                                 pixbuf = vd->pf->open;
195                                 }
196                         else if (fd->name[0] == '.' && fd->name[1] == '.' && fd->name[2] == '\0')
197                                 {
198                                 pixbuf = vd->pf->parent;
199                                 }
200                         else
201                                 {
202                                 pixbuf = vd->pf->close;
203                                 if (vd->layout && vd->layout->options.show_directory_date)
204                                         date = text_from_time(fd->date);
205                                 }
206                         }
207                 else
208                         {
209                         pixbuf = vd->pf->deny;
210                         }
211
212                 while (!done)
213                         {
214                         FileData *old_fd = NULL;
215
216                         if (valid)
217                                 {
218                                 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter,
219                                                    DIR_COLUMN_POINTER, &old_fd,
220                                                    -1);
221
222                                 if (fd == old_fd)
223                                         {
224                                         match = 0;
225                                         }
226                                 else
227                                         {
228                                         match = filelist_sort_compare_filedata_full(fd, old_fd, sort_type, sort_ascend);
229
230                                         if (match == 0) g_warning("multiple fd for the same path");
231                                         }
232                                         
233                                 }
234                         else
235                                 {
236                                 match = -1;
237                                 }
238
239                         if (match < 0)
240                                 {
241                                 GtkTreeIter new;
242
243                                 if (valid)
244                                         {
245                                         gtk_list_store_insert_before(store, &new, &iter);
246                                         }
247                                 else
248                                         {
249                                         gtk_list_store_append(store, &new);
250                                         }
251
252                                 gtk_list_store_set(store, &new,
253                                                    DIR_COLUMN_POINTER, fd,
254                                                    DIR_COLUMN_ICON, pixbuf,
255                                                    DIR_COLUMN_NAME, fd->name,
256                                                    DIR_COLUMN_DATE, date,
257                                                    -1);
258
259                                 done = TRUE;
260                                 }
261                         else if (match > 0)
262                                 {
263                                 valid = gtk_list_store_remove(store, &iter);
264                                 }
265                         else
266                                 {
267                                 gtk_list_store_set(store, &iter,
268                                                    DIR_COLUMN_ICON, pixbuf,
269                                                    DIR_COLUMN_NAME, fd->name,
270                                                    DIR_COLUMN_DATE, date,
271                                                    -1);
272
273                                 if (valid) valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter);
274
275                                 done = TRUE;
276                                 }
277                         }
278                 work = work->next;
279                 }
280
281         while (valid)
282                 {
283                 FileData *old_fd;
284                 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, DIR_COLUMN_POINTER, &old_fd, -1);
285
286                 valid = gtk_list_store_remove(store, &iter);
287                 }
288                 
289
290         vd->click_fd = NULL;
291         vd->drop_fd = NULL;
292
293         filelist_free(old_list);
294         return ret;
295 }
296
297 gboolean vdlist_set_fd(ViewDir *vd, FileData *dir_fd)
298 {
299         gboolean ret;
300         gchar *old_path = NULL;
301
302         if (!dir_fd) return FALSE;
303         if (vd->dir_fd == dir_fd) return TRUE;
304
305         if (vd->dir_fd)
306                 {
307                 gchar *base;
308
309                 base = remove_level_from_path(vd->dir_fd->path);
310                 if (strcmp(base, dir_fd->path) == 0)
311                         {
312                         old_path = g_strdup(vd->dir_fd->name);
313                         }
314                 g_free(base);
315                 }
316
317         file_data_unref(vd->dir_fd);
318         vd->dir_fd = file_data_ref(dir_fd);
319
320         ret = vdlist_populate(vd, TRUE);
321
322         if (old_path)
323                 {
324                 /* scroll to make last path visible */
325                 FileData *found = NULL;
326                 GList *work;
327
328                 work = VDLIST(vd)->list;
329                 while (work && !found)
330                         {
331                         FileData *fd = work->data;
332                         if (strcmp(old_path, fd->name) == 0) found = fd;
333                         work = work->next;
334                         }
335
336                 if (found) vdlist_scroll_to_row(vd, found, 0.5);
337
338                 g_free(old_path);
339                 return ret;
340                 }
341
342 #if GTK_CHECK_VERSION(2,20,0)
343         if (gtk_widget_get_realized(vd->view))
344 #else
345         if (GTK_WIDGET_REALIZED(vd->view))
346 #endif
347                 {
348                 gtk_tree_view_scroll_to_point(GTK_TREE_VIEW(vd->view), 0, 0);
349                 }
350
351         return ret;
352 }
353
354 void vdlist_refresh(ViewDir *vd)
355 {
356         vdlist_populate(vd, FALSE);
357 }
358
359 gboolean vdlist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data)
360 {
361         ViewDir *vd = data;
362         GtkTreePath *tpath;
363
364         if (event->keyval != GDK_Menu) return FALSE;
365
366         gtk_tree_view_get_cursor(GTK_TREE_VIEW(vd->view), &tpath, NULL);
367         if (tpath)
368                 {
369                 GtkTreeModel *store;
370                 GtkTreeIter iter;
371
372                 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
373                 gtk_tree_model_get_iter(store, &iter, tpath);
374                 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &vd->click_fd, -1);
375
376                 gtk_tree_path_free(tpath);
377                 }
378         else
379                 {
380                 vd->click_fd = NULL;
381                 }
382
383         vd_color_set(vd, vd->click_fd, TRUE);
384
385         vd->popup = vd_pop_menu(vd, vd->click_fd);
386
387         gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, vd_menu_position_cb, vd, 0, GDK_CURRENT_TIME);
388
389         return TRUE;
390 }
391
392 gboolean vdlist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data)
393 {
394         ViewDir *vd = data;
395         GtkTreePath *tpath;
396         GtkTreeIter iter;
397         FileData *fd = NULL;
398
399         if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y,
400                                           &tpath, NULL, NULL, NULL))
401                 {
402                 GtkTreeModel *store;
403
404                 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget));
405                 gtk_tree_model_get_iter(store, &iter, tpath);
406                 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1);
407                 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE);
408                 gtk_tree_path_free(tpath);
409                 }
410
411         vd->click_fd = fd;
412         vd_color_set(vd, vd->click_fd, TRUE);
413
414         if (bevent->button == MOUSE_BUTTON_RIGHT)
415                 {
416                 vd->popup = vd_pop_menu(vd, vd->click_fd);
417                 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, NULL, NULL,
418                                bevent->button, bevent->time);
419                 }
420
421         return TRUE;
422 }
423
424 void vdlist_destroy_cb(GtkWidget *widget, gpointer data)
425 {
426         ViewDir *vd = data;
427
428         vd_dnd_drop_scroll_cancel(vd);
429         widget_auto_scroll_stop(vd->view);
430
431         filelist_free(VDLIST(vd)->list);
432 }
433
434 ViewDir *vdlist_new(ViewDir *vd, FileData *dir_fd)
435 {
436         GtkListStore *store;
437         GtkTreeSelection *selection;
438         GtkTreeViewColumn *column;
439         GtkCellRenderer *renderer;
440
441         vd->info = g_new0(ViewDirInfoList, 1);
442
443         vd->type = DIRVIEW_LIST;
444
445         store = gtk_list_store_new(5, G_TYPE_POINTER, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_STRING);
446         vd->view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
447         g_object_unref(store);
448
449         gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vd->view), FALSE);
450         gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vd->view), FALSE);
451
452         selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vd->view));
453         gtk_tree_selection_set_mode(selection, GTK_SELECTION_NONE);
454
455         column = gtk_tree_view_column_new();
456         gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE);
457
458         renderer = gtk_cell_renderer_pixbuf_new();
459         gtk_tree_view_column_pack_start(column, renderer, FALSE);
460         gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", DIR_COLUMN_ICON);
461         gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL);
462
463         renderer = gtk_cell_renderer_text_new();
464         gtk_tree_view_column_pack_start(column, renderer, TRUE);
465         gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_NAME);
466         gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL);
467
468         renderer = gtk_cell_renderer_text_new();
469         gtk_tree_view_column_pack_start(column, renderer, TRUE);
470         gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_DATE);
471         gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL);
472
473         gtk_tree_view_append_column(GTK_TREE_VIEW(vd->view), column);
474
475         return vd;
476 }
477 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */