X-Git-Url: https://git.dlugolecki.net.pl/?a=blobdiff_plain;f=gom%2Fgom_modify.c;h=5bfb34592fb981b9be37d098261eb2ba9f8726ce;hb=3155d964e15a3c49121c01d2808e5a846779b812;hp=95bd926b884b2fba50ab91de4ce459510381b65a;hpb=38a93aa4ec4d643e63d797e1bdbf639b1082a4cd;p=gedcom-parse.git diff --git a/gom/gom_modify.c b/gom/gom_modify.c index 95bd926..5bfb345 100644 --- a/gom/gom_modify.c +++ b/gom/gom_modify.c @@ -23,7 +23,8 @@ #include #include -#include "utf8-locale.h" +#include "utf8tools.h" +#include "user_rec.h" #include "gom.h" #include "gom_internal.h" @@ -37,28 +38,176 @@ char* gom_get_string(char* data) return data; } -char* gom_get_string_locale(char* data, int* conversion_failures) +char* gom_get_string_for_locale(char* data, int* conversion_failures) { return convert_utf8_to_locale(gom_get_string(data), conversion_failures); } -char* gom_set_string(char** data, const char* utf8_value) +char* gom_set_string(char** data, const char* utf8_str) { char* result = NULL; - char* newptr = strdup(utf8_value); - - if (!newptr) - MEMORY_ERROR; + char* newptr; + + if (!is_utf8_string(utf8_str)) { + gedcom_error(_("The input '%s' is not a valid UTF-8 string"), utf8_str); + } else { - if (*data) free(*data); - *data = newptr; - result = *data; + newptr = strdup(utf8_str); + if (!newptr) + MEMORY_ERROR; + else { + if (*data) free(*data); + *data = newptr; + result = *data; + } } return result; } -char* gom_set_string_locale(char** data, const char* locale_value) +char* gom_set_string_for_locale(char** data, const char* locale_str) +{ + char* result = NULL; + char* utf8_str = convert_locale_to_utf8(locale_str); + + if (!utf8_str) + gedcom_error(_("The input '%s' is not a valid string for the locale"), + locale_str); + else + result = gom_set_string(data, utf8_str); + + return result; +} + +void unref_xref_value(struct xref_value *xref) +{ + if (xref) + gedcom_unlink_xref(xref->type, xref->string); +} + +void UNREFALLFUNC(xref_list)(struct xref_list* obj) +{ + if (obj) { + struct xref_list* runner; + for (runner = obj; runner; runner = runner->next) { + unref_xref_value(runner->xref); + UNREFALLFUNC(user_data)(runner->extra); + } + } +} + +void CLEANFUNC(xref_list)(struct xref_list *obj) +{ + if (obj) { + DESTROY_CHAIN_ELTS(user_data, obj->extra); + } +} + +struct xref_value* gom_set_xref(struct xref_value** data, const char* xref) +{ + struct xref_value* result = NULL; + struct xref_value* newval = NULL; + + if (data) { + if (xref) { + newval = gedcom_get_by_xref(xref); + if (!newval) + gedcom_error(_("No record found for xref '%s'"), xref); + } + + /* Unreference the old value if not NULL */ + if (*data) + result = gedcom_unlink_xref((*data)->type, (*data)->string); + else + result = newval; + + /* Reference the new value if not NULL */ + if (result != NULL && newval) { + result = gedcom_link_xref(newval->type, newval->string); + /* On error, perform rollback to old value (guaranteed to work) */ + if (result == NULL) + gedcom_link_xref((*data)->type, (*data)->string); + } + + if (result != NULL) { + *data = newval; + result = newval; + } + } + return result; +} + +struct xref_list* gom_add_xref(struct xref_list** data, const char* xref) { - return gom_set_string(data, convert_locale_to_utf8(locale_value)); + struct xref_value* result = NULL; + struct xref_value* newval = NULL; + struct xref_list* xrl = NULL; + + if (data && xref) { + newval = gedcom_get_by_xref(xref); + if (!newval) + gedcom_error(_("No record found for xref '%s'"), xref); + else { + result = gedcom_link_xref(newval->type, newval->string); + if (result != NULL) { + MAKE_CHAIN_ELT(xref_list, *data, xrl); + if (xrl) xrl->xref = newval; + } + } + } + + return xrl; +} + +struct xref_list* find_xref(struct xref_list** data, const char* xref) +{ + struct xref_list* result = NULL; + struct xref_value* xr = gedcom_get_by_xref(xref); + if (!xr) + gedcom_error(_("No record found for xref '%s'"), xref); + else { + struct xref_list* xrl; + for (xrl = *data ; xrl ; xrl = xrl->next) { + if (xrl->xref == xr) { + result = xrl; + break; + } + } + if (! result) + gedcom_error(_("Xref '%s' not part of chain"), xref); + } + return result; +} + +int gom_remove_xref(struct xref_list** data, const char* xref) +{ + int result = 1; + + if (data && xref) { + struct xref_list* xrl = find_xref(data, xref); + if (xrl) { + UNLINK_CHAIN_ELT(xref_list, *data, xrl); + gedcom_unlink_xref(xrl->xref->type, xrl->xref->string); + CLEANFUNC(xref_list)(xrl); + SAFE_FREE(xrl); + result = 0; + } + } + + return result; +} + +int gom_move_xref(Gom_direction dir, struct xref_list** data, const char* xref) +{ + int result = 1; + + if (data && xref) { + struct xref_list* xrl = find_xref(data, xref); + if (xrl) { + MOVE_CHAIN_ELT(xref_list, dir, *data, xrl); + result = 0; + } + } + + return result; }