4 * Copyright (C) 2008 The Geeqie Team
6 * Author: John Ellis, Laurent Monin
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!
21 #include "secure_save.h"
22 #include "ui_fileops.h"
25 #include "filefilter.h"
33 static const gchar *group_keys[] = {KEYWORD_KEY, COMMENT_KEY, NULL}; /* tags that will be written to all files in a group */
35 static gboolean metadata_write_queue_idle_cb(gpointer data);
36 static gint metadata_legacy_write(FileData *fd);
37 static void metadata_legacy_delete(FileData *fd, const gchar *except);
42 *-------------------------------------------------------------------
44 *-------------------------------------------------------------------
47 static GList *metadata_write_queue = NULL;
48 static gint metadata_write_idle_id = -1;
50 static void metadata_write_queue_add(FileData *fd)
52 if (!g_list_find(metadata_write_queue, fd))
54 metadata_write_queue = g_list_prepend(metadata_write_queue, fd);
58 if (metadata_write_idle_id != -1)
60 g_source_remove(metadata_write_idle_id);
61 metadata_write_idle_id = -1;
64 if (options->metadata.confirm_timeout > 0)
66 metadata_write_idle_id = g_timeout_add(options->metadata.confirm_timeout * 1000, metadata_write_queue_idle_cb, NULL);
71 gboolean metadata_write_queue_remove(FileData *fd)
73 g_hash_table_destroy(fd->modified_xmp);
74 fd->modified_xmp = NULL;
76 metadata_write_queue = g_list_remove(metadata_write_queue, fd);
78 file_data_increment_version(fd);
79 file_data_send_notification(fd, NOTIFY_TYPE_REREAD);
85 gboolean metadata_write_queue_remove_list(GList *list)
93 FileData *fd = work->data;
95 ret = ret && metadata_write_queue_remove(fd);
101 gboolean metadata_write_queue_confirm(FileUtilDoneFunc done_func, gpointer done_data)
104 GList *to_approve = NULL;
106 work = metadata_write_queue;
109 FileData *fd = work->data;
112 if (fd->change) continue; /* another operation in progress, skip this file for now */
114 to_approve = g_list_prepend(to_approve, file_data_ref(fd));
117 file_util_write_metadata(NULL, to_approve, NULL, done_func, done_data);
119 filelist_free(to_approve);
121 return (metadata_write_queue != NULL);
124 static gboolean metadata_write_queue_idle_cb(gpointer data)
126 metadata_write_queue_confirm(NULL, NULL);
127 metadata_write_idle_id = -1;
131 gboolean metadata_write_perform(FileData *fd)
136 g_assert(fd->change);
138 if (fd->change->dest &&
139 strcmp(extension_from_path(fd->change->dest), GQ_CACHE_EXT_METADATA) == 0)
141 success = metadata_legacy_write(fd);
142 if (success) metadata_legacy_delete(fd, fd->change->dest);
146 /* write via exiv2 */
147 /* we can either use cached metadata which have fd->modified_xmp already applied
148 or read metadata from file and apply fd->modified_xmp
149 metadata are read also if the file was modified meanwhile */
150 exif = exif_read_fd(fd);
151 if (!exif) return FALSE;
153 success = (fd->change->dest) ? exif_write_sidecar(exif, fd->change->dest) : exif_write(exif); /* write modified metadata */
154 exif_free_fd(fd, exif);
156 if (success) metadata_legacy_delete(fd, fd->change->dest);
160 static gboolean metadata_check_key(const gchar *keys[], const gchar *key)
162 const gchar **k = keys;
166 if (strcmp(key, *k) == 0) return TRUE;
172 gboolean metadata_write_list(FileData *fd, const gchar *key, const GList *values)
174 if (!fd->modified_xmp)
176 fd->modified_xmp = g_hash_table_new_full(g_str_hash, g_str_equal, g_free, (GDestroyNotify)string_list_free);
178 g_hash_table_insert(fd->modified_xmp, g_strdup(key), string_list_copy((GList *)values));
181 exif_update_metadata(fd->exif, key, values);
183 metadata_write_queue_add(fd);
184 file_data_increment_version(fd);
185 file_data_send_notification(fd, NOTIFY_TYPE_INTERNAL);
187 if (options->metadata.sync_grouped_files && metadata_check_key(group_keys, key))
189 GList *work = fd->sidecar_files;
193 FileData *sfd = work->data;
196 if (filter_file_class(sfd->extension, FORMAT_CLASS_META)) continue;
198 metadata_write_list(sfd, key, values);
206 gboolean metadata_write_string(FileData *fd, const gchar *key, const char *value)
208 GList *list = g_list_append(NULL, g_strdup(value));
209 gboolean ret = metadata_write_list(fd, key, list);
210 string_list_free(list);
216 *-------------------------------------------------------------------
217 * keyword / comment read/write
218 *-------------------------------------------------------------------
221 static gint metadata_file_write(gchar *path, GHashTable *modified_xmp)
224 GList *keywords = g_hash_table_lookup(modified_xmp, KEYWORD_KEY);
225 GList *comment_l = g_hash_table_lookup(modified_xmp, COMMENT_KEY);
226 gchar *comment = comment_l ? comment_l->data : NULL;
228 ssi = secure_open(path);
229 if (!ssi) return FALSE;
231 secure_fprintf(ssi, "#%s comment (%s)\n\n", GQ_APPNAME, VERSION);
233 secure_fprintf(ssi, "[keywords]\n");
234 while (keywords && secsave_errno == SS_ERR_NONE)
236 const gchar *word = keywords->data;
237 keywords = keywords->next;
239 secure_fprintf(ssi, "%s\n", word);
241 secure_fputc(ssi, '\n');
243 secure_fprintf(ssi, "[comment]\n");
244 secure_fprintf(ssi, "%s\n", (comment) ? comment : "");
246 secure_fprintf(ssi, "#end\n");
248 return (secure_close(ssi) == 0);
251 static gint metadata_legacy_write(FileData *fd)
253 gint success = FALSE;
255 g_assert(fd->change && fd->change->dest);
256 gchar *metadata_pathl;
258 DEBUG_1("Saving comment: %s", fd->change->dest);
260 metadata_pathl = path_from_utf8(fd->change->dest);
262 success = metadata_file_write(metadata_pathl, fd->modified_xmp);
264 g_free(metadata_pathl);
269 static gint metadata_file_read(gchar *path, GList **keywords, gchar **comment)
273 MetadataKey key = MK_NONE;
275 GString *comment_build = NULL;
277 f = fopen(path, "r");
278 if (!f) return FALSE;
280 while (fgets(s_buf, sizeof(s_buf), f))
284 if (*ptr == '#') continue;
285 if (*ptr == '[' && key != MK_COMMENT)
287 gchar *keystr = ++ptr;
290 while (*ptr != ']' && *ptr != '\n' && *ptr != '\0') ptr++;
295 if (g_ascii_strcasecmp(keystr, "keywords") == 0)
297 else if (g_ascii_strcasecmp(keystr, "comment") == 0)
309 while (*ptr != '\n' && *ptr != '\0') ptr++;
311 if (strlen(s_buf) > 0)
313 gchar *kw = utf8_validate_or_convert(s_buf);
315 list = g_list_prepend(list, kw);
320 if (!comment_build) comment_build = g_string_new("");
321 g_string_append(comment_build, s_buf);
328 *keywords = g_list_reverse(list);
334 gchar *ptr = comment_build->str;
336 /* strip leading and trailing newlines */
337 while (*ptr == '\n') ptr++;
339 while (len > 0 && ptr[len - 1] == '\n') len--;
340 if (ptr[len] == '\n') len++; /* keep the last one */
343 gchar *text = g_strndup(ptr, len);
345 *comment = utf8_validate_or_convert(text);
349 g_string_free(comment_build, TRUE);
355 static void metadata_legacy_delete(FileData *fd, const gchar *except)
357 gchar *metadata_path;
358 gchar *metadata_pathl;
361 metadata_path = cache_find_location(CACHE_TYPE_METADATA, fd->path);
362 if (metadata_path && (!except || strcmp(metadata_path, except) != 0))
364 metadata_pathl = path_from_utf8(metadata_path);
365 unlink(metadata_pathl);
366 g_free(metadata_pathl);
367 g_free(metadata_path);
369 metadata_path = cache_find_location(CACHE_TYPE_XMP_METADATA, fd->path);
370 if (metadata_path && (!except || strcmp(metadata_path, except) != 0))
372 metadata_pathl = path_from_utf8(metadata_path);
373 unlink(metadata_pathl);
374 g_free(metadata_pathl);
375 g_free(metadata_path);
379 static gint metadata_legacy_read(FileData *fd, GList **keywords, gchar **comment)
381 gchar *metadata_path;
382 gchar *metadata_pathl;
383 gint success = FALSE;
384 if (!fd) return FALSE;
386 metadata_path = cache_find_location(CACHE_TYPE_METADATA, fd->path);
387 if (!metadata_path) return FALSE;
389 metadata_pathl = path_from_utf8(metadata_path);
391 success = metadata_file_read(metadata_pathl, keywords, comment);
393 g_free(metadata_pathl);
394 g_free(metadata_path);
399 static GList *remove_duplicate_strings_from_list(GList *list)
402 GHashTable *hashtable = g_hash_table_new(g_str_hash, g_str_equal);
403 GList *newlist = NULL;
407 gchar *key = work->data;
409 if (g_hash_table_lookup(hashtable, key) == NULL)
411 g_hash_table_insert(hashtable, (gpointer) key, GINT_TO_POINTER(1));
412 newlist = g_list_prepend(newlist, key);
417 g_hash_table_destroy(hashtable);
420 return g_list_reverse(newlist);
424 static gint metadata_xmp_read(FileData *fd, GList **keywords, gchar **comment)
428 exif = exif_read_fd(fd);
429 if (!exif) return FALSE;
434 ExifItem *item = exif_get_item(exif, COMMENT_KEY);
436 text = exif_item_get_string(item, 0);
437 *comment = utf8_validate_or_convert(text);
447 item = exif_get_item(exif, KEYWORD_KEY);
448 for (i = 0; i < exif_item_get_elements(item); i++)
450 gchar *kw = exif_item_get_string(item, i);
455 utf8_kw = utf8_validate_or_convert(kw);
456 *keywords = g_list_append(*keywords, (gpointer) utf8_kw);
461 * Exiv2 handles Iptc keywords as multiple entries with the
462 * same key, thus exif_get_item returns only the first keyword
463 * and the only way to get all keywords is to iterate through
466 /* Read IPTC keywords only if there are no XMP keywords
467 * IPTC does not have standard charset, thus the encoding may differ
468 * from XMP and keyword merging is not reliable.
472 for (item = exif_get_first_item(exif);
474 item = exif_get_next_item(exif))
478 tag = exif_item_get_tag_id(item);
481 gchar *tag_name = exif_item_get_tag_name(item);
483 if (strcmp(tag_name, "Iptc.Application2.Keywords") == 0)
488 kw = exif_item_get_data_as_text(item);
491 utf8_kw = utf8_validate_or_convert(kw);
492 *keywords = g_list_append(*keywords, (gpointer) utf8_kw);
501 exif_free_fd(fd, exif);
503 return (comment && *comment) || (keywords && *keywords);
506 gint metadata_read(FileData *fd, GList **keywords, gchar **comment)
508 GList *keywords_xmp = NULL;
509 GList *keywords_legacy = NULL;
510 gchar *comment_xmp = NULL;
511 gchar *comment_legacy = NULL;
512 gint result_xmp, result_legacy;
514 if (!fd) return FALSE;
516 result_xmp = metadata_xmp_read(fd, &keywords_xmp, &comment_xmp);
517 result_legacy = metadata_legacy_read(fd, &keywords_legacy, &comment_legacy);
519 if (!result_xmp && !result_legacy)
526 if (result_xmp && result_legacy)
527 *keywords = g_list_concat(keywords_xmp, keywords_legacy);
529 *keywords = result_xmp ? keywords_xmp : keywords_legacy;
531 *keywords = remove_duplicate_strings_from_list(*keywords);
535 if (result_xmp) string_list_free(keywords_xmp);
536 if (result_legacy) string_list_free(keywords_legacy);
542 if (result_xmp && result_legacy && comment_xmp && comment_legacy && *comment_xmp && *comment_legacy)
543 *comment = g_strdup_printf("%s\n%s", comment_xmp, comment_legacy);
545 *comment = result_xmp ? comment_xmp : comment_legacy;
548 if (result_xmp && (!comment || *comment != comment_xmp)) g_free(comment_xmp);
549 if (result_legacy && (!comment || *comment != comment_legacy)) g_free(comment_legacy);
551 // return FALSE in the following cases:
552 // - only looking for a comment and didn't find one
553 // - only looking for keywords and didn't find any
554 // - looking for either a comment or keywords, but found nothing
555 if ((!keywords && comment && !*comment) ||
556 (!comment && keywords && !*keywords) ||
557 ( comment && !*comment && keywords && !*keywords))
563 void metadata_set(FileData *fd, GList *new_keywords, gchar *new_comment, gboolean append)
565 gchar *comment = NULL;
566 GList *keywords = NULL;
567 GList *keywords_list = NULL;
569 metadata_read(fd, &keywords, &comment);
573 if (append && comment && *comment)
575 gchar *tmp = comment;
577 comment = g_strconcat(tmp, new_comment, NULL);
583 comment = g_strdup(new_comment);
589 if (append && keywords && g_list_length(keywords) > 0)
605 gchar *needle = p->data;
608 if (strcmp(needle, key) == 0) key = NULL;
611 if (key) keywords = g_list_append(keywords, g_strdup(key));
613 keywords_list = keywords;
617 keywords_list = new_keywords;
621 metadata_write_string(fd, COMMENT_KEY, comment);
622 metadata_write_list(fd, KEYWORD_KEY, keywords);
624 string_list_free(keywords);
628 gboolean find_string_in_list(GList *list, const gchar *string)
632 gchar *haystack = list->data;
634 if (haystack && string && strcmp(haystack, string) == 0) return TRUE;
642 #define KEYWORDS_SEPARATOR(c) ((c) == ',' || (c) == ';' || (c) == '\n' || (c) == '\r' || (c) == '\b')
644 GList *string_to_keywords_list(const gchar *text)
647 const gchar *ptr = text;
654 while (KEYWORDS_SEPARATOR(*ptr)) ptr++;
656 while (*ptr != '\0' && !KEYWORDS_SEPARATOR(*ptr))
662 /* trim starting and ending whitespaces */
663 while (l > 0 && g_ascii_isspace(*begin)) begin++, l--;
664 while (l > 0 && g_ascii_isspace(begin[l-1])) l--;
668 gchar *keyword = g_strndup(begin, l);
670 /* only add if not already in the list */
671 if (find_string_in_list(list, keyword) == FALSE)
672 list = g_list_append(list, keyword);
686 gboolean meta_data_get_keyword_mark(FileData *fd, gint n, gpointer data)
689 gboolean found = FALSE;
690 if (metadata_read(fd, &keywords, NULL))
692 GList *work = keywords;
696 gchar *kw = work->data;
699 if (strcmp(kw, data) == 0)
705 string_list_free(keywords);
710 gboolean meta_data_set_keyword_mark(FileData *fd, gint n, gboolean value, gpointer data)
712 GList *keywords = NULL;
713 gboolean found = FALSE;
714 gboolean changed = FALSE;
716 metadata_read(fd, &keywords, NULL);
722 gchar *kw = work->data;
724 if (strcmp(kw, data) == 0)
730 keywords = g_list_delete_link(keywords, work);
740 keywords = g_list_append(keywords, g_strdup(data));
743 if (changed) metadata_write_list(fd, KEYWORD_KEY, keywords);
745 string_list_free(keywords);
750 /* vim: set shiftwidth=8 softtabstop=0 cindent cinoptions={1s: */