Mercurial > geeqie
annotate src/collect.c @ 1204:fa91098e4949
queue metadata and write them in an idle callback
author | nadvornik |
---|---|
date | Tue, 16 Dec 2008 12:59:23 +0000 |
parents | d3e51e1a02f8 |
children | e2bbe90b0dcd |
rev | line source |
---|---|
9 | 1 /* |
196 | 2 * Geeqie |
94
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
3 * (C) 2006 John Ellis |
475 | 4 * Copyright (C) 2008 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 | |
13 | |
281 | 14 #include "main.h" |
9 | 15 #include "collect.h" |
16 | |
17 #include "collect-dlg.h" | |
18 #include "collect-io.h" | |
19 #include "collect-table.h" | |
20 #include "editors.h" | |
586 | 21 #include "filedata.h" |
9 | 22 #include "img-view.h" |
23 #include "info.h" | |
24 #include "layout.h" | |
25 #include "layout_image.h" | |
1022
9962b24b6b43
Move miscellaneous functions to their own files (new misc.[ch]).
zas_
parents:
816
diff
changeset
|
26 #include "misc.h" |
91
d063f97503b7
Wed Nov 1 11:39:48 2006 John Ellis <johne@verizon.net>
gqview
parents:
85
diff
changeset
|
27 #include "pixbuf_util.h" |
94
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
28 #include "print.h" |
9 | 29 #include "ui_fileops.h" |
30 #include "ui_tree_edit.h" | |
1022
9962b24b6b43
Move miscellaneous functions to their own files (new misc.[ch]).
zas_
parents:
816
diff
changeset
|
31 #include "utilops.h" |
648
e34c1002e553
Move some functions from main.[ch] to new window.[ch].
zas_
parents:
586
diff
changeset
|
32 #include "window.h" |
9 | 33 |
34 #include <gdk/gdkkeysyms.h> /* for keyboard values */ | |
35 | |
36 | |
37 #define COLLECT_DEF_WIDTH 440 | |
38 #define COLLECT_DEF_HEIGHT 450 | |
39 | |
40 static GList *collection_list = NULL; | |
41 static GList *collection_window_list = NULL; | |
42 | |
43 static void collection_window_get_geometry(CollectWindow *cw); | |
44 static void collection_window_refresh(CollectWindow *cw); | |
45 static void collection_window_update_title(CollectWindow *cw); | |
46 static void collection_window_add(CollectWindow *cw, CollectInfo *ci); | |
47 static void collection_window_insert(CollectWindow *cw, CollectInfo *ci); | |
48 static void collection_window_remove(CollectWindow *cw, CollectInfo *ci); | |
49 static void collection_window_update(CollectWindow *cw, CollectInfo *ci); | |
50 | |
51 static void collection_window_close(CollectWindow *cw); | |
52 | |
799 | 53 static void collection_notify_cb(FileData *fd, NotifyType type, gpointer data); |
54 | |
9 | 55 /* |
56 *------------------------------------------------------------------- | |
57 * data, list handling | |
58 *------------------------------------------------------------------- | |
59 */ | |
60 | |
138 | 61 CollectInfo *collection_info_new(FileData *fd, struct stat *st, GdkPixbuf *pixbuf) |
9 | 62 { |
63 CollectInfo *ci; | |
64 | |
138 | 65 if (!fd) return NULL; |
9 | 66 |
67 ci = g_new0(CollectInfo, 1); | |
138 | 68 ci->fd = file_data_ref(fd); |
9 | 69 |
70 ci->pixbuf = pixbuf; | |
71 if (ci->pixbuf) g_object_ref(ci->pixbuf); | |
72 | |
73 return ci; | |
74 } | |
75 | |
76 void collection_info_free_thumb(CollectInfo *ci) | |
77 { | |
78 if (ci->pixbuf) g_object_unref(ci->pixbuf); | |
79 ci->pixbuf = NULL; | |
80 } | |
81 | |
82 void collection_info_free(CollectInfo *ci) | |
83 { | |
84 if (!ci) return; | |
85 | |
138 | 86 file_data_unref(ci->fd); |
9 | 87 collection_info_free_thumb(ci); |
88 g_free(ci); | |
89 } | |
90 | |
91 void collection_info_set_thumb(CollectInfo *ci, GdkPixbuf *pixbuf) | |
92 { | |
93 if (pixbuf) g_object_ref(pixbuf); | |
94 collection_info_free_thumb(ci); | |
95 ci->pixbuf = pixbuf; | |
96 } | |
97 | |
98 gint collection_info_load_thumb(CollectInfo *ci) | |
99 { | |
100 if (!ci) return FALSE; | |
101 | |
102 collection_info_free_thumb(ci); | |
103 | |
673
fbebf5cf4a55
Do not use printf() directly but use new wrapper function log_printf() instead.
zas_
parents:
671
diff
changeset
|
104 log_printf("collection_info_load_thumb not implemented!\n(because an instant thumb loader not implemented)"); |
9 | 105 return FALSE; |
442 | 106 #if 0 |
138 | 107 if (create_thumbnail(ci->fd->path, &ci->pixmap, &ci->mask) < 0) return FALSE; |
9 | 108 |
109 if (ci->pixmap) gdk_pixmap_ref(ci->pixmap); | |
110 if (ci->mask) gdk_bitmap_ref(ci->mask); | |
111 | |
112 return TRUE; | |
113 #endif | |
114 } | |
115 | |
116 void collection_list_free(GList *list) | |
117 { | |
118 GList *work; | |
119 work = list; | |
516 | 120 while (work) |
9 | 121 { |
122 collection_info_free((CollectInfo *)work->data); | |
123 work = work->next; | |
124 } | |
125 g_list_free(list); | |
126 } | |
127 | |
128 /* an ugly static var, well what ya gonna do ? */ | |
129 static SortType collection_list_sort_method = SORT_NAME; | |
130 | |
131 static gint collection_list_sort_cb(gconstpointer a, gconstpointer b) | |
132 { | |
133 const CollectInfo *cia = a; | |
134 const CollectInfo *cib = b; | |
135 | |
136 switch(collection_list_sort_method) | |
137 { | |
785 | 138 case SORT_NAME: |
139 break; | |
9 | 140 case SORT_NONE: |
141 return 0; | |
142 break; | |
143 case SORT_SIZE: | |
138 | 144 if (cia->fd->size < cib->fd->size) return -1; |
145 if (cia->fd->size > cib->fd->size) return 1; | |
9 | 146 return 0; |
147 break; | |
148 case SORT_TIME: | |
138 | 149 if (cia->fd->date < cib->fd->date) return -1; |
150 if (cia->fd->date > cib->fd->date) return 1; | |
9 | 151 return 0; |
152 break; | |
153 case SORT_PATH: | |
786
a20ff446347e
Compare paths using utf8_collate_key() since paths are utf8-encoded.
zas_
parents:
785
diff
changeset
|
154 return utf8_compare(cia->fd->path, cib->fd->path, options->file_sort.case_sensitive); |
9 | 155 break; |
156 #ifdef HAVE_STRVERSCMP | |
157 case SORT_NUMBER: | |
138 | 158 return strverscmp(cia->fd->name, cib->fd->name); |
9 | 159 break; |
160 #endif | |
161 default: | |
162 break; | |
163 } | |
164 | |
785 | 165 if (options->file_sort.case_sensitive) |
166 return strcmp(cia->fd->collate_key_name, cib->fd->collate_key_name); | |
167 else | |
168 return strcmp(cia->fd->collate_key_name_nocase, cib->fd->collate_key_name_nocase); | |
9 | 169 } |
170 | |
171 GList *collection_list_sort(GList *list, SortType method) | |
172 { | |
173 if (method == SORT_NONE) return list; | |
174 | |
175 collection_list_sort_method = method; | |
176 | |
177 return g_list_sort(list, collection_list_sort_cb); | |
178 } | |
179 | |
180 GList *collection_list_add(GList *list, CollectInfo *ci, SortType method) | |
181 { | |
182 if (method != SORT_NONE) | |
183 { | |
184 collection_list_sort_method = method; | |
185 list = g_list_insert_sorted(list, ci, collection_list_sort_cb); | |
186 } | |
187 else | |
188 { | |
189 list = g_list_append(list, ci); | |
190 } | |
191 | |
192 return list; | |
193 } | |
194 | |
195 GList *collection_list_insert(GList *list, CollectInfo *ci, CollectInfo *insert_ci, SortType method) | |
196 { | |
197 if (method != SORT_NONE) | |
198 { | |
199 collection_list_sort_method = method; | |
200 list = g_list_insert_sorted(list, ci, collection_list_sort_cb); | |
201 } | |
202 else | |
203 { | |
204 GList *point; | |
205 | |
206 point = g_list_find(list, insert_ci); | |
207 list = uig_list_insert_link(list, point, ci); | |
208 } | |
209 | |
210 return list; | |
211 } | |
212 | |
213 GList *collection_list_remove(GList *list, CollectInfo *ci) | |
214 { | |
215 list = g_list_remove(list, ci); | |
216 collection_info_free(ci); | |
217 return list; | |
218 } | |
219 | |
799 | 220 CollectInfo *collection_list_find_fd(GList *list, FileData *fd) |
9 | 221 { |
222 GList *work = list; | |
223 | |
516 | 224 while (work) |
9 | 225 { |
226 CollectInfo *ci = work->data; | |
799 | 227 if (ci->fd == fd) return ci; |
9 | 228 work = work->next; |
229 } | |
230 | |
231 return NULL; | |
232 } | |
233 | |
234 #if 0 | |
235 static GList *collection_list_find_link(GList *list, gchar *path) | |
236 { | |
237 GList *work = list; | |
238 | |
516 | 239 while (work) |
9 | 240 { |
241 CollectInfo *ci = work->data; | |
138 | 242 if (strcmp(ci->fd->path, path) == 0) return work; |
9 | 243 work = work->next; |
244 } | |
245 | |
246 return NULL; | |
247 } | |
248 | |
249 static gint collection_list_find_index(GList *list, gchar *path) | |
250 { | |
251 gint c = 0; | |
252 GList *work = list; | |
253 | |
516 | 254 while (work) |
9 | 255 { |
256 CollectInfo *ci = work->data; | |
138 | 257 if (strcmp(ci->fd->path, path) == 0) return c; |
9 | 258 work = work->next; |
259 c++; | |
260 } | |
261 | |
262 return -1; | |
263 } | |
264 #endif | |
265 | |
138 | 266 GList *collection_list_to_filelist(GList *list) |
9 | 267 { |
138 | 268 GList *filelist = NULL; |
9 | 269 GList *work = list; |
270 | |
271 while (work) | |
272 { | |
273 CollectInfo *info = work->data; | |
138 | 274 filelist = g_list_prepend(filelist, file_data_ref(info->fd)); |
9 | 275 work = work->next; |
276 } | |
277 | |
138 | 278 filelist = g_list_reverse(filelist); |
279 return filelist; | |
9 | 280 } |
281 | |
282 CollectWindow *collection_window_find(CollectionData *cd) | |
283 { | |
284 GList *work; | |
285 | |
286 work = collection_window_list; | |
287 while (work) | |
288 { | |
289 CollectWindow *cw = work->data; | |
290 if (cw->cd == cd) return cw; | |
291 work = work->next; | |
292 } | |
293 | |
294 return NULL; | |
295 } | |
296 | |
297 CollectWindow *collection_window_find_by_path(const gchar *path) | |
298 { | |
299 GList *work; | |
300 | |
301 if (!path) return NULL; | |
302 | |
303 work = collection_window_list; | |
304 while (work) | |
305 { | |
306 CollectWindow *cw = work->data; | |
307 if (cw->cd->path && strcmp(cw->cd->path, path) == 0) return cw; | |
308 work = work->next; | |
309 } | |
310 | |
311 return NULL; | |
312 } | |
313 | |
314 /* | |
315 *------------------------------------------------------------------- | |
316 * please use these to actually add/remove stuff | |
317 *------------------------------------------------------------------- | |
318 */ | |
319 | |
320 CollectionData *collection_new(const gchar *path) | |
321 { | |
322 CollectionData *cd; | |
323 static gint untitled_counter = 0; | |
324 | |
325 cd = g_new0(CollectionData, 1); | |
326 | |
327 collection_list = g_list_append(collection_list, cd); | |
328 | |
329 cd->ref = 1; /* starts with a ref of 1 */ | |
330 | |
331 cd->list = NULL; | |
332 cd->sort_method = SORT_NONE; | |
333 cd->thumb_loader = NULL; | |
334 cd->info_updated_func = NULL; | |
335 | |
336 cd->window_read = FALSE; | |
337 cd->window_x = 0; | |
338 cd->window_y = 0; | |
339 cd->window_w = COLLECT_DEF_WIDTH; | |
340 cd->window_h = COLLECT_DEF_HEIGHT; | |
341 | |
342 cd->changed = FALSE; | |
343 | |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
344 cd->existence = g_hash_table_new(NULL, NULL); |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
345 |
9 | 346 if (path) |
347 { | |
348 cd->path = g_strdup(path); | |
349 cd->name = g_strdup(filename_from_path(cd->path)); | |
350 /* load it */ | |
351 } | |
352 else | |
353 { | |
354 cd->path = NULL; | |
355 | |
356 if (untitled_counter == 0) | |
357 { | |
358 cd->name = g_strdup(_("Untitled")); | |
359 } | |
360 else | |
361 { | |
362 cd->name = g_strdup_printf(_("Untitled (%d)"), untitled_counter + 1); | |
363 } | |
364 | |
365 untitled_counter++; | |
366 } | |
367 | |
799 | 368 file_data_register_notify_func(collection_notify_cb, cd, NOTIFY_PRIORITY_MEDIUM); |
369 | |
9 | 370 return cd; |
371 } | |
372 | |
373 void collection_free(CollectionData *cd) | |
374 { | |
375 if (!cd) return; | |
376 | |
506
fc9c8a3e1a8b
Handle the newline in DEBUG_N() macro instead of adding one
zas_
parents:
495
diff
changeset
|
377 DEBUG_1("collection \"%s\" freed", cd->name); |
9 | 378 |
379 collection_load_stop(cd); | |
380 collection_list_free(cd->list); | |
799 | 381 |
382 file_data_unregister_notify_func(collection_notify_cb, cd); | |
9 | 383 |
384 collection_list = g_list_remove(collection_list, cd); | |
385 | |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
386 g_hash_table_destroy(cd->existence); |
442 | 387 |
9 | 388 g_free(cd->path); |
389 g_free(cd->name); | |
390 | |
391 g_free(cd); | |
392 } | |
393 | |
394 void collection_ref(CollectionData *cd) | |
395 { | |
396 cd->ref++; | |
397 | |
506
fc9c8a3e1a8b
Handle the newline in DEBUG_N() macro instead of adding one
zas_
parents:
495
diff
changeset
|
398 DEBUG_1("collection \"%s\" ref count = %d", cd->name, cd->ref); |
9 | 399 } |
400 | |
401 void collection_unref(CollectionData *cd) | |
402 { | |
403 cd->ref--; | |
404 | |
506
fc9c8a3e1a8b
Handle the newline in DEBUG_N() macro instead of adding one
zas_
parents:
495
diff
changeset
|
405 DEBUG_1("collection \"%s\" ref count = %d", cd->name, cd->ref); |
9 | 406 |
407 if (cd->ref < 1) | |
408 { | |
409 collection_free(cd); | |
410 } | |
411 } | |
412 | |
413 void collection_path_changed(CollectionData *cd) | |
414 { | |
415 collection_window_update_title(collection_window_find(cd)); | |
416 } | |
417 | |
418 gint collection_to_number(CollectionData *cd) | |
419 { | |
420 return g_list_index(collection_list, cd); | |
421 } | |
422 | |
423 CollectionData *collection_from_number(gint n) | |
424 { | |
425 return g_list_nth_data(collection_list, n); | |
426 } | |
427 | |
428 CollectionData *collection_from_dnd_data(const gchar *data, GList **list, GList **info_list) | |
429 { | |
430 CollectionData *cd; | |
816 | 431 gint collection_number; |
432 const gchar *ptr; | |
433 | |
434 if (list) *list = NULL; | |
435 if (info_list) *info_list = NULL; | |
9 | 436 |
437 if (strncmp(data, "COLLECTION:", 11) != 0) return NULL; | |
816 | 438 |
439 ptr = data + 11; | |
440 | |
441 collection_number = atoi(ptr); | |
442 cd = collection_from_number(collection_number); | |
443 if (!cd) return NULL; | |
9 | 444 |
816 | 445 if (!list && !info_list) return cd; |
446 | |
447 while (*ptr != '\0' && *ptr != '\n' ) ptr++; | |
448 if (*ptr == '\0') return cd; | |
449 ptr++; | |
9 | 450 |
816 | 451 while (*ptr != '\0') |
452 { | |
453 guint item_number; | |
454 CollectInfo *info; | |
455 | |
456 item_number = (guint) atoi(ptr); | |
457 while (*ptr != '\n' && *ptr != '\0') ptr++; | |
458 if (*ptr == '\0') | |
459 break; | |
460 else | |
461 while (*ptr == '\n') ptr++; | |
9 | 462 |
816 | 463 info = g_list_nth_data(cd->list, item_number); |
464 if (!info) continue; | |
9 | 465 |
816 | 466 if (list) *list = g_list_append(*list, file_data_ref(info->fd)); |
467 if (info_list) *info_list = g_list_append(*info_list, info); | |
9 | 468 } |
816 | 469 |
9 | 470 return cd; |
471 } | |
472 | |
473 gchar *collection_info_list_to_dnd_data(CollectionData *cd, GList *list, gint *length) | |
474 { | |
475 GList *work; | |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
476 GList *temp = NULL; |
9 | 477 gchar *ptr; |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
478 gchar *text; |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
479 gchar *uri_text; |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
480 gint collection_number; |
9 | 481 |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
482 *length = 0; |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
483 if (!list) return NULL; |
9 | 484 |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
485 collection_number = collection_to_number(cd); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
486 if (collection_number < 0) return NULL; |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
487 |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
488 text = g_strdup_printf("COLLECTION:%d\n", collection_number); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
489 *length += strlen(text); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
490 temp = g_list_prepend(temp, text); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
491 |
9 | 492 work = list; |
516 | 493 while (work) |
9 | 494 { |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
495 gint item_number = g_list_index(cd->list, work->data); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
496 |
9 | 497 work = work->next; |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
498 |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
499 if (item_number < 0) continue; |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
500 |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
501 text = g_strdup_printf("%d\n", item_number); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
502 temp = g_list_prepend(temp, text); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
503 *length += strlen(text); |
9 | 504 } |
505 | |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
506 *length += 1; /* ending nul char */ |
9 | 507 |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
508 uri_text = g_malloc(*length); |
9 | 509 ptr = uri_text; |
510 | |
511 work = g_list_last(temp); | |
516 | 512 while (work) |
9 | 513 { |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
514 gint len; |
9 | 515 gchar *text = work->data; |
516 | |
517 work = work->prev; | |
518 | |
815
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
519 len = strlen(text); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
520 memcpy(ptr, text, len); |
7daea970a89a
collection_info_list_to_dnd_data(): simplify and optimize.
zas_
parents:
803
diff
changeset
|
521 ptr += len; |
9 | 522 } |
523 | |
524 ptr[0] = '\0'; | |
525 | |
138 | 526 string_list_free(temp); |
9 | 527 |
528 return uri_text; | |
529 } | |
530 | |
531 gint collection_info_valid(CollectionData *cd, CollectInfo *info) | |
532 { | |
533 if (collection_to_number(cd) < 0) return FALSE; | |
534 | |
535 return (g_list_index(cd->list, info) != 0); | |
536 } | |
537 | |
538 CollectInfo *collection_next_by_info(CollectionData *cd, CollectInfo *info) | |
539 { | |
540 GList *work; | |
541 | |
542 work = g_list_find(cd->list, info); | |
543 | |
544 if (!work) return NULL; | |
545 work = work->next; | |
546 if (work) return work->data; | |
547 return NULL; | |
548 } | |
549 | |
550 CollectInfo *collection_prev_by_info(CollectionData *cd, CollectInfo *info) | |
551 { | |
552 GList *work; | |
553 | |
554 work = g_list_find(cd->list, info); | |
555 | |
556 if (!work) return NULL; | |
557 work = work->prev; | |
558 if (work) return work->data; | |
559 return NULL; | |
560 } | |
561 | |
562 CollectInfo *collection_get_first(CollectionData *cd) | |
563 { | |
564 if (cd->list) return cd->list->data; | |
565 | |
566 return NULL; | |
567 } | |
568 | |
569 CollectInfo *collection_get_last(CollectionData *cd) | |
570 { | |
571 GList *list; | |
572 | |
573 list = g_list_last(cd->list); | |
574 | |
575 if (list) return list->data; | |
576 | |
577 return NULL; | |
578 } | |
579 | |
580 void collection_set_sort_method(CollectionData *cd, SortType method) | |
581 { | |
582 if (!cd) return; | |
583 | |
584 if (cd->sort_method == method) return; | |
585 | |
586 cd->sort_method = method; | |
587 cd->list = collection_list_sort(cd->list, cd->sort_method); | |
588 if (cd->list) cd->changed = TRUE; | |
589 | |
590 collection_window_refresh(collection_window_find(cd)); | |
591 } | |
592 | |
593 void collection_set_update_info_func(CollectionData *cd, | |
594 void (*func)(CollectionData *, CollectInfo *, gpointer), gpointer data) | |
595 { | |
596 cd->info_updated_func = func; | |
597 cd->info_updated_data = data; | |
598 } | |
599 | |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
600 static CollectInfo *collection_info_new_if_not_exists(CollectionData *cd, struct stat *st, FileData *fd) |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
601 { |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
602 CollectInfo *ci; |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
603 |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
604 if (g_hash_table_lookup(cd->existence, fd->path)) return NULL; |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
605 |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
606 ci = collection_info_new(fd, st, NULL); |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
607 if (ci) g_hash_table_insert(cd->existence, fd->path, ""); |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
608 return ci; |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
609 } |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
610 |
138 | 611 gint collection_add_check(CollectionData *cd, FileData *fd, gint sorted, gint must_exist) |
9 | 612 { |
613 struct stat st; | |
614 gint valid; | |
615 | |
616 if (must_exist) | |
617 { | |
138 | 618 valid = (stat_utf8(fd->path, &st) && !S_ISDIR(st.st_mode)); |
9 | 619 } |
620 else | |
621 { | |
622 valid = TRUE; | |
623 st.st_size = 0; | |
624 st.st_mtime = 0; | |
625 } | |
626 | |
627 if (valid) | |
628 { | |
629 CollectInfo *ci; | |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
630 |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
631 ci = collection_info_new_if_not_exists(cd, &st, fd); |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
632 if (!ci) return FALSE; |
506
fc9c8a3e1a8b
Handle the newline in DEBUG_N() macro instead of adding one
zas_
parents:
495
diff
changeset
|
633 DEBUG_3("add to collection: %s", fd->path); |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
634 |
9 | 635 cd->list = collection_list_add(cd->list, ci, sorted ? cd->sort_method : SORT_NONE); |
636 cd->changed = TRUE; | |
637 | |
638 if (!sorted || cd->sort_method == SORT_NONE) | |
639 { | |
640 collection_window_add(collection_window_find(cd), ci); | |
641 } | |
642 else | |
643 { | |
644 collection_window_insert(collection_window_find(cd), ci); | |
645 } | |
646 } | |
647 | |
648 return valid; | |
649 } | |
650 | |
138 | 651 gint collection_add(CollectionData *cd, FileData *fd, gint sorted) |
9 | 652 { |
138 | 653 return collection_add_check(cd, fd, sorted, TRUE); |
9 | 654 } |
655 | |
138 | 656 gint collection_insert(CollectionData *cd, FileData *fd, CollectInfo *insert_ci, gint sorted) |
9 | 657 { |
658 struct stat st; | |
659 | |
138 | 660 if (!insert_ci) return collection_add(cd, fd, sorted); |
9 | 661 |
138 | 662 if (stat_utf8(fd->path, &st) >= 0 && !S_ISDIR(st.st_mode)) |
9 | 663 { |
664 CollectInfo *ci; | |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
665 |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
666 ci = collection_info_new_if_not_exists(cd, &st, fd); |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
667 if (!ci) return FALSE; |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
668 |
506
fc9c8a3e1a8b
Handle the newline in DEBUG_N() macro instead of adding one
zas_
parents:
495
diff
changeset
|
669 DEBUG_3("insert in collection: %s", fd->path); |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
670 |
9 | 671 cd->list = collection_list_insert(cd->list, ci, insert_ci, sorted ? cd->sort_method : SORT_NONE); |
672 cd->changed = TRUE; | |
673 | |
674 collection_window_insert(collection_window_find(cd), ci); | |
675 | |
676 return TRUE; | |
677 } | |
678 | |
679 return FALSE; | |
680 } | |
681 | |
138 | 682 gint collection_remove(CollectionData *cd, FileData *fd) |
9 | 683 { |
684 CollectInfo *ci; | |
685 | |
799 | 686 ci = collection_list_find_fd(cd->list, fd); |
9 | 687 |
688 if (!ci) return FALSE; | |
689 | |
303
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
690 g_hash_table_remove(cd->existence, fd->path); |
6ff0bc50ac46
Prevent multiple insertion of the same file in a collection.
zas_
parents:
289
diff
changeset
|
691 |
9 | 692 cd->list = g_list_remove(cd->list, ci); |
693 cd->changed = TRUE; | |
694 | |
695 collection_window_remove(collection_window_find(cd), ci); | |
696 collection_info_free(ci); | |
697 | |
698 return TRUE; | |
699 } | |
700 | |
701 static void collection_remove_by_info(CollectionData *cd, CollectInfo *info) | |
702 { | |
703 if (!info || !g_list_find(cd->list, info)) return; | |
704 | |
705 cd->list = g_list_remove(cd->list, info); | |
706 cd->changed = (cd->list != NULL); | |
707 | |
708 collection_window_remove(collection_window_find(cd), info); | |
709 collection_info_free(info); | |
710 } | |
711 | |
712 void collection_remove_by_info_list(CollectionData *cd, GList *list) | |
713 { | |
714 GList *work; | |
715 | |
716 if (!list) return; | |
717 | |
718 if (!list->next) | |
719 { | |
720 /* more efficient (in collect-table) to remove a single item this way */ | |
721 collection_remove_by_info(cd, (CollectInfo *)list->data); | |
722 return; | |
723 } | |
724 | |
725 work = list; | |
516 | 726 while (work) |
9 | 727 { |
728 cd->list = collection_list_remove(cd->list, work->data); | |
729 work = work->next; | |
730 } | |
731 cd->changed = (cd->list != NULL); | |
732 | |
733 collection_window_refresh(collection_window_find(cd)); | |
734 } | |
735 | |
138 | 736 gint collection_rename(CollectionData *cd, FileData *fd) |
9 | 737 { |
738 CollectInfo *ci; | |
799 | 739 ci = collection_list_find_fd(cd->list, fd); |
9 | 740 |
741 if (!ci) return FALSE; | |
742 | |
743 cd->changed = TRUE; | |
744 | |
745 collection_window_update(collection_window_find(cd), ci); | |
746 | |
747 return TRUE; | |
748 } | |
749 | |
750 void collection_update_geometry(CollectionData *cd) | |
751 { | |
752 collection_window_get_geometry(collection_window_find(cd)); | |
753 } | |
754 | |
755 /* | |
756 *------------------------------------------------------------------- | |
757 * simple maintenance for renaming, deleting | |
758 *------------------------------------------------------------------- | |
759 */ | |
760 | |
799 | 761 static void collection_notify_cb(FileData *fd, NotifyType type, gpointer data) |
9 | 762 { |
799 | 763 CollectionData *cd = data; |
9 | 764 |
803 | 765 if (type != NOTIFY_TYPE_CHANGE || !fd->change) return; |
799 | 766 |
767 switch(fd->change->type) | |
9 | 768 { |
799 | 769 case FILEDATA_CHANGE_MOVE: |
770 case FILEDATA_CHANGE_RENAME: | |
771 collection_rename(cd, fd); | |
772 break; | |
773 case FILEDATA_CHANGE_COPY: | |
774 break; | |
775 case FILEDATA_CHANGE_DELETE: | |
776 while (collection_remove(cd, fd)); | |
777 break; | |
778 case FILEDATA_CHANGE_UNSPECIFIED: | |
779 break; | |
9 | 780 } |
799 | 781 |
9 | 782 } |
783 | |
784 | |
785 /* | |
786 *------------------------------------------------------------------- | |
787 * window key presses | |
788 *------------------------------------------------------------------- | |
789 */ | |
790 | |
791 static gint collection_window_keypress(GtkWidget *widget, GdkEventKey *event, gpointer data) | |
792 { | |
793 CollectWindow *cw = data; | |
794 gint stop_signal = FALSE; | |
795 gint edit_val = -1; | |
796 GList *list; | |
797 | |
798 if (event->state & GDK_CONTROL_MASK) | |
799 { | |
85
9d5c75b5ec28
Fri Oct 20 09:20:10 2006 John Ellis <johne@verizon.net>
gqview
parents:
9
diff
changeset
|
800 stop_signal = TRUE; |
9 | 801 switch (event->keyval) |
802 { | |
803 case '1': | |
804 edit_val = 0; | |
805 break; | |
806 case '2': | |
807 edit_val = 1; | |
808 break; | |
809 case '3': | |
810 edit_val = 2; | |
811 break; | |
812 case '4': | |
813 edit_val = 3; | |
814 break; | |
815 case '5': | |
816 edit_val = 4; | |
817 break; | |
818 case '6': | |
819 edit_val = 5; | |
820 break; | |
821 case '7': | |
822 edit_val = 6; | |
823 break; | |
824 case '8': | |
825 edit_val = 7; | |
826 break; | |
827 case '9': | |
828 edit_val = 8; | |
829 break; | |
830 case '0': | |
831 edit_val = 9; | |
832 break; | |
833 case 'A': case 'a': | |
834 if (event->state & GDK_SHIFT_MASK) | |
835 { | |
836 collection_table_unselect_all(cw->table); | |
837 } | |
838 else | |
839 { | |
840 collection_table_select_all(cw->table); | |
841 } | |
842 break; | |
843 case 'L': case 'l': | |
844 list = layout_list(NULL); | |
845 if (list) | |
846 { | |
138 | 847 collection_table_add_filelist(cw->table, list); |
848 filelist_free(list); | |
9 | 849 } |
850 break; | |
851 case 'C': case 'c': | |
852 file_util_copy(NULL, collection_table_selection_get_list(cw->table), NULL, cw->window); | |
853 break; | |
854 case 'M': case 'm': | |
855 file_util_move(NULL, collection_table_selection_get_list(cw->table), NULL, cw->window); | |
856 break; | |
857 case 'R': case 'r': | |
858 file_util_rename(NULL, collection_table_selection_get_list(cw->table), cw->window); | |
859 break; | |
860 case 'D': case 'd': | |
861 file_util_delete(NULL, collection_table_selection_get_list(cw->table), cw->window); | |
862 break; | |
863 case 'P': case 'p': | |
479
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
864 info_window_new(NULL, collection_table_selection_get_list(cw->table), NULL); |
9 | 865 break; |
866 case 'S': case 's': | |
867 collection_dialog_save_as(NULL, cw->cd); | |
868 break; | |
869 case 'W': case 'w': | |
870 collection_window_close(cw); | |
871 break; | |
872 default: | |
85
9d5c75b5ec28
Fri Oct 20 09:20:10 2006 John Ellis <johne@verizon.net>
gqview
parents:
9
diff
changeset
|
873 stop_signal = FALSE; |
9 | 874 break; |
875 } | |
876 } | |
877 else | |
878 { | |
85
9d5c75b5ec28
Fri Oct 20 09:20:10 2006 John Ellis <johne@verizon.net>
gqview
parents:
9
diff
changeset
|
879 stop_signal = TRUE; |
9 | 880 switch (event->keyval) |
881 { | |
882 case GDK_Return: case GDK_KP_Enter: | |
883 layout_image_set_collection(NULL, cw->cd, | |
884 collection_table_get_focus_info(cw->table)); | |
885 break; | |
886 case 'V': case 'v': | |
887 view_window_new_from_collection(cw->cd, | |
888 collection_table_get_focus_info(cw->table)); | |
889 break; | |
890 case 'S': case 's': | |
891 if (!cw->cd->path) | |
892 { | |
893 collection_dialog_save_as(NULL, cw->cd); | |
442 | 894 } |
9 | 895 else if (!collection_save(cw->cd, cw->cd->path)) |
896 { | |
673
fbebf5cf4a55
Do not use printf() directly but use new wrapper function log_printf() instead.
zas_
parents:
671
diff
changeset
|
897 log_printf("failed saving to collection path: %s\n", cw->cd->path); |
9 | 898 } |
899 break; | |
900 case 'A': case 'a': | |
85
9d5c75b5ec28
Fri Oct 20 09:20:10 2006 John Ellis <johne@verizon.net>
gqview
parents:
9
diff
changeset
|
901 collection_dialog_append(NULL, cw->cd); |
9 | 902 break; |
903 case 'N': case 'n': | |
904 collection_set_sort_method(cw->cd, SORT_NAME); | |
905 break; | |
906 #ifdef HAVE_STRVERSCMP | |
907 case 'I': case 'i': | |
908 collection_set_sort_method(cw->cd, SORT_NUMBER); | |
909 break; | |
910 #endif | |
911 case 'D': case 'd': | |
912 collection_set_sort_method(cw->cd, SORT_TIME); | |
913 break; | |
914 case 'B': case 'b': | |
915 collection_set_sort_method(cw->cd, SORT_SIZE); | |
916 break; | |
917 case 'P': case 'p': | |
94
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
918 if (event->state & GDK_SHIFT_MASK) |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
919 { |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
920 CollectInfo *info; |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
921 |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
922 info = collection_table_get_focus_info(cw->table); |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
923 |
138 | 924 print_window_new(info->fd, collection_table_selection_get_list(cw->table), |
925 collection_list_to_filelist(cw->cd->list), cw->window); | |
94
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
926 } |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
927 else |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
928 { |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
929 collection_set_sort_method(cw->cd, SORT_PATH); |
50dc5a14d37b
Thu Nov 2 17:51:31 2006 John Ellis <johne@verizon.net>
gqview
parents:
91
diff
changeset
|
930 } |
9 | 931 break; |
932 case GDK_Delete: case GDK_KP_Delete: | |
933 list = g_list_copy(cw->table->selection); | |
934 if (list) | |
935 { | |
936 collection_remove_by_info_list(cw->cd, list); | |
937 g_list_free(list); | |
938 } | |
939 else | |
940 { | |
941 collection_remove_by_info(cw->cd, collection_table_get_focus_info(cw->table)); | |
942 } | |
943 break; | |
944 default: | |
85
9d5c75b5ec28
Fri Oct 20 09:20:10 2006 John Ellis <johne@verizon.net>
gqview
parents:
9
diff
changeset
|
945 stop_signal = FALSE; |
9 | 946 break; |
947 } | |
948 } | |
949 | |
950 if (edit_val != -1) | |
951 { | |
952 list = collection_table_selection_get_list(cw->table); | |
753 | 953 file_util_start_editor_from_filelist(edit_val, list, cw->window); |
138 | 954 filelist_free(list); |
9 | 955 } |
956 | |
957 return stop_signal; | |
958 } | |
959 | |
960 /* | |
961 *------------------------------------------------------------------- | |
962 * window | |
963 *------------------------------------------------------------------- | |
964 */ | |
965 static void collection_window_get_geometry(CollectWindow *cw) | |
966 { | |
967 CollectionData *cd; | |
968 | |
969 if (!cw) return; | |
970 | |
971 cd = cw->cd; | |
512
f9bf33be53ff
Remove whitespace between function name and first parenthesis for the sake of consistency.
zas_
parents:
507
diff
changeset
|
972 gdk_window_get_position(cw->window->window, &cd->window_x, &cd->window_y); |
9 | 973 gdk_drawable_get_size(cw->window->window, &cd->window_w, &cd->window_h); |
974 cd->window_read = TRUE; | |
975 } | |
976 | |
977 static void collection_window_refresh(CollectWindow *cw) | |
978 { | |
979 if (!cw) return; | |
980 | |
981 collection_table_refresh(cw->table); | |
982 } | |
983 | |
984 static void collection_window_update_title(CollectWindow *cw) | |
985 { | |
1176
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
986 gboolean free_name = FALSE; |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
987 gchar *name; |
9 | 988 gchar *buf; |
989 | |
990 if (!cw) return; | |
991 | |
1176
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
992 if (file_extension_match(cw->cd->name, GQ_COLLECTION_EXT)) |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
993 { |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
994 name = remove_extension_from_path(cw->cd->name); |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
995 free_name = TRUE; |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
996 } |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
997 else |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
998 { |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
999 name = cw->cd->name; |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
1000 } |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
1001 |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
1002 buf = g_strdup_printf(_("%s - Collection - %s"), name, GQ_APPNAME); |
d3e51e1a02f8
Hide .gqv extension from collection name in collection window title and recent menu list.
zas_
parents:
1174
diff
changeset
|
1003 if (free_name) g_free(name); |
9 | 1004 gtk_window_set_title(GTK_WINDOW(cw->window), buf); |
1005 g_free(buf); | |
1006 } | |
1007 | |
1008 static void collection_window_update_info(CollectionData *cd, CollectInfo *ci, gpointer data) | |
1009 { | |
1010 CollectWindow *cw = data; | |
1011 | |
1012 collection_table_file_update(cw->table, ci); | |
1013 } | |
1014 | |
1015 static void collection_window_add(CollectWindow *cw, CollectInfo *ci) | |
1016 { | |
1017 if (!cw) return; | |
1018 | |
1019 if (!ci->pixbuf) collection_load_thumb_idle(cw->cd); | |
1020 collection_table_file_add(cw->table, ci); | |
1021 } | |
1022 | |
1023 static void collection_window_insert(CollectWindow *cw, CollectInfo *ci) | |
1024 { | |
1025 if (!cw) return; | |
1026 | |
1027 if (!ci->pixbuf) collection_load_thumb_idle(cw->cd); | |
1028 collection_table_file_insert(cw->table, ci); | |
1029 if (!cw) return; | |
1030 } | |
1031 | |
1032 #if 0 | |
1033 static void collection_window_move(CollectWindow *cw, CollectInfo *ci) | |
1034 { | |
1035 if (!cw) return; | |
1036 } | |
1037 #endif | |
1038 | |
1039 static void collection_window_remove(CollectWindow *cw, CollectInfo *ci) | |
1040 { | |
1041 if (!cw) return; | |
1042 | |
1043 collection_table_file_remove(cw->table, ci); | |
1044 } | |
1045 | |
1046 static void collection_window_update(CollectWindow *cw, CollectInfo *ci) | |
1047 { | |
1048 if (!cw) return; | |
1049 | |
1050 collection_table_file_update(cw->table, ci); | |
1051 collection_table_file_update(cw->table, NULL); | |
1052 } | |
1053 | |
1054 static void collection_window_close_final(CollectWindow *cw) | |
1055 { | |
1056 if (cw->close_dialog) return; | |
1057 | |
1058 collection_window_list = g_list_remove(collection_window_list, cw); | |
1059 collection_window_get_geometry(cw); | |
1060 | |
1061 gtk_widget_destroy(cw->window); | |
1062 | |
1063 collection_set_update_info_func(cw->cd, NULL, NULL); | |
1064 collection_unref(cw->cd); | |
1065 | |
1066 g_free(cw); | |
1067 } | |
1068 | |
1069 static void collection_close_save_cb(GenericDialog *gd, gpointer data) | |
1070 { | |
1071 CollectWindow *cw = data; | |
1072 | |
1073 cw->close_dialog = NULL; | |
1074 generic_dialog_close(gd); | |
1075 | |
1076 if (!cw->cd->path) | |
442 | 1077 { |
9 | 1078 collection_dialog_save_close(NULL, cw->cd); |
1079 return; | |
1080 } | |
1081 else if (!collection_save(cw->cd, cw->cd->path)) | |
1082 { | |
1083 gchar *buf; | |
1084 buf = g_strdup_printf(_("Failed to save the collection:\n%s"), cw->cd->path); | |
1085 warning_dialog(_("Save Failed"), buf, GTK_STOCK_DIALOG_ERROR, cw->window); | |
1086 g_free(buf); | |
1087 return; | |
1088 } | |
1089 | |
1090 collection_window_close_final(cw); | |
1091 } | |
1092 | |
1093 static void collection_close_close_cb(GenericDialog *gd, gpointer data) | |
1094 { | |
1095 CollectWindow *cw = data; | |
1096 | |
1097 cw->close_dialog = NULL; | |
1098 generic_dialog_close(gd); | |
1099 | |
1100 collection_window_close_final(cw); | |
1101 } | |
1102 | |
1103 static void collection_close_cancel_cb(GenericDialog *gd, gpointer data) | |
1104 { | |
1105 CollectWindow *cw = data; | |
1106 | |
1107 cw->close_dialog = NULL; | |
1108 generic_dialog_close(gd); | |
1109 } | |
1110 | |
1111 static void collection_close_dlg_show(CollectWindow *cw) | |
1112 { | |
1113 GenericDialog *gd; | |
1114 | |
1115 if (cw->close_dialog) | |
1116 { | |
1117 gtk_window_present(GTK_WINDOW(cw->close_dialog)); | |
1118 return; | |
1119 } | |
1120 | |
1121 gd = generic_dialog_new(_("Close collection"), | |
1174
0bea79d87065
Drop useless wmclass stuff. Gtk will take care of it and as said in the documentation using gtk_window_set_wmclass() is sort of pointless.
zas_
parents:
1055
diff
changeset
|
1122 "close_collection", cw->window, FALSE, |
9 | 1123 collection_close_cancel_cb, cw); |
1124 generic_dialog_add_message(gd, GTK_STOCK_DIALOG_QUESTION, | |
1125 _("Close collection"), | |
1126 _("Collection has been modified.\nSave first?")); | |
1127 | |
1128 generic_dialog_add_button(gd, GTK_STOCK_SAVE, NULL, collection_close_save_cb, TRUE); | |
1129 generic_dialog_add_button(gd, GTK_STOCK_DELETE, _("_Discard"), collection_close_close_cb, FALSE); | |
1130 | |
1131 cw->close_dialog = gd->dialog; | |
1132 | |
1133 gtk_widget_show(gd->dialog); | |
1134 } | |
1135 | |
1136 static void collection_window_close(CollectWindow *cw) | |
1137 { | |
1138 if (!cw->cd->changed && !cw->close_dialog) | |
1139 { | |
1140 collection_window_close_final(cw); | |
1141 return; | |
1142 } | |
1143 | |
1144 collection_close_dlg_show(cw); | |
1145 } | |
1146 | |
1147 void collection_window_close_by_collection(CollectionData *cd) | |
1148 { | |
1149 CollectWindow *cw; | |
1150 | |
1151 cw = collection_window_find(cd); | |
1152 if (cw) collection_window_close_final(cw); | |
1153 } | |
1154 | |
1155 gint collection_window_modified_exists(void) | |
1156 { | |
1157 GList *work; | |
1158 | |
1159 work = collection_window_list; | |
1160 while (work) | |
1161 { | |
1162 CollectWindow *cw = work->data; | |
1163 if (cw->cd->changed) return TRUE; | |
1164 work = work->next; | |
1165 } | |
1166 | |
1167 return FALSE; | |
1168 } | |
1169 | |
1170 static gint collection_window_delete(GtkWidget *widget, GdkEvent *event, gpointer data) | |
1171 { | |
1172 CollectWindow *cw = data; | |
1173 collection_window_close(cw); | |
1174 | |
1175 return TRUE; | |
1176 } | |
1177 | |
1178 CollectWindow *collection_window_new(const gchar *path) | |
1179 { | |
1180 CollectWindow *cw; | |
1181 GtkWidget *vbox; | |
1182 GtkWidget *frame; | |
1183 GtkWidget *status_label; | |
1184 GtkWidget *extra_label; | |
1185 GdkGeometry geometry; | |
1186 | |
1187 cw = g_new0(CollectWindow, 1); | |
1188 | |
1189 cw->close_dialog = NULL; | |
1190 | |
1191 collection_window_list = g_list_append(collection_window_list, cw); | |
1192 | |
1193 cw->cd = collection_new(path); | |
1194 | |
289
6a7298988a7a
Simplify and unify gtk_window creation with the help of
zas_
parents:
288
diff
changeset
|
1195 cw->window = window_new(GTK_WINDOW_TOPLEVEL, "collection", PIXBUF_INLINE_ICON_BOOK, NULL, NULL); |
9 | 1196 |
1029
1a4b18c58556
Use a constant for minimal window size. Set it to 32 for all dialogs.
zas_
parents:
1022
diff
changeset
|
1197 geometry.min_width = DEFAULT_MINIMAL_WINDOW_SIZE; |
1a4b18c58556
Use a constant for minimal window size. Set it to 32 for all dialogs.
zas_
parents:
1022
diff
changeset
|
1198 geometry.min_height = DEFAULT_MINIMAL_WINDOW_SIZE; |
9 | 1199 geometry.base_width = COLLECT_DEF_WIDTH; |
1200 geometry.base_height = COLLECT_DEF_HEIGHT; | |
1201 gtk_window_set_geometry_hints(GTK_WINDOW(cw->window), NULL, &geometry, | |
1202 GDK_HINT_MIN_SIZE | GDK_HINT_BASE_SIZE); | |
1203 | |
1204 | |
338
41c3cb73120f
Rename window options (moved to layout) and re-order rc file.
zas_
parents:
318
diff
changeset
|
1205 if (options->layout.save_window_positions && path && collection_load_only_geometry(cw->cd, path)) |
9 | 1206 { |
1207 /* FIXME: x, y is not implemented */ | |
1208 gtk_window_set_default_size(GTK_WINDOW(cw->window), cw->cd->window_w, cw->cd->window_h); | |
1209 } | |
1210 else | |
1211 { | |
1212 gtk_window_set_default_size(GTK_WINDOW(cw->window), COLLECT_DEF_WIDTH, COLLECT_DEF_HEIGHT); | |
1213 } | |
1214 | |
1215 gtk_window_set_resizable(GTK_WINDOW(cw->window), TRUE); | |
1216 collection_window_update_title(cw); | |
513
985fdfebd89e
Remove whitespace between function name and first parenthesis for the sake of consistency. (pass 2)
zas_
parents:
512
diff
changeset
|
1217 gtk_container_set_border_width(GTK_CONTAINER(cw->window), 0); |
9 | 1218 |
1219 g_signal_connect(G_OBJECT(cw->window), "delete_event", | |
1220 G_CALLBACK(collection_window_delete), cw); | |
1221 | |
513
985fdfebd89e
Remove whitespace between function name and first parenthesis for the sake of consistency. (pass 2)
zas_
parents:
512
diff
changeset
|
1222 g_signal_connect(G_OBJECT(cw->window), "key_press_event", |
9 | 1223 G_CALLBACK(collection_window_keypress), cw); |
1224 | |
1225 vbox = gtk_vbox_new(FALSE, 0); | |
1226 gtk_container_add(GTK_CONTAINER(cw->window), vbox); | |
1227 gtk_widget_show(vbox); | |
1228 | |
1229 cw->table = collection_table_new(cw->cd); | |
1230 gtk_box_pack_start(GTK_BOX(vbox), cw->table->scrolled, TRUE, TRUE, 0); | |
1231 gtk_widget_show(cw->table->scrolled); | |
1232 | |
1233 cw->status_box = gtk_hbox_new(TRUE, 0); | |
1234 gtk_box_pack_start(GTK_BOX(vbox), cw->status_box, FALSE, FALSE, 0); | |
1235 gtk_widget_show(cw->status_box); | |
1236 | |
1237 frame = gtk_frame_new(NULL); | |
1238 gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_IN); | |
1239 gtk_box_pack_start(GTK_BOX(cw->status_box), frame, TRUE, TRUE, 0); | |
1240 gtk_widget_show(frame); | |
1241 | |
1242 status_label = gtk_label_new(""); | |
1243 gtk_container_add(GTK_CONTAINER(frame), status_label); | |
1244 gtk_widget_show(status_label); | |
1245 | |
1246 extra_label = gtk_progress_bar_new(); | |
1247 gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(extra_label), 0.0); | |
1248 gtk_box_pack_start(GTK_BOX(cw->status_box), extra_label, TRUE, TRUE, 0); | |
1249 gtk_widget_show(extra_label); | |
1250 | |
1251 collection_table_set_labels(cw->table, status_label, extra_label); | |
1252 | |
1253 gtk_widget_show(cw->window); | |
1254 gtk_widget_grab_focus(cw->table->listview); | |
1255 | |
1256 collection_set_update_info_func(cw->cd, collection_window_update_info, cw); | |
1257 | |
726 | 1258 if (path && *path == G_DIR_SEPARATOR) collection_load_begin(cw->cd, NULL, COLLECTION_LOAD_NONE); |
9 | 1259 |
1260 return cw; | |
1261 } | |
1055
1646720364cf
Adding a vim modeline to all files - patch by Klaus Ethgen
nadvornik
parents:
1029
diff
changeset
|
1262 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */ |