Mercurial > geeqie
annotate src/info.c @ 847:77fc0ea3457d
measure pixbuf cache size in bytes
author | nadvornik |
---|---|
date | Sat, 21 Jun 2008 19:26:01 +0000 |
parents | 339db85846da |
children | db6977f8b0f1 |
rev | line source |
---|---|
9 | 1 /* |
196 | 2 * Geeqie |
9 | 3 * (C) 2004 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 "info.h" |
16 | |
17 #include "bar_info.h" | |
18 #include "bar_exif.h" | |
19 #include "dnd.h" | |
586 | 20 #include "filedata.h" |
9 | 21 #include "image.h" |
22 #include "image-load.h" | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
23 #include "pixbuf-renderer.h" |
9 | 24 #include "ui_bookmark.h" |
25 #include "ui_fileops.h" | |
26 #include "ui_misc.h" | |
648
e34c1002e553
Move some functions from main.[ch] to new window.[ch].
zas_
parents:
586
diff
changeset
|
27 #include "window.h" |
9 | 28 |
29 #include <pwd.h> | |
30 #include <grp.h> | |
31 | |
32 | |
33 #define IMAGE_SIZE_W 200 | |
34 #define IMAGE_SIZE_H 200 | |
35 | |
36 | |
37 typedef struct _TabData TabData; | |
38 struct _TabData | |
39 { | |
40 void (*func_free)(gpointer data); | |
41 void (*func_sync)(InfoData *id, gpointer data); | |
42 void (*func_image)(InfoData *id, gpointer data); | |
43 gpointer data; | |
221 | 44 TabData *(*func_new)(InfoData *id); |
45 GtkWidget *child; | |
9 | 46 }; |
47 | |
221 | 48 typedef struct _InfoTabsPos InfoTabsPos; |
49 struct _InfoTabsPos | |
50 { | |
51 TabData *(*func)(InfoData *id); | |
736 | 52 guint pos; |
221 | 53 }; |
54 | |
55 static GList *info_tabs_pos_list = NULL; | |
9 | 56 |
271
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
57 static void notebook_set_tab_reorderable(GtkNotebook *notebook, GtkWidget *child, gboolean reorderable) |
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
58 { |
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
59 #if GTK_CHECK_VERSION(2,10,0) |
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
60 gtk_notebook_set_tab_reorderable(notebook, child, reorderable); |
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
61 #endif |
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
62 } |
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
63 |
9 | 64 /* |
65 *------------------------------------------------------------------- | |
66 * table utils | |
67 *------------------------------------------------------------------- | |
68 */ | |
69 | |
70 GtkWidget *table_add_line(GtkWidget *table, gint x, gint y, | |
71 const gchar *description, const gchar *text) | |
72 { | |
73 GtkWidget *label; | |
74 | |
75 if (!text) text = ""; | |
76 | |
77 label = pref_table_label(table, x, y, description, 1.0); | |
78 pref_label_bold(label, TRUE, FALSE); | |
79 | |
80 label = pref_table_label(table, x + 1, y, text, 0.0); | |
81 return label; | |
82 } | |
83 | |
84 /* | |
85 *------------------------------------------------------------------- | |
86 * EXIF tab | |
87 *------------------------------------------------------------------- | |
88 */ | |
89 | |
90 static void info_tab_exif_image(InfoData *id, gpointer data) | |
91 { | |
92 GtkWidget *bar = data; | |
138 | 93 FileData *fd; |
9 | 94 |
95 if (id->image->unknown) | |
96 { | |
138 | 97 fd = NULL; |
9 | 98 } |
99 else | |
100 { | |
138 | 101 fd = id->image->image_fd; |
9 | 102 } |
103 | |
138 | 104 bar_exif_set(bar, fd); |
9 | 105 } |
106 | |
107 static void info_tab_exif_sync(InfoData *id, gpointer data) | |
108 { | |
109 GtkWidget *bar = data; | |
110 | |
111 bar_exif_set(bar, NULL); | |
112 } | |
113 | |
114 static TabData *info_tab_exif_new(InfoData *id) | |
115 { | |
116 TabData *td; | |
117 GtkWidget *bar; | |
118 GtkWidget *label; | |
119 | |
120 bar = bar_exif_new(FALSE, NULL, FALSE, NULL); | |
121 gtk_container_set_border_width(GTK_CONTAINER(bar), PREF_PAD_BORDER); | |
122 | |
123 label = gtk_label_new(_("Exif")); | |
124 gtk_notebook_append_page(GTK_NOTEBOOK(id->notebook), bar, label); | |
271
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
125 notebook_set_tab_reorderable(GTK_NOTEBOOK(id->notebook), bar, TRUE); |
9 | 126 gtk_widget_show(bar); |
127 | |
128 /* register */ | |
129 td = g_new0(TabData, 1); | |
130 td->func_free = NULL; | |
131 td->func_sync = info_tab_exif_sync; | |
132 td->func_image = info_tab_exif_image; | |
133 td->data = bar; | |
221 | 134 td->func_new = info_tab_exif_new; |
135 td->child = bar; | |
9 | 136 |
137 return td; | |
138 } | |
139 | |
140 /* | |
141 *------------------------------------------------------------------- | |
142 * file attributes tab | |
143 *------------------------------------------------------------------- | |
144 */ | |
145 | |
146 typedef struct _InfoTabMeta InfoTabMeta; | |
147 struct _InfoTabMeta | |
148 { | |
149 GtkWidget *bar_info; | |
150 }; | |
151 | |
152 static void info_tab_meta_free(gpointer data) | |
153 { | |
154 InfoTabMeta *tab = data; | |
155 | |
156 g_free(tab); | |
157 } | |
158 | |
159 static void info_tab_meta_sync(InfoData *id, gpointer data) | |
160 { | |
161 InfoTabMeta *tab = data; | |
162 | |
138 | 163 bar_info_set(tab->bar_info, id->fd); |
9 | 164 } |
165 | |
166 static GList *info_tab_meta_list_cb(gpointer data) | |
167 { | |
168 InfoData *id = data; | |
169 | |
576
9dc0513837b5
dropped path_list functions, use filelist functions everywhere
nadvornik
parents:
551
diff
changeset
|
170 return filelist_copy(id->list); |
9 | 171 } |
172 | |
173 static TabData *info_tab_meta_new(InfoData *id) | |
174 { | |
175 TabData *td; | |
176 InfoTabMeta *tab; | |
177 GtkWidget *label; | |
178 | |
179 tab = g_new0(InfoTabMeta, 1); | |
180 | |
181 tab->bar_info = bar_info_new(NULL, TRUE, NULL); | |
182 bar_info_set_selection_func(tab->bar_info, info_tab_meta_list_cb, id); | |
183 bar_info_selection(tab->bar_info, g_list_length(id->list) - 1); | |
184 | |
185 gtk_container_set_border_width(GTK_CONTAINER(tab->bar_info), PREF_PAD_BORDER); | |
186 | |
187 label = gtk_label_new(_("Keywords")); | |
188 gtk_notebook_append_page(GTK_NOTEBOOK(id->notebook), tab->bar_info, label); | |
271
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
189 notebook_set_tab_reorderable(GTK_NOTEBOOK(id->notebook), tab->bar_info, TRUE); |
9 | 190 gtk_widget_show(tab->bar_info); |
191 | |
192 /* register */ | |
193 td = g_new0(TabData, 1); | |
194 td->func_free = info_tab_meta_free; | |
195 td->func_sync = info_tab_meta_sync; | |
196 td->func_image = NULL; | |
197 td->data = tab; | |
221 | 198 td->func_new = info_tab_meta_new; |
199 td->child = tab->bar_info; | |
9 | 200 |
201 return td; | |
202 } | |
203 | |
204 /* | |
205 *------------------------------------------------------------------- | |
206 * general tab | |
207 *------------------------------------------------------------------- | |
208 */ | |
209 | |
210 typedef struct _InfoTabGeneral InfoTabGeneral; | |
211 struct _InfoTabGeneral | |
212 { | |
213 GtkWidget *label_file_time; | |
214 GtkWidget *label_file_size; | |
215 GtkWidget *label_dimensions; | |
216 GtkWidget *label_transparent; | |
217 GtkWidget *label_image_size; | |
218 GtkWidget *label_compression; | |
219 GtkWidget *label_mime_type; | |
220 | |
221 GtkWidget *label_user; | |
222 GtkWidget *label_group; | |
223 GtkWidget *label_perms; | |
224 | |
225 gint compression_done; | |
226 gint64 byte_size; | |
227 }; | |
228 | |
229 static void info_tab_general_image(InfoData *id, gpointer data) | |
230 { | |
231 InfoTabGeneral *tab = data; | |
232 gchar *buf; | |
233 guint mem_size; | |
234 gint has_alpha; | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
235 GdkPixbuf *pixbuf; |
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
236 gint width, height; |
9 | 237 |
238 if (id->image->unknown) return; | |
239 | |
531 | 240 image_get_image_size(id->image, &width, &height); |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
241 |
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
242 buf = g_strdup_printf("%d x %d", width, height); |
9 | 243 gtk_label_set_text(GTK_LABEL(tab->label_dimensions), buf); |
244 g_free(buf); | |
245 | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
246 pixbuf = image_get_pixbuf(id->image); |
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
247 if (pixbuf) |
9 | 248 { |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
249 has_alpha = gdk_pixbuf_get_has_alpha(pixbuf); |
9 | 250 } |
251 else | |
252 { | |
253 has_alpha = FALSE; | |
254 } | |
255 gtk_label_set_text(GTK_LABEL(tab->label_transparent), has_alpha ? _("yes") : _("no")); | |
256 | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
257 mem_size = width * height * ((has_alpha) ? 4 : 3); |
9 | 258 buf = text_from_size_abrev(mem_size); |
259 gtk_label_set_text(GTK_LABEL(tab->label_image_size), buf); | |
260 g_free(buf); | |
261 | |
262 if (!tab->compression_done && mem_size > 0) | |
263 { | |
15
3263965d5f9e
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
9
diff
changeset
|
264 buf = g_strdup_printf("%.1f%%", (double)tab->byte_size / mem_size * 100.0); |
9 | 265 gtk_label_set_text(GTK_LABEL(tab->label_compression), buf); |
266 g_free(buf); | |
267 | |
268 tab->compression_done = TRUE; | |
269 } | |
270 | |
271 buf = image_loader_get_format(id->image->il); | |
272 if (buf) | |
273 gtk_label_set_text(GTK_LABEL(tab->label_mime_type), buf); | |
274 g_free(buf); | |
275 } | |
276 | |
277 static gchar *mode_number(mode_t m) | |
278 { | |
279 int mb, mu, mg, mo; | |
280 | |
281 mb = mu = mg = mo = 0; | |
282 | |
283 if (m & S_ISUID) mb |= 4; | |
284 if (m & S_ISGID) mb |= 2; | |
285 if (m & S_ISVTX) mb |= 1; | |
286 | |
287 if (m & S_IRUSR) mu |= 4; | |
288 if (m & S_IWUSR) mu |= 2; | |
289 if (m & S_IXUSR) mu |= 1; | |
290 | |
291 if (m & S_IRGRP) mg |= 4; | |
292 if (m & S_IWGRP) mg |= 2; | |
293 if (m & S_IXGRP) mg |= 1; | |
294 | |
295 if (m & S_IROTH) mo |= 4; | |
296 if (m & S_IWOTH) mo |= 2; | |
297 if (m & S_IXOTH) mo |= 1; | |
298 | |
299 return g_strdup_printf("%d%d%d%d", mb, mu, mg, mo); | |
300 } | |
301 | |
302 static void info_tab_general_sync_perm(InfoTabGeneral *tab, InfoData *id) | |
303 { | |
304 struct stat st; | |
305 | |
138 | 306 if (!stat_utf8(id->fd->path, &st)) |
9 | 307 { |
308 gtk_label_set_text(GTK_LABEL(tab->label_user), ""); | |
309 gtk_label_set_text(GTK_LABEL(tab->label_group), ""); | |
310 gtk_label_set_text(GTK_LABEL(tab->label_perms), ""); | |
311 } | |
312 else | |
313 { | |
314 struct passwd *user; | |
315 struct group *grp; | |
316 gchar pbuf[12]; | |
317 gchar *pmod; | |
318 gchar *buf; | |
319 | |
320 user = getpwuid(st.st_uid); | |
321 gtk_label_set_text(GTK_LABEL(tab->label_user), (user) ? user->pw_name : ""); | |
322 | |
323 grp = getgrgid(st.st_gid); | |
324 gtk_label_set_text(GTK_LABEL(tab->label_group), (grp) ? grp->gr_name : ""); | |
325 | |
326 pbuf[0] = (st.st_mode & S_IRUSR) ? 'r' : '-'; | |
327 pbuf[1] = (st.st_mode & S_IWUSR) ? 'w' : '-'; | |
328 pbuf[2] = (st.st_mode & S_IXUSR) ? 'x' : '-'; | |
329 pbuf[3] = (st.st_mode & S_IRGRP) ? 'r' : '-'; | |
330 pbuf[4] = (st.st_mode & S_IWGRP) ? 'w' : '-'; | |
331 pbuf[5] = (st.st_mode & S_IXGRP) ? 'x' : '-'; | |
332 pbuf[6] = (st.st_mode & S_IROTH) ? 'r' : '-'; | |
333 pbuf[7] = (st.st_mode & S_IWOTH) ? 'w' : '-'; | |
334 pbuf[8] = (st.st_mode & S_IXOTH) ? 'x' : '-'; | |
335 pbuf[9] = '\0'; | |
336 | |
337 pmod = mode_number(st.st_mode); | |
338 buf = g_strdup_printf("%s (%s)", pbuf, pmod); | |
339 gtk_label_set_text(GTK_LABEL(tab->label_perms), buf); | |
340 g_free(buf); | |
341 g_free(pmod); | |
342 } | |
343 } | |
344 | |
345 static void info_tab_general_sync(InfoData *id, gpointer data) | |
346 { | |
347 InfoTabGeneral *tab = data; | |
348 gchar *buf; | |
349 | |
138 | 350 gtk_label_set_text(GTK_LABEL(tab->label_file_time), text_from_time(id->fd->date)); |
9 | 351 |
138 | 352 tab->byte_size = id->fd->size; |
9 | 353 |
354 buf = text_from_size(tab->byte_size); | |
355 gtk_label_set_text(GTK_LABEL(tab->label_file_size), buf); | |
356 g_free(buf); | |
357 | |
358 gtk_label_set_text(GTK_LABEL(tab->label_dimensions), ""); | |
359 gtk_label_set_text(GTK_LABEL(tab->label_transparent), ""); | |
360 gtk_label_set_text(GTK_LABEL(tab->label_image_size), ""); | |
361 | |
362 gtk_label_set_text(GTK_LABEL(tab->label_compression), ""); | |
363 gtk_label_set_text(GTK_LABEL(tab->label_mime_type), ""); | |
364 | |
365 info_tab_general_sync_perm(tab, id); | |
366 | |
367 tab->compression_done = FALSE; | |
368 } | |
369 | |
370 static void info_tab_general_free(gpointer data) | |
371 { | |
372 InfoTabGeneral *tab = data; | |
373 | |
374 g_free(tab); | |
375 } | |
376 | |
377 static TabData *info_tab_general_new(InfoData *id) | |
378 { | |
379 TabData *td; | |
380 InfoTabGeneral *tab; | |
381 GtkWidget *table; | |
382 GtkWidget *label; | |
383 | |
384 tab = g_new0(InfoTabGeneral, 1); | |
385 | |
386 table = pref_table_new(NULL, 2, 11, FALSE, FALSE); | |
387 gtk_container_set_border_width(GTK_CONTAINER(table), PREF_PAD_BORDER); | |
388 | |
389 tab->label_file_time = table_add_line(table, 0, 0, _("File date:"), NULL); | |
390 tab->label_file_size = table_add_line(table, 0, 1, _("File size:"), NULL); | |
391 | |
392 tab->label_dimensions = table_add_line(table, 0, 2, _("Dimensions:"), NULL); | |
393 tab->label_transparent = table_add_line(table, 0, 3, _("Transparent:"), NULL); | |
394 tab->label_image_size = table_add_line(table, 0, 4, _("Image size:"), NULL); | |
395 | |
396 tab->label_compression = table_add_line(table, 0, 5, _("Compress ratio:"), NULL); | |
397 tab->label_mime_type = table_add_line(table, 0, 6, _("File type:"), NULL); | |
398 | |
399 tab->label_user = table_add_line(table, 0, 7, _("Owner:"), NULL); | |
400 tab->label_group = table_add_line(table, 0, 8, _("Group:"), NULL); | |
401 tab->label_perms = table_add_line(table, 0, 9, "", NULL); | |
402 | |
403 label = gtk_label_new(_("General")); | |
404 gtk_notebook_append_page(GTK_NOTEBOOK(id->notebook), table, label); | |
271
35f9083c6b95
gtk_notebook_set_tab_reorderable() is only available sinc gtk+ 2.10.
zas_
parents:
254
diff
changeset
|
405 notebook_set_tab_reorderable(GTK_NOTEBOOK(id->notebook), table, TRUE); |
9 | 406 gtk_widget_show(table); |
407 | |
408 /* register */ | |
409 td = g_new0(TabData, 1); | |
410 td->func_free = info_tab_general_free; | |
411 td->func_sync = info_tab_general_sync; | |
412 td->func_image = info_tab_general_image; | |
413 td->data = tab; | |
221 | 414 td->func_new = info_tab_general_new; |
415 td->child = table; | |
9 | 416 |
417 return td; | |
418 } | |
419 | |
420 /* | |
421 *------------------------------------------------------------------- | |
422 * tabs | |
423 *------------------------------------------------------------------- | |
424 */ | |
425 | |
426 static void info_tabs_sync(InfoData *id, gint image) | |
427 { | |
428 GList *work; | |
429 | |
430 work = id->tab_list; | |
431 while (work) | |
432 { | |
433 TabData *td = work->data; | |
434 work = work->next; | |
435 | |
436 if (image) | |
437 { | |
438 if (td->func_image) td->func_image(id, td->data); | |
439 } | |
440 else | |
441 { | |
442 if (td->func_sync) td->func_sync(id, td->data); | |
443 } | |
444 } | |
445 } | |
446 | |
447 static void info_tabs_free(InfoData *id) | |
448 { | |
449 GList *work; | |
450 | |
451 work = id->tab_list; | |
452 while (work) | |
453 { | |
454 TabData *td = work->data; | |
455 work = work->next; | |
456 | |
457 if (td->func_free) td->func_free(td->data); | |
458 g_free(td); | |
459 } | |
460 g_list_free(id->tab_list); | |
461 id->tab_list = NULL; | |
462 } | |
463 | |
221 | 464 static InfoTabsPos *info_tabs_pos_new(gpointer func, gint pos) |
465 { | |
466 InfoTabsPos *t = g_new0(InfoTabsPos, 1); | |
467 t->func = func; | |
468 t->pos = pos; | |
469 | |
470 return t; | |
471 } | |
472 | |
473 static void info_tabs_pos_list_append(gpointer func) | |
474 { | |
475 static gint pos = 0; | |
476 | |
477 info_tabs_pos_list = g_list_append(info_tabs_pos_list, info_tabs_pos_new(func, pos++)); | |
478 } | |
479 | |
480 static gint compare_info_tabs_pos(gconstpointer a, gconstpointer b) | |
481 { | |
482 InfoTabsPos *ta = (InfoTabsPos *) a; | |
483 InfoTabsPos *tb = (InfoTabsPos *) b; | |
484 | |
485 if (ta->pos > tb->pos) return 1; | |
486 return -1; | |
487 } | |
488 | |
684 | 489 static gpointer info_tab_new_funcs[] = { |
490 info_tab_general_new, | |
491 info_tab_meta_new, | |
492 info_tab_exif_new, | |
493 }; | |
494 | |
495 gchar *info_tab_default_order(void) | |
496 { | |
736 | 497 guint i; |
684 | 498 static gchar str[G_N_ELEMENTS(info_tab_new_funcs) + 1]; |
499 | |
500 for (i = 0; i < G_N_ELEMENTS(info_tab_new_funcs); i++) | |
501 str[i] = i + '1'; | |
502 str[i] = '\0'; | |
503 | |
504 return str; | |
505 } | |
506 | |
507 static void info_tab_get_order_string(gchar **dest) | |
508 { | |
509 GList *work; | |
510 gchar str[G_N_ELEMENTS(info_tab_new_funcs) + 1]; | |
511 | |
512 g_assert(dest); | |
513 | |
514 if (!info_tabs_pos_list) | |
515 return; | |
516 | |
517 memset(str, 0, G_N_ELEMENTS(info_tab_new_funcs) + 1); | |
518 | |
519 work = info_tabs_pos_list; | |
520 while (work) | |
521 { | |
736 | 522 guint i; |
684 | 523 InfoTabsPos *t = work->data; |
524 work = work->next; | |
525 | |
526 for (i = 0; i < G_N_ELEMENTS(info_tab_new_funcs); i++) | |
527 { | |
528 if (t->func == info_tab_new_funcs[i]) | |
529 { | |
530 g_assert(t->pos >= 0 && t->pos < G_N_ELEMENTS(info_tab_new_funcs)); | |
531 str[t->pos] = i + '1'; | |
532 } | |
533 } | |
534 } | |
535 | |
536 if (strlen(str) != G_N_ELEMENTS(info_tab_new_funcs)) return; | |
537 | |
538 g_free(*dest); | |
539 *dest = g_strdup(str); | |
540 } | |
541 | |
9 | 542 static void info_tabs_init(InfoData *id) |
543 { | |
221 | 544 GList *work; |
545 | |
546 if (!info_tabs_pos_list) | |
547 { | |
736 | 548 guint count = 0; |
549 guint i; | |
684 | 550 gchar *order = options->properties.tabs_order; |
551 | |
552 for (i = 0; i < strlen(order); i++) | |
553 { | |
736 | 554 guint n = order[i] - '1'; |
684 | 555 |
736 | 556 if (n >= G_N_ELEMENTS(info_tab_new_funcs)) break; |
684 | 557 count++; |
558 } | |
559 | |
560 if (count != G_N_ELEMENTS(info_tab_new_funcs)) | |
561 order = info_tab_default_order(); | |
562 | |
563 for (i = 0; i < strlen(order); i++) | |
564 { | |
736 | 565 guint n = order[i] - '1'; |
684 | 566 |
736 | 567 if (n >= G_N_ELEMENTS(info_tab_new_funcs)) continue; |
684 | 568 if (g_list_find(info_tabs_pos_list, info_tab_new_funcs[n])) continue; |
569 info_tabs_pos_list_append(info_tab_new_funcs[n]); | |
570 } | |
221 | 571 } |
572 else | |
573 info_tabs_pos_list = g_list_sort(info_tabs_pos_list, compare_info_tabs_pos); | |
574 | |
684 | 575 info_tab_get_order_string(&options->properties.tabs_order); |
576 | |
221 | 577 work = info_tabs_pos_list; |
578 while (work) | |
579 { | |
580 InfoTabsPos *t = work->data; | |
581 work = work->next; | |
582 | |
583 id->tab_list = g_list_append(id->tab_list, t->func(id)); | |
584 } | |
9 | 585 } |
586 | |
587 /* | |
588 *------------------------------------------------------------------- | |
589 * sync | |
590 *------------------------------------------------------------------- | |
591 */ | |
592 | |
138 | 593 static void info_window_sync(InfoData *id, FileData *fd) |
9 | 594 { |
595 | |
138 | 596 if (!fd) return; |
9 | 597 |
138 | 598 gtk_entry_set_text(GTK_ENTRY(id->name_entry), fd->name); |
9 | 599 |
600 if (id->label_count) | |
601 { | |
602 gchar *buf; | |
603 buf = g_strdup_printf(_("Image %d of %d"), | |
219
92d2444893b8
Fix display of image number in properties dialog when more than one image is selected.
zas_
parents:
196
diff
changeset
|
604 g_list_index(id->list, (gpointer)fd) + 1, |
9 | 605 g_list_length(id->list)); |
606 gtk_label_set_text(GTK_LABEL(id->label_count), buf); | |
607 g_free(buf); | |
608 } | |
609 | |
610 info_tabs_sync(id, FALSE); | |
611 | |
612 id->updated = FALSE; | |
138 | 613 image_change_fd(id->image, fd, 0.0); |
9 | 614 } |
615 | |
221 | 616 static void info_notebook_reordered_cb(GtkNotebook *notebook, GtkWidget *child, guint page_num, gpointer data) |
617 { | |
618 InfoData *id = data; | |
619 GList *work; | |
620 | |
621 /* Save current tabs position to be able to restore them later. */ | |
622 work = id->tab_list; | |
623 while (work) | |
624 { | |
625 GList *tabpos; | |
626 TabData *td = work->data; | |
627 gint pos = gtk_notebook_page_num(GTK_NOTEBOOK(id->notebook), GTK_WIDGET(td->child)); | |
628 work = work->next; | |
629 | |
630 tabpos = info_tabs_pos_list; | |
631 while (tabpos) | |
632 { | |
633 InfoTabsPos *t = tabpos->data; | |
634 tabpos = tabpos->next; | |
635 | |
636 if (t->func == td->func_new) | |
637 { | |
638 t->pos = pos; | |
639 break; | |
640 } | |
442 | 641 } |
551
5b127951daa1
info_notebook_reordered_cb(): do not call info_tabs_sync(), it causes exif
zas_
parents:
531
diff
changeset
|
642 } |
684 | 643 |
644 info_tabs_pos_list = g_list_sort(info_tabs_pos_list, compare_info_tabs_pos); | |
645 info_tab_get_order_string(&options->properties.tabs_order); | |
221 | 646 } |
647 | |
9 | 648 /* |
649 *------------------------------------------------------------------- | |
650 * drag n drop (dropping not supported!) | |
651 *------------------------------------------------------------------- | |
652 */ | |
653 | |
654 static void info_window_dnd_data_set(GtkWidget *widget, GdkDragContext *context, | |
655 GtkSelectionData *selection_data, guint info, | |
656 guint time, gpointer data) | |
657 { | |
658 InfoData *id = data; | |
138 | 659 FileData *fd; |
9 | 660 |
138 | 661 fd = image_get_fd(id->image); |
662 if (fd) | |
9 | 663 { |
664 gchar *text; | |
665 gint len; | |
666 GList *list; | |
667 gint plain_text; | |
668 | |
669 switch (info) | |
670 { | |
671 case TARGET_URI_LIST: | |
672 plain_text = FALSE; | |
673 break; | |
674 case TARGET_TEXT_PLAIN: | |
675 default: | |
676 plain_text = TRUE; | |
677 break; | |
678 } | |
138 | 679 list = g_list_append(NULL, fd); |
680 text = uri_text_from_filelist(list, &len, plain_text); | |
9 | 681 g_list_free(list); |
682 | |
683 gtk_selection_data_set(selection_data, selection_data->target, | |
64
04ff0df3ad2f
Mon Aug 15 17:13:57 2005 John Ellis <johne@verizon.net>
gqview
parents:
23
diff
changeset
|
684 8, (guchar *)text, len); |
9 | 685 g_free(text); |
686 } | |
687 } | |
688 | |
689 static void info_window_dnd_init(InfoData *id) | |
690 { | |
691 ImageWindow *imd; | |
692 | |
693 imd = id->image; | |
694 | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
695 gtk_drag_source_set(imd->pr, GDK_BUTTON2_MASK, |
9 | 696 dnd_file_drag_types, dnd_file_drag_types_count, |
697 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK); | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
698 g_signal_connect(G_OBJECT(imd->pr), "drag_data_get", |
9 | 699 G_CALLBACK(info_window_dnd_data_set), id); |
700 | |
701 #if 0 | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
702 gtk_drag_dest_set(imd->pr, |
9 | 703 GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_DROP, |
704 dnd_file_drop_types, dnd_file_drop_types_count, | |
705 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK); | |
23
17acca639a86
##### Note: GQview CVS on sourceforge is not always up to date, please use #####
gqview
parents:
15
diff
changeset
|
706 g_signal_connect(G_OBJECT(imd->pr), "drag_data_received", |
9 | 707 G_CALLBACK(info_window_dnd_data_get), id); |
708 #endif | |
709 } | |
710 | |
711 /* | |
712 *------------------------------------------------------------------- | |
713 * base window | |
714 *------------------------------------------------------------------- | |
715 */ | |
716 | |
717 static void info_window_image_update_cb(ImageWindow *imd, gpointer data) | |
718 { | |
719 InfoData *id = data; | |
720 | |
721 /* only do this once after when loading a new image, | |
722 * for tabs that depend on image data (exif) | |
723 * Subsequent updates are ignored, as the image | |
724 * should not really changed if id->updated is TRUE. | |
725 */ | |
726 | |
727 if (id->updated) return; | |
728 if (imd->unknown) return; | |
729 | |
730 info_tabs_sync(id, TRUE); | |
731 id->updated = TRUE; | |
732 } | |
733 | |
734 static void info_window_back_cb(GtkWidget *widget, gpointer data) | |
735 { | |
736 InfoData *id = data; | |
737 GList *work; | |
738 | |
138 | 739 work = g_list_find(id->list, (gpointer)id->fd); |
9 | 740 if (!work || !work->prev) return; |
741 | |
742 work = work->prev; | |
138 | 743 id->fd = work->data; |
9 | 744 |
138 | 745 info_window_sync(id, id->fd); |
9 | 746 |
747 gtk_widget_set_sensitive(id->button_back, (work->prev != NULL)); | |
748 gtk_widget_set_sensitive(id->button_next, TRUE); | |
749 } | |
750 | |
751 static void info_window_next_cb(GtkWidget *widget, gpointer data) | |
752 { | |
753 InfoData *id = data; | |
754 GList *work; | |
755 | |
138 | 756 work = g_list_find(id->list, (gpointer)id->fd); |
9 | 757 if (!work || !work->next) return; |
758 | |
759 work = work->next; | |
138 | 760 id->fd = work->data; |
9 | 761 |
138 | 762 info_window_sync(id, id->fd); |
9 | 763 |
764 gtk_widget_set_sensitive(id->button_next, (work->next != NULL)); | |
765 gtk_widget_set_sensitive(id->button_back, TRUE); | |
766 } | |
767 | |
768 static void info_window_image_button_cb(ImageWindow *imd, gint button, guint32 time, | |
769 gdouble x, gdouble y, guint state, gpointer data) | |
770 { | |
448
a73cc0fa14d0
Use explicit names for mouse buttons instead of numbers.
zas_
parents:
446
diff
changeset
|
771 if (button == MOUSE_BUTTON_LEFT) |
9 | 772 { |
773 info_window_next_cb(NULL, data); | |
774 } | |
448
a73cc0fa14d0
Use explicit names for mouse buttons instead of numbers.
zas_
parents:
446
diff
changeset
|
775 else if (button == MOUSE_BUTTON_MIDDLE || button == MOUSE_BUTTON_RIGHT) |
9 | 776 { |
777 info_window_back_cb(NULL, data); | |
778 } | |
779 } | |
780 | |
781 static void info_window_image_scroll_cb(ImageWindow *imd, GdkScrollDirection direction, guint32 time, | |
782 gdouble x, gdouble y, guint state, gpointer data) | |
783 { | |
784 if (direction == GDK_SCROLL_UP) | |
785 { | |
786 info_window_back_cb(NULL, data); | |
787 } | |
788 else if (direction == GDK_SCROLL_DOWN) | |
789 { | |
790 info_window_next_cb(NULL, data); | |
791 } | |
792 } | |
793 | |
794 static void info_window_close(InfoData *id) | |
795 { | |
764
ae618ebec3e9
Save properties window width and height to rc file and restore
zas_
parents:
736
diff
changeset
|
796 gdk_drawable_get_size(id->window->window, &options->layout.properties_window.w, &options->layout.properties_window.h); |
ae618ebec3e9
Save properties window width and height to rc file and restore
zas_
parents:
736
diff
changeset
|
797 options->layout.properties_window.w = MAX(options->layout.properties_window.w, DEF_PROPERTY_WIDTH); |
ae618ebec3e9
Save properties window width and height to rc file and restore
zas_
parents:
736
diff
changeset
|
798 options->layout.properties_window.h = MAX(options->layout.properties_window.h, DEF_PROPERTY_HEIGHT); |
9 | 799 |
800 gtk_widget_destroy(id->window); | |
801 } | |
802 | |
803 static void info_window_close_cb(GtkWidget *widget, gpointer data) | |
804 { | |
805 InfoData *id = data; | |
806 | |
807 info_window_close(id); | |
808 } | |
809 | |
810 static gint info_window_delete_cb(GtkWidget *widget, GdkEventAny *event, gpointer data) | |
811 { | |
812 InfoData *id = data; | |
813 | |
814 info_window_close(id); | |
815 return TRUE; | |
816 } | |
817 | |
818 static void info_window_destroy_cb(GtkWidget *widget, gpointer data) | |
819 { | |
820 InfoData *id = data; | |
821 | |
822 info_tabs_free(id); | |
138 | 823 filelist_free(id->list); |
9 | 824 g_free(id); |
825 } | |
826 | |
479
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
827 void info_window_new(FileData *fd, GList *list, GtkWidget *parent) |
9 | 828 { |
829 InfoData *id; | |
830 GtkWidget *main_vbox; | |
831 GtkWidget *paned; | |
832 GtkWidget *hbox; | |
833 GtkWidget *button; | |
834 GtkWidget *label; | |
835 GdkGeometry geometry; | |
765
339db85846da
Preserve properties window width and height during session even when
zas_
parents:
764
diff
changeset
|
836 static gboolean run_once = FALSE; |
9 | 837 |
138 | 838 if (!fd && !list) return; |
9 | 839 |
765
339db85846da
Preserve properties window width and height during session even when
zas_
parents:
764
diff
changeset
|
840 run_once = TRUE; |
339db85846da
Preserve properties window width and height during session even when
zas_
parents:
764
diff
changeset
|
841 |
9 | 842 if (!list) |
843 { | |
138 | 844 list = g_list_append(NULL, file_data_ref(fd)); |
9 | 845 } |
846 | |
847 id = g_new0(InfoData, 1); | |
848 | |
849 id->list = list; | |
138 | 850 id->fd = (FileData *)id->list->data; |
9 | 851 id->updated = FALSE; |
852 | |
289
6a7298988a7a
Simplify and unify gtk_window creation with the help of
zas_
parents:
288
diff
changeset
|
853 id->window = window_new(GTK_WINDOW_TOPLEVEL, "properties", NULL, NULL, _("Image properties")); |
9 | 854 gtk_window_set_type_hint(GTK_WINDOW(id->window), GDK_WINDOW_TYPE_HINT_DIALOG); |
479
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
855 id->parent = parent; |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
856 if (GTK_IS_WINDOW(id->parent)) { |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
857 gtk_window_set_keep_above(GTK_WINDOW(id->window), TRUE); |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
858 #if 0 |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
859 /* work, but behavior is not that great */ |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
860 gtk_window_set_transient_for(GTK_WINDOW(id->window), GTK_WINDOW(id->parent)); |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
861 #endif |
5212d4fed37f
Ensure Properties dialog is displayed above fullscreen window.
zas_
parents:
475
diff
changeset
|
862 } |
9 | 863 gtk_window_set_resizable(GTK_WINDOW(id->window), TRUE); |
864 | |
865 geometry.min_width = 32; | |
866 geometry.min_height = 32; | |
867 geometry.base_width = DEF_PROPERTY_WIDTH; | |
868 geometry.base_height = DEF_PROPERTY_HEIGHT; | |
869 gtk_window_set_geometry_hints(GTK_WINDOW(id->window), NULL, &geometry, | |
870 GDK_HINT_MIN_SIZE | GDK_HINT_BASE_SIZE); | |
871 | |
765
339db85846da
Preserve properties window width and height during session even when
zas_
parents:
764
diff
changeset
|
872 if (options->layout.save_window_positions || run_once) |
764
ae618ebec3e9
Save properties window width and height to rc file and restore
zas_
parents:
736
diff
changeset
|
873 gtk_window_set_default_size(GTK_WINDOW(id->window), options->layout.properties_window.w, options->layout.properties_window.h); |
ae618ebec3e9
Save properties window width and height to rc file and restore
zas_
parents:
736
diff
changeset
|
874 else |
ae618ebec3e9
Save properties window width and height to rc file and restore
zas_
parents:
736
diff
changeset
|
875 gtk_window_set_default_size(GTK_WINDOW(id->window), DEF_PROPERTY_WIDTH, DEF_PROPERTY_HEIGHT); |
9 | 876 |
877 gtk_container_set_border_width(GTK_CONTAINER(id->window), PREF_PAD_BORDER); | |
878 | |
879 g_signal_connect(G_OBJECT(id->window), "delete_event", | |
880 G_CALLBACK(info_window_delete_cb), id); | |
881 g_signal_connect(G_OBJECT(id->window), "destroy", | |
882 G_CALLBACK(info_window_destroy_cb), id); | |
883 | |
884 paned = gtk_hpaned_new(); | |
885 gtk_container_add(GTK_CONTAINER(id->window), paned); | |
886 gtk_widget_show(paned); | |
887 | |
888 id->image = image_new(FALSE); | |
889 image_set_update_func(id->image, info_window_image_update_cb, id); | |
890 | |
891 image_set_button_func(id->image, info_window_image_button_cb, id); | |
892 image_set_scroll_func(id->image, info_window_image_scroll_cb, id); | |
893 | |
894 gtk_widget_set_size_request(id->image->widget, IMAGE_SIZE_W, IMAGE_SIZE_H); | |
895 gtk_paned_pack1(GTK_PANED(paned), id->image->widget, FALSE, TRUE); | |
896 gtk_widget_show(id->image->widget); | |
897 | |
898 main_vbox = gtk_vbox_new(FALSE, 0); | |
899 gtk_paned_pack2(GTK_PANED(paned), main_vbox, TRUE, TRUE); | |
900 gtk_widget_show(main_vbox); | |
901 | |
902 hbox = pref_box_new(main_vbox, FALSE, GTK_ORIENTATION_HORIZONTAL, PREF_PAD_SPACE); | |
903 label = pref_label_new(hbox, _("Filename:")); | |
904 pref_label_bold(label, TRUE, FALSE); | |
905 | |
906 id->name_entry = gtk_entry_new(); | |
907 gtk_editable_set_editable(GTK_EDITABLE(id->name_entry), FALSE); | |
908 gtk_box_pack_start(GTK_BOX(hbox), id->name_entry, TRUE, TRUE, 0); | |
909 gtk_widget_show(id->name_entry); | |
910 | |
911 id->notebook = gtk_notebook_new(); | |
912 gtk_notebook_set_tab_pos(GTK_NOTEBOOK(id->notebook), GTK_POS_TOP); | |
913 gtk_box_pack_start(GTK_BOX(main_vbox), id->notebook, TRUE, TRUE, 5); | |
221 | 914 g_signal_connect(G_OBJECT(id->notebook), "page-reordered", |
915 G_CALLBACK(info_notebook_reordered_cb), id); | |
916 | |
9 | 917 gtk_widget_show(id->notebook); |
918 | |
919 pref_spacer(main_vbox, PREF_PAD_GAP); | |
920 | |
921 hbox = pref_box_new(main_vbox, FALSE, GTK_ORIENTATION_HORIZONTAL, PREF_PAD_GAP); | |
922 | |
923 id->button_back = pref_button_new(hbox, GTK_STOCK_GO_BACK, NULL, TRUE, | |
924 G_CALLBACK(info_window_back_cb), id); | |
925 gtk_widget_set_sensitive(id->button_back, FALSE); | |
926 | |
927 id->button_next = pref_button_new(hbox, GTK_STOCK_GO_FORWARD, NULL, TRUE, | |
928 G_CALLBACK(info_window_next_cb), id); | |
929 gtk_widget_set_sensitive(id->button_next, (id->list->next != NULL)); | |
930 | |
931 if (id->list->next) | |
932 { | |
933 id->label_count = pref_label_new(hbox, ""); | |
934 } | |
935 | |
936 button = pref_button_new(NULL, GTK_STOCK_CLOSE, NULL, FALSE, | |
937 G_CALLBACK(info_window_close_cb), id); | |
938 gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, FALSE, 0); | |
939 gtk_widget_show(button); | |
940 | |
941 /* set up tabs */ | |
942 | |
943 info_tabs_init(id); | |
944 | |
945 /* fill it */ | |
946 | |
138 | 947 info_window_sync(id, id->fd); |
9 | 948 |
949 /* finish */ | |
950 | |
951 info_window_dnd_init(id); | |
952 | |
953 gtk_widget_show(id->window); | |
954 } |