Mercurial > geeqie.yaz
annotate src/view_dir_list.c @ 1738:808dd5257146
Some systems do not have SA_SIGINFO (siginfo_t), like GNU/Hurd, so just test if defined or not. Patch by Michal iha (Bug ID: 2894271).
author | zas_ |
---|---|
date | Fri, 04 Dec 2009 21:01:11 +0000 |
parents | 94e4a47ccaff |
children | 9b5eb41f4311 |
rev | line source |
---|---|
9 | 1 /* |
196 | 2 * Geeqie |
9 | 3 * (C) 2004 John Ellis |
1284 | 4 * Copyright (C) 2008 - 2009 The Geeqie Team |
9 | 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 | |
281 | 13 #include "main.h" |
9 | 14 #include "view_dir_list.h" |
15 | |
16 #include "dnd.h" | |
17 #include "dupe.h" | |
586 | 18 #include "filedata.h" |
9 | 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" | |
380
5afe77bb563a
Introduce a new struct ViewDir to handle directory views common
zas_
parents:
373
diff
changeset
|
26 #include "view_dir.h" |
9 | 27 |
28 #include <gdk/gdkkeysyms.h> /* for keyboard values */ | |
29 | |
30 | |
979 | 31 #define VDLIST(_vd_) ((ViewDirInfoList *)(_vd_->info)) |
9 | 32 |
33 | |
34 /* | |
35 *----------------------------------------------------------------------------- | |
36 * misc | |
37 *----------------------------------------------------------------------------- | |
38 */ | |
39 | |
1452 | 40 gboolean vdlist_find_row(ViewDir *vd, FileData *fd, GtkTreeIter *iter) |
9 | 41 { |
42 GtkTreeModel *store; | |
1452 | 43 gboolean valid; |
9 | 44 |
381 | 45 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view)); |
9 | 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); | |
1501 | 51 if (fd_n == fd) return TRUE; |
9 | 52 |
53 valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), iter); | |
54 } | |
55 | |
1501 | 56 return FALSE; |
9 | 57 } |
58 | |
59 | |
388
5186f8e38cb8
Make directory view popup menu common and move it to view_dir.{c,h}.
zas_
parents:
385
diff
changeset
|
60 FileData *vdlist_row_by_path(ViewDir *vd, const gchar *path, gint *row) |
9 | 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; | |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
72 work = VDLIST(vd)->list; |
9 | 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 | |
381 | 95 static void vdlist_scroll_to_row(ViewDir *vd, FileData *fd, gfloat y_align) |
9 | 96 { |
97 GtkTreeIter iter; | |
98 | |
1553 | 99 if (GTK_WIDGET_REALIZED(vd->view) && vd_find_row(vd, fd, &iter)) |
9 | 100 { |
101 GtkTreeModel *store; | |
102 GtkTreePath *tpath; | |
103 | |
381 | 104 store = gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view)); |
9 | 105 tpath = gtk_tree_model_get_path(store, &iter); |
381 | 106 gtk_tree_view_scroll_to_cell(GTK_TREE_VIEW(vd->view), tpath, NULL, TRUE, y_align, 0.0); |
107 gtk_tree_view_set_cursor(GTK_TREE_VIEW(vd->view), tpath, NULL, FALSE); | |
9 | 108 gtk_tree_path_free(tpath); |
109 | |
381 | 110 if (!GTK_WIDGET_HAS_FOCUS(vd->view)) gtk_widget_grab_focus(vd->view); |
9 | 111 } |
112 } | |
113 | |
114 /* | |
115 *----------------------------------------------------------------------------- | |
116 * main | |
117 *----------------------------------------------------------------------------- | |
442 | 118 */ |
9 | 119 |
381 | 120 const gchar *vdlist_row_get_path(ViewDir *vd, gint row) |
9 | 121 { |
122 FileData *fd; | |
123 | |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
124 fd = g_list_nth_data(VDLIST(vd)->list, row); |
9 | 125 |
126 if (fd) return fd->path; | |
127 | |
128 return NULL; | |
129 } | |
130 | |
1452 | 131 static gboolean vdlist_populate(ViewDir *vd, gboolean clear) |
9 | 132 { |
133 GtkListStore *store; | |
134 GList *work; | |
973 | 135 GtkTreeIter iter; |
1452 | 136 gboolean valid; |
973 | 137 gchar *filepath; |
138 GList *old_list; | |
1452 | 139 gboolean ret; |
973 | 140 FileData *fd; |
141 | |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
142 old_list = VDLIST(vd)->list; |
973 | 143 |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
144 ret = filelist_read(vd->dir_fd, NULL, &VDLIST(vd)->list); |
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
145 VDLIST(vd)->list = filelist_sort(VDLIST(vd)->list, SORT_NAME, TRUE); |
973 | 146 |
147 /* add . and .. */ | |
148 | |
149 if (strcmp(vd->dir_fd->path, G_DIR_SEPARATOR_S) != 0) | |
150 { | |
151 filepath = g_build_filename(vd->dir_fd->path, "..", NULL); | |
152 fd = file_data_new_simple(filepath); | |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
153 VDLIST(vd)->list = g_list_prepend(VDLIST(vd)->list, fd); |
973 | 154 g_free(filepath); |
155 } | |
156 | |
157 if (options->file_filter.show_dot_directory) | |
158 { | |
159 filepath = g_build_filename(vd->dir_fd->path, ".", NULL); | |
160 fd = file_data_new_simple(filepath); | |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
161 VDLIST(vd)->list = g_list_prepend(VDLIST(vd)->list, fd); |
973 | 162 g_free(filepath); |
163 } | |
9 | 164 |
381 | 165 store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(vd->view))); |
973 | 166 if (clear) gtk_list_store_clear(store); |
167 | |
168 valid = gtk_tree_model_iter_children(GTK_TREE_MODEL(store), &iter, NULL); | |
9 | 169 |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
170 work = VDLIST(vd)->list; |
9 | 171 while (work) |
172 { | |
973 | 173 gint match; |
9 | 174 GdkPixbuf *pixbuf; |
825 | 175 const gchar *date = ""; |
1452 | 176 gboolean done = FALSE; |
177 | |
9 | 178 fd = work->data; |
179 | |
180 if (access_file(fd->path, R_OK | X_OK) && fd->name) | |
181 { | |
182 if (fd->name[0] == '.' && fd->name[1] == '\0') | |
183 { | |
381 | 184 pixbuf = vd->pf->open; |
9 | 185 } |
186 else if (fd->name[0] == '.' && fd->name[1] == '.' && fd->name[2] == '\0') | |
187 { | |
381 | 188 pixbuf = vd->pf->parent; |
9 | 189 } |
190 else | |
191 { | |
381 | 192 pixbuf = vd->pf->close; |
1436
d7a6fb7a90dd
completely separated global and layout window options
nadvornik
parents:
1365
diff
changeset
|
193 if (vd->layout && vd->layout->options.show_directory_date) |
825 | 194 date = text_from_time(fd->date); |
9 | 195 } |
196 } | |
197 else | |
198 { | |
381 | 199 pixbuf = vd->pf->deny; |
9 | 200 } |
201 | |
973 | 202 while (!done) |
203 { | |
204 FileData *old_fd = NULL; | |
205 | |
206 if (valid) | |
207 { | |
208 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, | |
209 DIR_COLUMN_POINTER, &old_fd, | |
210 -1); | |
211 | |
212 if (fd == old_fd) | |
213 { | |
214 match = 0; | |
215 } | |
216 else | |
217 { | |
995 | 218 match = filelist_sort_compare_filedata_full(fd, old_fd, SORT_NAME, TRUE); |
973 | 219 |
220 if (match == 0) g_warning("multiple fd for the same path"); | |
221 } | |
222 | |
223 } | |
224 else | |
225 { | |
226 match = -1; | |
227 } | |
228 | |
229 if (match < 0) | |
230 { | |
231 GtkTreeIter new; | |
9 | 232 |
973 | 233 if (valid) |
234 { | |
235 gtk_list_store_insert_before(store, &new, &iter); | |
236 } | |
237 else | |
238 { | |
239 gtk_list_store_append(store, &new); | |
240 } | |
241 | |
242 gtk_list_store_set(store, &new, | |
243 DIR_COLUMN_POINTER, fd, | |
244 DIR_COLUMN_ICON, pixbuf, | |
245 DIR_COLUMN_NAME, fd->name, | |
246 DIR_COLUMN_DATE, date, | |
247 -1); | |
248 | |
249 done = TRUE; | |
250 } | |
251 else if (match > 0) | |
252 { | |
253 valid = gtk_list_store_remove(store, &iter); | |
254 } | |
255 else | |
256 { | |
257 gtk_list_store_set(store, &iter, | |
258 DIR_COLUMN_ICON, pixbuf, | |
259 DIR_COLUMN_NAME, fd->name, | |
260 DIR_COLUMN_DATE, date, | |
261 -1); | |
262 | |
263 if (valid) valid = gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &iter); | |
264 | |
265 done = TRUE; | |
266 } | |
267 } | |
9 | 268 work = work->next; |
269 } | |
270 | |
973 | 271 while (valid) |
272 { | |
273 FileData *old_fd; | |
274 gtk_tree_model_get(GTK_TREE_MODEL(store), &iter, DIR_COLUMN_POINTER, &old_fd, -1); | |
275 | |
276 valid = gtk_list_store_remove(store, &iter); | |
277 } | |
278 | |
279 | |
381 | 280 vd->click_fd = NULL; |
281 vd->drop_fd = NULL; | |
973 | 282 |
283 filelist_free(old_list); | |
284 return ret; | |
9 | 285 } |
286 | |
1452 | 287 gboolean vdlist_set_fd(ViewDir *vd, FileData *dir_fd) |
9 | 288 { |
1452 | 289 gboolean ret; |
9 | 290 gchar *old_path = NULL; |
291 | |
783 | 292 if (!dir_fd) return FALSE; |
293 if (vd->dir_fd == dir_fd) return TRUE; | |
9 | 294 |
783 | 295 if (vd->dir_fd) |
9 | 296 { |
297 gchar *base; | |
298 | |
783 | 299 base = remove_level_from_path(vd->dir_fd->path); |
300 if (strcmp(base, dir_fd->path) == 0) | |
9 | 301 { |
783 | 302 old_path = g_strdup(vd->dir_fd->name); |
9 | 303 } |
304 g_free(base); | |
305 } | |
306 | |
783 | 307 file_data_unref(vd->dir_fd); |
308 vd->dir_fd = file_data_ref(dir_fd); | |
9 | 309 |
973 | 310 ret = vdlist_populate(vd, TRUE); |
9 | 311 |
312 if (old_path) | |
313 { | |
314 /* scroll to make last path visible */ | |
315 FileData *found = NULL; | |
316 GList *work; | |
317 | |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
318 work = VDLIST(vd)->list; |
9 | 319 while (work && !found) |
320 { | |
321 FileData *fd = work->data; | |
322 if (strcmp(old_path, fd->name) == 0) found = fd; | |
323 work = work->next; | |
324 } | |
325 | |
381 | 326 if (found) vdlist_scroll_to_row(vd, found, 0.5); |
9 | 327 |
328 g_free(old_path); | |
329 return ret; | |
330 } | |
331 | |
381 | 332 if (GTK_WIDGET_REALIZED(vd->view)) |
9 | 333 { |
381 | 334 gtk_tree_view_scroll_to_point(GTK_TREE_VIEW(vd->view), 0, 0); |
9 | 335 } |
336 | |
337 return ret; | |
338 } | |
339 | |
381 | 340 void vdlist_refresh(ViewDir *vd) |
9 | 341 { |
973 | 342 vdlist_populate(vd, FALSE); |
9 | 343 } |
344 | |
1452 | 345 gboolean vdlist_press_key_cb(GtkWidget *widget, GdkEventKey *event, gpointer data) |
9 | 346 { |
381 | 347 ViewDir *vd = data; |
9 | 348 GtkTreePath *tpath; |
442 | 349 |
9 | 350 if (event->keyval != GDK_Menu) return FALSE; |
351 | |
381 | 352 gtk_tree_view_get_cursor(GTK_TREE_VIEW(vd->view), &tpath, NULL); |
9 | 353 if (tpath) |
354 { | |
355 GtkTreeModel *store; | |
356 GtkTreeIter iter; | |
357 | |
358 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget)); | |
359 gtk_tree_model_get_iter(store, &iter, tpath); | |
381 | 360 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &vd->click_fd, -1); |
442 | 361 |
9 | 362 gtk_tree_path_free(tpath); |
363 } | |
364 else | |
365 { | |
381 | 366 vd->click_fd = NULL; |
9 | 367 } |
368 | |
383
499d7ba62261
Move some dnd common code from view_dir_list.c and view_dir_tree.c
zas_
parents:
381
diff
changeset
|
369 vd_color_set(vd, vd->click_fd, TRUE); |
9 | 370 |
388
5186f8e38cb8
Make directory view popup menu common and move it to view_dir.{c,h}.
zas_
parents:
385
diff
changeset
|
371 vd->popup = vd_pop_menu(vd, vd->click_fd); |
9 | 372 |
395 | 373 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, vd_menu_position_cb, vd, 0, GDK_CURRENT_TIME); |
9 | 374 |
375 return TRUE; | |
376 } | |
377 | |
1452 | 378 gboolean vdlist_press_cb(GtkWidget *widget, GdkEventButton *bevent, gpointer data) |
9 | 379 { |
381 | 380 ViewDir *vd = data; |
9 | 381 GtkTreePath *tpath; |
382 GtkTreeIter iter; | |
383 FileData *fd = NULL; | |
384 | |
385 if (gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget), bevent->x, bevent->y, | |
386 &tpath, NULL, NULL, NULL)) | |
387 { | |
388 GtkTreeModel *store; | |
389 | |
390 store = gtk_tree_view_get_model(GTK_TREE_VIEW(widget)); | |
391 gtk_tree_model_get_iter(store, &iter, tpath); | |
392 gtk_tree_model_get(store, &iter, DIR_COLUMN_POINTER, &fd, -1); | |
393 gtk_tree_view_set_cursor(GTK_TREE_VIEW(widget), tpath, NULL, FALSE); | |
394 gtk_tree_path_free(tpath); | |
395 } | |
396 | |
381 | 397 vd->click_fd = fd; |
383
499d7ba62261
Move some dnd common code from view_dir_list.c and view_dir_tree.c
zas_
parents:
381
diff
changeset
|
398 vd_color_set(vd, vd->click_fd, TRUE); |
9 | 399 |
448
a73cc0fa14d0
Use explicit names for mouse buttons instead of numbers.
zas_
parents:
442
diff
changeset
|
400 if (bevent->button == MOUSE_BUTTON_RIGHT) |
9 | 401 { |
388
5186f8e38cb8
Make directory view popup menu common and move it to view_dir.{c,h}.
zas_
parents:
385
diff
changeset
|
402 vd->popup = vd_pop_menu(vd, vd->click_fd); |
381 | 403 gtk_menu_popup(GTK_MENU(vd->popup), NULL, NULL, NULL, NULL, |
9 | 404 bevent->button, bevent->time); |
405 } | |
406 | |
407 return TRUE; | |
408 } | |
409 | |
401
0a2e1b130a25
Add some wrappers in view_dir.c and simplify even more.
zas_
parents:
397
diff
changeset
|
410 void vdlist_destroy_cb(GtkWidget *widget, gpointer data) |
9 | 411 { |
381 | 412 ViewDir *vd = data; |
9 | 413 |
394 | 414 vd_dnd_drop_scroll_cancel(vd); |
381 | 415 widget_auto_scroll_stop(vd->view); |
9 | 416 |
978
9ff64efe11eb
Replace macros VDLIST_INFO() and VDTREE_INFO() by shorter VDLIST() and VDTREE(). VDLIST_INFO(vd, part) becomes VDLIST(vd)->part.
zas_
parents:
973
diff
changeset
|
417 filelist_free(VDLIST(vd)->list); |
9 | 418 } |
419 | |
783 | 420 ViewDir *vdlist_new(ViewDir *vd, FileData *dir_fd) |
9 | 421 { |
422 GtkListStore *store; | |
423 GtkTreeSelection *selection; | |
424 GtkTreeViewColumn *column; | |
425 GtkCellRenderer *renderer; | |
426 | |
381 | 427 vd->info = g_new0(ViewDirInfoList, 1); |
1365
249bf204004a
When g_new0() is used, drop redundant initializations to NULL, FALSE or 0.
zas_
parents:
1284
diff
changeset
|
428 |
381 | 429 vd->type = DIRVIEW_LIST; |
9 | 430 |
825 | 431 store = gtk_list_store_new(5, G_TYPE_POINTER, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_STRING); |
381 | 432 vd->view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store)); |
9 | 433 g_object_unref(store); |
434 | |
381 | 435 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(vd->view), FALSE); |
436 gtk_tree_view_set_enable_search(GTK_TREE_VIEW(vd->view), FALSE); | |
9 | 437 |
381 | 438 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(vd->view)); |
9 | 439 gtk_tree_selection_set_mode(selection, GTK_SELECTION_NONE); |
440 | |
441 column = gtk_tree_view_column_new(); | |
442 gtk_tree_view_column_set_sizing(column, GTK_TREE_VIEW_COLUMN_AUTOSIZE); | |
443 | |
444 renderer = gtk_cell_renderer_pixbuf_new(); | |
445 gtk_tree_view_column_pack_start(column, renderer, FALSE); | |
446 gtk_tree_view_column_add_attribute(column, renderer, "pixbuf", DIR_COLUMN_ICON); | |
396 | 447 gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL); |
9 | 448 |
449 renderer = gtk_cell_renderer_text_new(); | |
450 gtk_tree_view_column_pack_start(column, renderer, TRUE); | |
451 gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_NAME); | |
396 | 452 gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL); |
9 | 453 |
825 | 454 renderer = gtk_cell_renderer_text_new(); |
455 gtk_tree_view_column_pack_start(column, renderer, TRUE); | |
456 gtk_tree_view_column_add_attribute(column, renderer, "text", DIR_COLUMN_DATE); | |
457 gtk_tree_view_column_set_cell_data_func(column, renderer, vd_color_cb, vd, NULL); | |
458 | |
381 | 459 gtk_tree_view_append_column(GTK_TREE_VIEW(vd->view), column); |
9 | 460 |
381 | 461 return vd; |
9 | 462 } |
1055
1646720364cf
Adding a vim modeline to all files - patch by Klaus Ethgen
nadvornik
parents:
995
diff
changeset
|
463 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */ |