/* * moolangmgr.c * * Copyright (C) 2004-2006 by Yevgen Muntyan * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * See COPYING file that comes with this distribution. */ #define MOOEDIT_COMPILATION #ifdef HAVE_CONFIG_H #include "config.h" #endif #include "mooedit/moolangmgr-private.h" #include "mooedit/moolang-private.h" #include "mooedit/mooeditprefs.h" #include "mooutils/mooutils-misc.h" #include "mooutils/mooutils-fs.h" #include "mooutils/mooprefs.h" #include "mooutils/moomarshals.h" #include "mooutils/xdgmime/xdgmime.h" #include #define LANGUAGE_DIR "language-specs" #define ELEMENT_LANG_CONFIG MOO_EDIT_PREFS_PREFIX "/langs" #define ELEMENT_LANG "lang" #define ELEMENT_EXTENSIONS "globs" #define ELEMENT_MIME_TYPES "mime-types" #define ELEMENT_CONFIG "config" #define PROP_LANG_ID "id" #define SCHEME_DEFAULT "kate" typedef struct { MooLang *lang; GSList *globs; GSList *mime_types; guint globs_modified : 1; guint mime_types_modified : 1; } LangInfo; static LangInfo *lang_info_new (void); static void lang_info_free (LangInfo *info); static void string_list_free (GSList *list); static GSList *string_list_copy (GSList *list); static void read_langs (MooLangMgr *mgr); static void read_schemes (MooLangMgr *mgr); static void load_config (MooLangMgr *mgr); static MooLang *get_lang_for_filename (MooLangMgr *mgr, const char *filename); static MooLang *get_lang_for_mime_type (MooLangMgr *mgr, const char *mime_type); G_DEFINE_TYPE (MooLangMgr, moo_lang_mgr, GTK_TYPE_SOURCE_LANGUAGES_MANAGER) static void moo_lang_mgr_init (MooLangMgr *mgr) { char **dirs; guint n_dirs, i; GSList *list = NULL; mgr->schemes = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_object_unref); mgr->langs = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, (GDestroyNotify) lang_info_free); mgr->config = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free); dirs = moo_get_data_subdirs (LANGUAGE_DIR, MOO_DATA_SHARE, &n_dirs); for (i = 0; i < n_dirs; ++i) list = g_slist_prepend (list, dirs[i]); g_object_set (mgr, "lang-files-dirs", list, NULL); mgr->style_mgr = gtk_source_style_manager_new (); gtk_source_style_manager_set_search_path (mgr->style_mgr, dirs, n_dirs); load_config (mgr); g_slist_free (list); g_strfreev (dirs); } static void moo_lang_mgr_dispose (GObject *object) { MooLangMgr *mgr = MOO_LANG_MGR (object); if (mgr->langs) { g_object_unref (mgr->style_mgr); g_hash_table_destroy (mgr->langs); g_hash_table_destroy (mgr->config); g_hash_table_destroy (mgr->schemes); mgr->style_mgr = NULL; mgr->langs = NULL; mgr->config = NULL; mgr->active_scheme = NULL; mgr->schemes = NULL; } G_OBJECT_CLASS (moo_lang_mgr_parent_class)->dispose (object); } static void moo_lang_mgr_class_init (MooLangMgrClass *klass) { G_OBJECT_CLASS(klass)->dispose = moo_lang_mgr_dispose; _moo_signal_new_cb ("loaded", G_OBJECT_CLASS_TYPE (klass), G_SIGNAL_RUN_LAST, NULL, NULL, NULL, _moo_marshal_VOID__VOID, G_TYPE_NONE, 0); } MooLangMgr * moo_lang_mgr_new (void) { return g_object_new (MOO_TYPE_LANG_MGR, NULL); } static LangInfo * lang_info_new (void) { LangInfo *info = g_new0 (LangInfo, 1); info->lang = NULL; info->globs = NULL; info->mime_types = NULL; return info; } static void lang_info_free (LangInfo *info) { if (info) { if (info->lang) g_object_unref (info->lang); string_list_free (info->globs); string_list_free (info->mime_types); g_free (info); } } static LangInfo * get_lang_info (MooLangMgr *mgr, const char *lang_id, gboolean create) { LangInfo *info; info = g_hash_table_lookup (mgr->langs, lang_id); if (!info && create) { info = lang_info_new (); g_hash_table_insert (mgr->langs, g_strdup (lang_id), info); } return info; } static GSList * string_list_copy (GSList *list) { GSList *copy = NULL; while (list) { copy = g_slist_prepend (copy, g_strdup (list->data)); list = list->next; } return g_slist_reverse (copy); } static void string_list_free (GSList *list) { g_slist_foreach (list, (GFunc) g_free, NULL); g_slist_free (list); } MooLang * moo_lang_mgr_get_lang (MooLangMgr *mgr, const char *name) { MooLang *lang; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); if (!name) return NULL; lang = _moo_lang_mgr_find_lang (mgr, name); if (!lang) g_warning ("could not find language '%s'", name); return lang; } MooLang * _moo_lang_mgr_find_lang (MooLangMgr *mgr, const char *name) { char *id; LangInfo *info; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); g_return_val_if_fail (name != NULL, NULL); if (!strcmp (name, MOO_LANG_NONE)) return NULL; if (!g_ascii_strcasecmp (name, "c++")) name = "cpp"; read_langs (mgr); id = _moo_lang_id_from_name (name); info = get_lang_info (mgr, id, FALSE); g_free (id); return info ? info->lang : NULL; } static MooLang * get_lang_by_extension (MooLangMgr *mgr, const char *filename) { MooLang *lang = NULL; char *basename, *utf8_basename; GSList *langs, *l; gboolean found = FALSE; g_return_val_if_fail (filename != NULL, NULL); read_langs (mgr); /* TODO: is this right? */ basename = g_path_get_basename (filename); g_return_val_if_fail (basename != NULL, NULL); utf8_basename = g_filename_display_name (basename); g_return_val_if_fail (utf8_basename != NULL, NULL); langs = moo_lang_mgr_get_available_langs (mgr); for (l = langs; !found && l != NULL; l = l->next) { GSList *g, *globs; lang = l->data; globs = _moo_lang_mgr_get_globs (mgr, _moo_lang_id (lang)); for (g = globs; !found && g != NULL; g = g->next) { if (_moo_glob_match_simple ((char*) g->data, utf8_basename)) { found = TRUE; break; } } string_list_free (globs); } if (!found) lang = NULL; g_slist_foreach (langs, (GFunc) g_object_unref, NULL); g_slist_free (langs); g_free (utf8_basename); g_free (basename); return lang; } static MooLang * lang_mgr_get_lang_for_bak_filename (MooLangMgr *mgr, const char *filename) { MooLang *lang = NULL; char *utf8_name, *utf8_base = NULL; int len; guint i; static const char *bak_globs[] = {"*~", "*.bak", "*.in", "*.orig"}; read_langs (mgr); utf8_name = g_filename_to_utf8 (filename, -1, NULL, NULL, NULL); g_return_val_if_fail (utf8_name != NULL, NULL); len = strlen (utf8_name); for (i = 0; i < G_N_ELEMENTS (bak_globs); ++i) { int ext_len = strlen (bak_globs[i]) - 1; if (len > ext_len && _moo_glob_match_simple (bak_globs[i], utf8_name)) { utf8_base = g_strndup (utf8_name, len - ext_len); break; } } if (utf8_base) { char *base = g_filename_from_utf8 (utf8_base, -1, NULL, NULL, NULL); if (base) lang = get_lang_for_filename (mgr, base); g_free (base); g_free (utf8_base); } g_free (utf8_name); return lang; } static gboolean filename_blacklisted (MooLangMgr *mgr, const char *filename) { /* XXX bak files */ char *basename, *utf8_basename; gboolean result = FALSE; LangInfo *info; GSList *l; read_langs (mgr); basename = g_path_get_basename (filename); g_return_val_if_fail (basename != NULL, FALSE); utf8_basename = g_filename_display_name (basename); g_return_val_if_fail (utf8_basename != NULL, FALSE); info = get_lang_info (mgr, MOO_LANG_NONE, FALSE); if (info) for (l = info->globs; !result && l != NULL; l = l->next) if (_moo_glob_match_simple ((char*) l->data, utf8_basename)) result = TRUE; g_free (utf8_basename); g_free (basename); return result; } static gboolean file_blacklisted (MooLangMgr *mgr, const char *filename) { /* XXX mime type */ return filename_blacklisted (mgr, filename); } MooLang * moo_lang_mgr_get_lang_for_file (MooLangMgr *mgr, const char *filename) { MooLang *lang = NULL; const char *mime_type; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); g_return_val_if_fail (filename != NULL, NULL); read_langs (mgr); if (file_blacklisted (mgr, filename)) return NULL; #ifdef MOO_USE_XDGMIME /* XXX: xdgmime wants utf8-encoded filename here. is it a problem? */ /* It's a big problem! */ mime_type = xdg_mime_get_mime_type_for_file (filename, NULL); if (mime_type != XDG_MIME_TYPE_UNKNOWN) lang = get_lang_for_mime_type (mgr, mime_type); if (lang) return lang; #else #ifdef __GNUC__ #warning "Implement moo_lang_mgr_get_lang_for_file()" #endif #endif /* MOO_USE_XDGMIME */ lang = get_lang_by_extension (mgr, filename); if (lang) return lang; lang = lang_mgr_get_lang_for_bak_filename (mgr, filename); if (lang) return lang; return NULL; } static MooLang * get_lang_for_filename (MooLangMgr *mgr, const char *filename) { MooLang *lang = NULL; const char *mime_type; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); g_return_val_if_fail (filename != NULL, NULL); read_langs (mgr); if (filename_blacklisted (mgr, filename)) return NULL; lang = get_lang_by_extension (mgr, filename); if (lang) return lang; #ifdef MOO_USE_XDGMIME /* XXX: xdgmime wants utf8-encoded filename here. is it a problem? */ /* It's a big problem! */ mime_type = xdg_mime_get_mime_type_from_file_name (filename); if (mime_type != XDG_MIME_TYPE_UNKNOWN) lang = get_lang_for_mime_type (mgr, mime_type); if (lang) return lang; #else #ifdef __GNUC__ #warning "Implement moo_lang_mgr_get_lang_for_filename()" #endif #endif /* MOO_USE_XDGMIME */ lang = lang_mgr_get_lang_for_bak_filename (mgr, filename); if (lang) return lang; return NULL; } #ifdef MOO_USE_XDGMIME static int check_mime_subclass (const char *base, const char *mime) { return !xdg_mime_mime_type_subclass (mime, base); } static MooLang * get_lang_for_mime_type (MooLangMgr *mgr, const char *mime) { GSList *l, *langs; MooLang *lang = NULL; gboolean found = FALSE; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); g_return_val_if_fail (mime != NULL, NULL); langs = moo_lang_mgr_get_available_langs (mgr); for (l = langs; !found && l != NULL; l = l->next) { GSList *mimetypes; lang = l->data; mimetypes = _moo_lang_mgr_get_mime_types (mgr, _moo_lang_id (lang)); if (g_slist_find_custom (mimetypes, mime, (GCompareFunc) strcmp)) found = TRUE; string_list_free (mimetypes); } for (l = langs; !found && l != NULL; l = l->next) { GSList *mimetypes; lang = l->data; mimetypes = _moo_lang_mgr_get_mime_types (mgr, _moo_lang_id (lang)); if (g_slist_find_custom (mimetypes, mime, (GCompareFunc) check_mime_subclass)) found = TRUE; string_list_free (mimetypes); } g_slist_foreach (langs, (GFunc) g_object_unref, NULL); g_slist_free (langs); return found ? lang : NULL; } #else /* MOO_USE_XDGMIME */ static MooLang * get_lang_for_mime_type (MooLangMgr *mgr, const char *mime) { g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); g_return_val_if_fail (mime != NULL, NULL); g_warning ("%s: implement me?", G_STRLOC); return NULL; } #endif /* MOO_USE_XDGMIME */ static void read_langs (MooLangMgr *mgr) { const GSList *langs = NULL; if (mgr->got_langs) return; read_schemes (mgr); mgr->got_langs = TRUE; #ifdef MOO_USE_XML langs = gtk_source_languages_manager_get_available_languages (GTK_SOURCE_LANGUAGES_MANAGER (mgr)); #endif get_lang_info (mgr, MOO_LANG_NONE, TRUE); while (langs) { LangInfo *info; MooLang *lang = langs->data; const char *id = _moo_lang_id (lang); info = get_lang_info (mgr, id, TRUE); info->lang = g_object_ref (lang); if (!info->globs_modified) info->globs = _moo_lang_get_globs (lang); if (!info->mime_types_modified) info->mime_types = _moo_lang_get_mime_types (lang); langs = langs->next; } g_signal_emit_by_name (mgr, "loaded"); } static void set_list (MooLangMgr *mgr, const char *lang_id, const char *string, gboolean globs) { GSList **ptr; LangInfo *info; info = get_lang_info (mgr, lang_id, TRUE); ptr = globs ? &info->globs : &info->mime_types; if (*ptr) string_list_free (*ptr); *ptr = _moo_lang_parse_string_list (string); if (globs) info->globs_modified = TRUE; else info->mime_types_modified = TRUE; } static void load_lang_node (MooLangMgr *mgr, MooMarkupNode *lang_node) { const char *lang_id; MooMarkupNode *node; lang_id = moo_markup_get_prop (lang_node, PROP_LANG_ID); g_return_if_fail (lang_id != NULL); for (node = lang_node->children; node != NULL; node = node->next) { if (!MOO_MARKUP_IS_ELEMENT (node)) continue; if (!strcmp (node->name, ELEMENT_EXTENSIONS)) { set_list (mgr, lang_id, moo_markup_get_content (node), TRUE); } else if (!strcmp (node->name, ELEMENT_MIME_TYPES)) { set_list (mgr, lang_id, moo_markup_get_content (node), FALSE); } else if (!strcmp (node->name, ELEMENT_CONFIG)) { const char *string = moo_markup_get_content (node); g_hash_table_insert (mgr->config, g_strdup (lang_id), g_strdup (string)); } else { /* 'extensions' is leftover from previous versions */ if (strcmp (node->name, "extensions") != 0) g_warning ("%s: ignoring node '%s'", G_STRLOC, node->name); } } } static void set_default_config (MooLangMgr *mgr) { _moo_lang_mgr_set_config (mgr, "makefile", "use-tabs: true"); } static void load_config (MooLangMgr *mgr) { MooMarkupDoc *xml; MooMarkupNode *root, *node; set_default_config (mgr); mgr->modified = FALSE; xml = moo_prefs_get_markup (MOO_PREFS_RC); g_return_if_fail (xml != NULL); root = moo_markup_get_element (MOO_MARKUP_NODE (xml), ELEMENT_LANG_CONFIG); if (!root) return; for (node = root->children; node != NULL; node = node->next) { if (!MOO_MARKUP_IS_ELEMENT (node)) continue; if (strcmp (node->name, ELEMENT_LANG)) { g_warning ("%s: invalid '%s' element", G_STRLOC, node->name); continue; } load_lang_node (mgr, node); } mgr->modified = FALSE; } MooTextStyleScheme * _moo_lang_mgr_get_active_scheme (MooLangMgr *mgr) { g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_schemes (mgr); return mgr->active_scheme; } void _moo_lang_mgr_set_active_scheme (MooLangMgr *mgr, const char *name) { MooTextStyleScheme *scheme = NULL; g_return_if_fail (MOO_IS_LANG_MGR (mgr)); read_schemes (mgr); if (name) scheme = g_hash_table_lookup (mgr->schemes, name); if (!scheme) scheme = g_hash_table_lookup (mgr->schemes, SCHEME_DEFAULT); if (!scheme) g_warning ("%s: could not find style scheme '%s'", G_STRLOC, name ? name : ""); else mgr->active_scheme = scheme; } static void prepend_scheme (G_GNUC_UNUSED const char *name, MooTextStyleScheme *scheme, GSList **list) { *list = g_slist_prepend (*list, g_object_ref (scheme)); } GSList * _moo_lang_mgr_list_schemes (MooLangMgr *mgr) { GSList *list = NULL; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_schemes (mgr); g_hash_table_foreach (mgr->schemes, (GHFunc) prepend_scheme, &list); return list; } static void read_schemes (MooLangMgr *mgr) { const GSList *list; if (mgr->got_schemes) return; mgr->got_schemes = TRUE; list = gtk_source_style_manager_list_schemes (mgr->style_mgr); while (list) { MooTextStyleScheme *scheme = list->data; if (!mgr->active_scheme || !strcmp (moo_text_style_scheme_get_id (scheme), SCHEME_DEFAULT)) mgr->active_scheme = scheme; g_hash_table_insert (mgr->schemes, g_strdup (moo_text_style_scheme_get_id (scheme)), g_object_ref (scheme)); list = list->next; } } GSList * moo_lang_mgr_get_sections (MooLangMgr *mgr) { GSList *sections = NULL; const GSList *list = NULL; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_langs (mgr); #ifdef MOO_USE_XML list = gtk_source_languages_manager_get_available_languages (GTK_SOURCE_LANGUAGES_MANAGER (mgr)); #endif while (list) { const char *section = _moo_lang_get_section (list->data); if (section && !g_slist_find_custom (sections, section, (GCompareFunc) strcmp)) sections = g_slist_prepend (sections, g_strdup (section)); list = list->next; } return sections; } GSList * moo_lang_mgr_get_available_langs (MooLangMgr *mgr) { GSList *langs = NULL; const GSList *list = NULL; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_langs (mgr); #ifdef MOO_USE_XML list = gtk_source_languages_manager_get_available_languages (GTK_SOURCE_LANGUAGES_MANAGER (mgr)); #endif while (list) { langs = g_slist_prepend (langs, g_object_ref (list->data)); list = list->next; } return g_slist_reverse (langs); } GSList * _moo_lang_mgr_get_globs (MooLangMgr *mgr, const char *lang_id) { char *id; LangInfo *info; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_langs (mgr); id = _moo_lang_id_from_name (lang_id); info = get_lang_info (mgr, id, FALSE); g_return_val_if_fail (info != NULL, NULL); g_free (id); return string_list_copy (info->globs); } GSList * _moo_lang_mgr_get_mime_types (MooLangMgr *mgr, const char *lang_id) { char *id; LangInfo *info; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_langs (mgr); id = _moo_lang_id_from_name (lang_id); info = get_lang_info (mgr, id, FALSE); g_return_val_if_fail (info != NULL, NULL); g_free (id); return string_list_copy (info->mime_types); } static gboolean string_list_equal (GSList *list1, GSList *list2) { while (list1 && list2) { if (strcmp (list1->data, list2->data) != 0) return FALSE; list1 = list1->next; list2 = list2->next; } return list1 == list2; } static void set_globs_or_mime_types (MooLangMgr *mgr, const char *lang_id, const char *string, gboolean globs) { LangInfo *info; GSList *new; GSList *old = NULL; GSList **ptr; char *id = NULL; gboolean modified = TRUE; read_langs (mgr); id = _moo_lang_id_from_name (lang_id); info = get_lang_info (mgr, id, FALSE); g_return_if_fail (info != NULL); new = _moo_lang_parse_string_list (string); old = globs ? _moo_lang_mgr_get_globs (mgr, id) : _moo_lang_mgr_get_mime_types (mgr, id); if (string_list_equal (new, old)) goto out; mgr->modified = TRUE; if (info->lang) { GSList *builtin = globs ? _moo_lang_get_globs (info->lang) : _moo_lang_get_mime_types (info->lang); modified = !string_list_equal (builtin, new); if (modified) { string_list_free (builtin); } else { string_list_free (new); new = builtin; } } ptr = globs ? &info->globs : &info->mime_types; string_list_free (*ptr); *ptr = new; if (globs) info->globs_modified = modified; else info->mime_types_modified = modified; out: g_free (id); string_list_free (old); } void _moo_lang_mgr_set_globs (MooLangMgr *mgr, const char *lang_id, const char *globs) { g_return_if_fail (MOO_IS_LANG_MGR (mgr)); set_globs_or_mime_types (mgr, lang_id, globs, TRUE); } void _moo_lang_mgr_set_mime_types (MooLangMgr *mgr, const char *lang_id, const char *mime) { g_return_if_fail (MOO_IS_LANG_MGR (mgr)); set_globs_or_mime_types (mgr, lang_id, mime, FALSE); } const char * _moo_lang_mgr_get_config (MooLangMgr *mgr, const char *lang_id) { char *id; const char *config; g_return_val_if_fail (MOO_IS_LANG_MGR (mgr), NULL); read_langs (mgr); id = _moo_lang_id_from_name (lang_id); config = g_hash_table_lookup (mgr->config, id); g_free (id); return config; } void _moo_lang_mgr_set_config (MooLangMgr *mgr, const char *lang_id, const char *config) { char *id; char *norm = NULL; const char *old; g_return_if_fail (MOO_IS_LANG_MGR (mgr)); read_langs (mgr); if (config && config[0]) { norm = g_strstrip (g_strdup (config)); if (!norm[0]) { g_free (norm); norm = NULL; } } id = _moo_lang_id_from_name (lang_id); old = g_hash_table_lookup (mgr->config, id); if (!_moo_str_equal (old, norm)) mgr->modified = TRUE; g_hash_table_insert (mgr->config, id, norm); } void _moo_lang_mgr_update_config (MooLangMgr *mgr, MooEditConfig *config, const char *lang_id) { const char *lang_config; g_return_if_fail (MOO_IS_LANG_MGR (mgr)); g_return_if_fail (MOO_IS_EDIT_CONFIG (config)); read_langs (mgr); g_object_freeze_notify (G_OBJECT (config)); moo_edit_config_unset_by_source (config, MOO_EDIT_CONFIG_SOURCE_LANG); lang_config = _moo_lang_mgr_get_config (mgr, lang_id); if (lang_config) moo_edit_config_parse (config, lang_config, MOO_EDIT_CONFIG_SOURCE_LANG); g_object_thaw_notify (G_OBJECT (config)); } static char * list_to_string (GSList *list) { GString *string = g_string_new (NULL); while (list) { g_string_append (string, list->data); if ((list = list->next)) g_string_append_c (string, ';'); } return g_string_free (string, FALSE); } static void save_one (const char *lang_id, LangInfo *info, gpointer user_data) { const char *config; MooMarkupNode *lang_node; struct { MooLangMgr *mgr; MooMarkupDoc *xml; MooMarkupNode *root; } *data; data = user_data; config = g_hash_table_lookup (data->mgr->config, lang_id); if (!config && !info->globs_modified && !info->mime_types_modified) return; if (!data->root) data->root = moo_markup_create_element (MOO_MARKUP_NODE (data->xml), ELEMENT_LANG_CONFIG); lang_node = moo_markup_create_element (data->root, ELEMENT_LANG); moo_markup_set_prop (lang_node, PROP_LANG_ID, lang_id); if (info->globs_modified) { char *string = list_to_string (info->globs); moo_markup_create_text_element (lang_node, ELEMENT_EXTENSIONS, string ? string : ""); g_free (string); } if (info->mime_types_modified) { char *string = list_to_string (info->mime_types); moo_markup_create_text_element (lang_node, ELEMENT_EXTENSIONS, string ? string : ""); g_free (string); } if (config) moo_markup_create_text_element (lang_node, ELEMENT_CONFIG, config); } void _moo_lang_mgr_save_config (MooLangMgr *mgr) { MooMarkupDoc *xml; MooMarkupNode *root; struct { MooLangMgr *mgr; MooMarkupDoc *xml; MooMarkupNode *root; } data; g_return_if_fail (MOO_IS_LANG_MGR (mgr)); if (!mgr->modified) return; xml = moo_prefs_get_markup (MOO_PREFS_RC); g_return_if_fail (xml != NULL); mgr->modified = FALSE; root = moo_markup_get_element (MOO_MARKUP_NODE (xml), ELEMENT_LANG_CONFIG); if (root) moo_markup_delete_node (root); root = NULL; data.mgr = mgr; data.xml = xml; data.root = NULL; g_hash_table_foreach (mgr->langs, (GHFunc) save_one, &data); }