36 #include "gnc-prefs-p.h" 38 #include <boost/property_tree/ptree.hpp> 39 #include <boost/property_tree/xml_parser.hpp> 42 #include <unordered_map> 44 namespace bpt = boost::property_tree;
46 #define GSET_SCHEMA_PREFIX "org.gnucash.GnuCash" 47 #define GSET_SCHEMA_OLD_PREFIX "org.gnucash" 51 void operator()(GSettings* gsp)
59 using GSettingsPtr = std::unique_ptr<GSettings, GSettingsDeleter>;
61 static std::unordered_map<std::string,GSettingsPtr> schema_hash;
64 static QofLogModule log_module =
"gnc.app-utils.gsettings";
69 static bool gnc_gsettings_is_valid_key(GSettings *settings,
const gchar *key)
72 if (!G_IS_SETTINGS(settings))
75 GSettingsSchema *schema;
76 g_object_get (settings,
"settings-schema", &schema,
nullptr);
80 auto keys = g_settings_schema_list_keys (schema);
81 auto found = (keys && g_strv_contains(keys, key));
83 g_settings_schema_unref (schema);
89 normalize_schema_name (
const gchar *name)
92 return GSET_SCHEMA_PREFIX;
94 if (g_str_has_prefix (name, GSET_SCHEMA_PREFIX) ||
95 (g_str_has_prefix (name, GSET_SCHEMA_OLD_PREFIX)))
98 return std::string{GSET_SCHEMA_PREFIX} +
'.' + name;
101 static GSettings * gnc_gsettings_get_settings_obj (
const gchar *schema_str)
105 auto full_name_str = normalize_schema_name (schema_str);
106 auto full_name = full_name_str.c_str();
107 auto schema_source {g_settings_schema_source_get_default()};
108 auto schema {g_settings_schema_source_lookup(schema_source, full_name,
true)};
109 auto gset = g_settings_new_full (schema,
nullptr,
nullptr);
110 DEBUG (
"Created gsettings object %p for schema %s", gset, full_name);
112 if (!G_IS_SETTINGS(gset))
113 PWARN (
"Ignoring attempt to access unknown gsettings schema %s", full_name);
116 g_settings_schema_unref (schema);
121 schema_to_gsettings (
const char *schema,
bool can_retrieve)
123 auto full_name = normalize_schema_name (schema);
124 auto iter = schema_hash.find (full_name);
125 if (iter != schema_hash.end())
126 return iter->second.get();
131 auto gs_obj = gnc_gsettings_get_settings_obj (schema);
132 if (!G_IS_SETTINGS (gs_obj))
134 PWARN (
"Ignoring attempt to access unknown gsettings schema %s", full_name.c_str());
138 schema_hash[full_name] = GSettingsPtr (gs_obj, g_settings_deleter);
149 return GSET_SCHEMA_PREFIX;
162 g_return_val_if_fail (func, 0);
164 auto gs_obj = schema_to_gsettings (schema,
true);
165 g_return_val_if_fail (G_IS_SETTINGS (gs_obj), 0);
167 auto signal =
static_cast<char *
> (
nullptr);
169 signal = g_strdup (
"changed");
170 else if (gnc_gsettings_is_valid_key(gs_obj, key))
171 signal = g_strconcat (
"changed::", key,
nullptr);
173 auto handlerid = g_signal_connect (gs_obj, signal, G_CALLBACK (func), user_data);
176 g_object_ref (gs_obj);
178 PINFO(
"schema: %s, key: %s, gs_obj: %p, handler_id: %ld",
179 schema, key, gs_obj, handlerid);
189 gnc_gsettings_remove_cb_by_id_internal (GSettings *gs_obj, guint handlerid)
192 g_return_if_fail (G_IS_SETTINGS (gs_obj));
194 g_signal_handler_disconnect (gs_obj, handlerid);
195 g_object_unref (gs_obj);
197 LEAVE (
"Schema: %p, handlerid: %d - removed for handler",
204 gpointer func, gpointer user_data)
207 g_return_if_fail (func);
209 auto gs_obj = schema_to_gsettings (schema,
false);
211 if (!G_IS_SETTINGS (gs_obj))
213 LEAVE (
"No valid GSettings object retrieved from hash table");
217 auto match_type =
static_cast<GSignalMatchType
> (G_SIGNAL_MATCH_DETAIL | G_SIGNAL_MATCH_FUNC | G_SIGNAL_MATCH_DATA);
218 auto changed_signal = g_signal_lookup (
"changed", G_TYPE_SETTINGS);
219 auto quark = g_quark_from_string (key);
222 guint handler_id = 0;
225 handler_id = g_signal_handler_find (gs_obj, match_type,
226 changed_signal, quark,
nullptr,
231 gnc_gsettings_remove_cb_by_id_internal (gs_obj, handler_id);
234 if (!G_IS_SETTINGS (gs_obj))
237 }
while (handler_id);
239 LEAVE (
"Schema: %s, key: %s - removed %d handlers for 'changed' signal",
240 schema, key, matched);
249 auto gs_obj = schema_to_gsettings (schema,
false);
251 if (!G_IS_SETTINGS (gs_obj))
253 LEAVE (
"No valid GSettings object retrieved from hash table");
257 gnc_gsettings_remove_cb_by_id_internal (gs_obj, handlerid);
259 LEAVE (
"Schema: %p, handlerid: %d - removed for handler",
282 static gboolean gnc_gsettings_enum_bool_mapping_get (GValue *value,
286 g_value_set_boolean (value,
287 !g_strcmp0 ((
const gchar *)user_data,
288 g_variant_get_string (variant,
nullptr)));
293 static GVariant* gnc_gsettings_enum_bool_mapping_set (
const GValue *value,
294 const GVariantType *expected_type,
297 if (g_value_get_boolean (value))
299 return g_variant_new_string ((
const gchar *)user_data);
313 const gchar *property)
315 auto gs_obj = gnc_gsettings_get_settings_obj (schema);
316 g_return_if_fail (G_IS_SETTINGS (gs_obj));
318 if (gnc_gsettings_is_valid_key (gs_obj, key))
322 g_settings_bind_with_mapping (gs_obj, key,
object, property,
323 G_SETTINGS_BIND_DEFAULT,
324 gnc_gsettings_enum_bool_mapping_get,
325 gnc_gsettings_enum_bool_mapping_set,
326 g_strdup (value), g_free);
330 g_settings_bind (gs_obj, key,
object, property, G_SETTINGS_BIND_DEFAULT);
335 PERR (
"Invalid key %s for schema %s", key, schema);
341 gs_obj_block_handlers ([[maybe_unused]] gpointer key, gpointer gs_obj,
342 [[maybe_unused]] gpointer pointer)
344 g_signal_handlers_block_matched (gs_obj, G_SIGNAL_MATCH_CLOSURE, 0, 0,
nullptr,
nullptr,
nullptr);
345 PINFO(
"Block all handlers for GSettings object %p", gs_obj);
349 gs_obj_unblock_handlers ([[maybe_unused]] gpointer key, gpointer gs_obj,
350 [[maybe_unused]] gpointer pointer)
352 g_signal_handlers_unblock_matched (gs_obj, G_SIGNAL_MATCH_CLOSURE, 0, 0,
nullptr,
nullptr,
nullptr);
353 PINFO(
"Unblock all handlers for GSettings object %p", gs_obj);
359 for (
const auto& it : schema_hash)
360 gs_obj_block_handlers (
nullptr, it.second.get(),
nullptr);
368 for (
const auto& it : schema_hash)
369 gs_obj_unblock_handlers (
nullptr, it.second.get(),
nullptr);
378 T gnc_gsettings_get(
const char *schema,
const char *key,
379 auto getter(GSettings*,
const char *)->T, T default_val)
381 auto gs_obj = gnc_gsettings_get_settings_obj (schema);
382 g_return_val_if_fail (G_IS_SETTINGS (gs_obj), default_val);
385 if (gnc_gsettings_is_valid_key (gs_obj, key))
386 val = getter (gs_obj, key);
388 PERR (
"Invalid key %s for schema %s", key, schema);
390 g_object_unref (gs_obj);
397 return gnc_gsettings_get (schema, key, g_settings_get_boolean,
398 static_cast<gboolean>(
false));
404 return gnc_gsettings_get (schema, key, g_settings_get_int, 0);
410 return gnc_gsettings_get (schema, key, g_settings_get_double, 0.0);
416 return gnc_gsettings_get (schema, key, g_settings_get_string,
417 static_cast<gchar *> (
nullptr));
423 return gnc_gsettings_get (schema, key, g_settings_get_enum, 0);
429 return gnc_gsettings_get (schema, key, g_settings_get_value,
430 static_cast<GVariant *> (
nullptr));
436 template<
typename T> gboolean
437 gnc_gsettings_set (
const gchar *schema,
440 gboolean setter(GSettings*,
const char *, T))
442 ENTER(
"schema: %s, key: %s", schema, key);
444 auto gs_obj = gnc_gsettings_get_settings_obj (schema);
445 g_return_val_if_fail (G_IS_SETTINGS (gs_obj),
false);
448 if (gnc_gsettings_is_valid_key (gs_obj, key))
450 result = setter (gs_obj, key, value);
452 PERR (
"Unable to set value for key %s in schema %s", key, schema);
455 PERR (
"Invalid key %s for schema %s", key, schema);
457 g_object_unref (gs_obj);
458 LEAVE(
"result %i", result);
465 return gnc_gsettings_set (schema, key, value, g_settings_set_boolean);
471 return gnc_gsettings_set (schema, key, value, g_settings_set_int);
477 return gnc_gsettings_set (schema, key, value, g_settings_set_double);
483 return gnc_gsettings_set (schema, key, value, g_settings_set_string);
489 return gnc_gsettings_set (schema, key, value, g_settings_set_enum);
495 return gnc_gsettings_set (schema, key, value, g_settings_set_value);
502 auto gs_obj = gnc_gsettings_get_settings_obj (schema);
503 g_return_if_fail (G_IS_SETTINGS (gs_obj));
505 if (gnc_gsettings_is_valid_key (gs_obj, key))
506 g_settings_reset (gs_obj, key);
508 PERR (
"Invalid key %s for schema %s", key, schema);
510 g_object_unref (gs_obj);
516 auto gs_obj = gnc_gsettings_get_settings_obj (schema_str);
521 GSettingsSchema *schema;
522 g_object_get (gs_obj,
"settings-schema", &schema,
nullptr);
525 g_object_unref (gs_obj);
529 auto keys = g_settings_schema_list_keys (schema);
533 for (
auto key = *fkeys; key; key = *++fkeys)
537 g_object_unref (gs_obj);
538 g_settings_schema_unref (schema);
543 gnc_settings_dump_schema_paths (
void)
545 gchar **non_relocatable;
547 auto schema_source {g_settings_schema_source_get_default()};
548 g_settings_schema_source_list_schemas (schema_source,
true,
549 &non_relocatable,
nullptr);
551 for (gint i = 0; non_relocatable[i] !=
nullptr; i++)
552 PINFO(
"Schema entry %d is '%s'", i, non_relocatable[i]);
554 g_strfreev (non_relocatable);
565 if (g_strcmp0 (g_getenv (
"GNC_UNINSTALLED"),
"1") == 0)
568 g_free (prefsbackend);
595 gnc_settings_dump_schema_paths ();
601 LEAVE(
"Prefsbackend bind = %p", prefsbackend->bind);
608 g_free (prefsbackend);
613 gnc_gsettings_get_user_value (
const gchar *schema,
616 auto gs_obj = gnc_gsettings_get_settings_obj (schema);
617 g_return_val_if_fail (G_IS_SETTINGS (gs_obj),
nullptr);
619 auto val =
static_cast<GVariant *
> (
nullptr);
620 if (gnc_gsettings_is_valid_key (gs_obj, key))
621 val = g_settings_get_user_value (gs_obj, key);
623 PERR (
"Invalid key %s for schema %s", key, schema);
625 g_object_unref (gs_obj);
629 using opt_str_vec = boost::optional<std::string>;
632 deprecate_one_key (
const opt_str_vec &oldpath,
const opt_str_vec &oldkey)
634 if (!oldpath || !oldkey )
636 DEBUG (
"Skipping <deprecate> node - missing attribute (old-path or old-key)");
640 PINFO (
"'%s:%s' has been marked deprecated", oldpath->c_str(), oldkey->c_str());
646 migrate_one_key (
const opt_str_vec &oldpath,
const opt_str_vec &oldkey,
647 const opt_str_vec &newpath,
const opt_str_vec &newkey)
649 if (!oldpath || !oldkey || !newpath || !newkey)
651 DEBUG (
"Skipping <migrate> node - missing attribute (old-path, old-key, new-path or new-key)");
655 PINFO (
"Migrating '%s:%s' to '%s:%s'", oldpath->c_str(), oldkey->c_str(),
656 newpath->c_str(), newkey->c_str());
658 auto user_value = gnc_gsettings_get_user_value (oldpath->c_str(), oldkey->c_str());
664 obsolete_one_key (
const opt_str_vec &oldpath,
const opt_str_vec &oldkey)
666 if (!oldpath || !oldkey )
668 DEBUG (
"Skipping <obsolete> node - missing attribute (old-path or old-key)");
672 PINFO (
"Resetting obsolete '%s:%s'", oldpath->c_str(), oldkey->c_str());
677 parse_one_release_node (bpt::ptree &pt)
680 std::for_each (pt.begin(), pt.end(),
681 [] (std::pair<bpt::ptree::key_type, bpt::ptree> node)
683 if (node.first ==
"<xmlattr>")
685 else if (node.first ==
"deprecate")
686 deprecate_one_key (node.second.get_optional<std::string> (
"<xmlattr>.old-path"),
687 node.second.get_optional<std::string> (
"<xmlattr>.old-key"));
688 else if (node.first ==
"migrate")
689 migrate_one_key (node.second.get_optional<std::string> (
"<xmlattr>.old-path"),
690 node.second.get_optional<std::string> (
"<xmlattr>.old-key"),
691 node.second.get_optional<std::string> (
"<xmlattr>.new-path"),
692 node.second.get_optional<std::string> (
"<xmlattr>.new-key"));
693 else if (node.first ==
"obsolete")
694 obsolete_one_key (node.second.get_optional<std::string> (
"<xmlattr>.old-path"),
695 node.second.get_optional<std::string> (
"<xmlattr>.old-key"));
698 DEBUG (
"Skipping unknown node <%s>", node.first.c_str());
705 transform_settings (
int old_maj_min,
int cur_maj_min)
709 auto pkg_data_dir = gnc_path_get_pkgdatadir();
710 auto transform_file = std::string (pkg_data_dir) +
"/pref_transformations.xml";
711 g_free (pkg_data_dir);
713 std::ifstream transform_stream {transform_file};
714 if (!transform_stream.is_open())
716 PWARN(
"Failed to load preferences transformation file '%s'", transform_file.c_str());
722 bpt::read_xml (transform_stream, pt);
724 catch (bpt::xml_parser_error &e) {
725 PWARN (
"Failed to parse GnuCash preferences transformation file.\n");
726 PWARN (
"Error message:\n");
727 PWARN (
"%s\n", e.what());
731 PWARN (
"Unknown error while parsing GnuCash preferences transformation file.\n");
736 std::for_each (pt.begin(), pt.end(),
737 [&old_maj_min, &cur_maj_min] (std::pair<bpt::ptree::key_type, bpt::ptree> node)
739 if (node.first !=
"release")
741 DEBUG (
"Skipping non-<release> node <%s>", node.first.c_str());
744 auto version = node.second.get_optional<
int> (
"<xmlattr>.version");
747 DEBUG (
"Skipping <release> node - no version attribute found");
750 if (*version <= old_maj_min)
752 DEBUG (
"Skipping <release> node - version %i is less than current compatibility level %i", *version, old_maj_min);
755 if (*version > cur_maj_min)
757 DEBUG (
"Skipping <release> node - version %i is greater than current version level %i", *version, cur_maj_min);
760 DEBUG (
"Retrieved version value '%i'", *version);
762 parse_one_release_node (node.second);
791 ENTER(
"Start of settings transform routine.");
793 auto ogG_maj_min = gnc_gsettings_get_user_value (GNC_PREFS_GROUP_GENERAL, GNC_PREF_VERSION);
794 auto og_maj_min = gnc_gsettings_get_user_value (GSET_SCHEMA_OLD_PREFIX
"." GNC_PREFS_GROUP_GENERAL, GNC_PREF_VERSION);
796 auto cur_maj_min = PROJECT_VERSION_MAJOR * 1000 + PROJECT_VERSION_MINOR;
798 if (!ogG_maj_min && !og_maj_min)
801 LEAVE (
"Setting Previous compatibility level to current version: %i", cur_maj_min);
805 auto old_maj_min = 0;
807 old_maj_min =
gnc_gsettings_get_int (GSET_SCHEMA_OLD_PREFIX
"." GNC_PREFS_GROUP_GENERAL, GNC_PREF_VERSION);
810 g_variant_unref (ogG_maj_min);
814 g_variant_unref (og_maj_min);
816 PINFO (
"Previous setting compatibility level: %i, Current version: %i", old_maj_min, cur_maj_min);
818 transform_settings (old_maj_min, cur_maj_min);
821 if (cur_maj_min > old_maj_min)
gboolean gnc_gsettings_set_int(const gchar *schema, const gchar *key, gint value)
Store an integer value into GSettings.
gboolean gnc_gsettings_set_float(const gchar *schema, const gchar *key, gdouble value)
Store a float value into GSettings.
#define PINFO(format, args...)
Print an informational note.
gint gnc_gsettings_get_int(const gchar *schema, const gchar *key)
Get an integer value from GSettings.
#define DEBUG(format, args...)
Print a debugging message.
#define PERR(format, args...)
Log a serious error.
#define ENTER(format, args...)
Print a function entry debugging message.
void gnc_gsettings_remove_any_cb_by_func(const gchar *schema, gpointer func, gpointer user_data)
Remove a function that was registered for a callback when any key in the given settings schema change...
gboolean gnc_gsettings_set_enum(const gchar *schema, const gchar *key, gint value)
Store an enum value into GSettings.
#define PWARN(format, args...)
Log a warning.
gboolean qof_log_check(QofLogModule domain, QofLogLevel level)
Check to see if the given log_module is configured to log at the given log_level. ...
GVariant * gnc_gsettings_get_value(const gchar *schema, const gchar *key)
Get an arbitrary combination of values from GSettings.
guint gnc_gsettings_register_any_cb(const gchar *schema, gpointer func, gpointer user_data)
Register a callback for when any key in the settings schema is changed.
void gnc_gsettings_unblock_all(void)
UnBlock all prefs callbacks, used while preference dialog is loaded.
void gnc_gsettings_reset_schema(const gchar *schema_str)
Reset all keys in a schema to their default values in GSettings.
void gnc_gsettings_shutdown(void)
Free the GSettings resources.
gdouble gnc_gsettings_get_float(const gchar *schema, const gchar *key)
Get an float value from GSettings.
void gnc_gsettings_reset(const gchar *schema, const gchar *key)
Reset a key to its default value in GSettings.
gchar * gnc_gsettings_get_string(const gchar *schema, const gchar *key)
Get a string value from GSettings.
void gnc_gsettings_bind(const gchar *schema, const gchar *key, const gchar *value, gpointer object, const gchar *property)
Bind a setting to a g_object property.
gint gnc_gsettings_get_enum(const gchar *schema, const gchar *key)
Get an enum value from GSettings.
gboolean gnc_gsettings_set_bool(const gchar *schema, const gchar *key, gboolean value)
Store a boolean value into GSettings.
void gnc_gsettings_load_backend(void)
Configure gsettings as the backend for the gnucash preferences api.
void gnc_gsettings_version_upgrade(void)
Check whether we need to adjust the user settings to a newer version.
void gnc_gsettings_block_all(void)
Block all prefs callbacks, used while preference dialog is loaded.
#define LEAVE(format, args...)
Print a function exit debugging message.
gboolean gnc_gsettings_get_bool(const gchar *schema, const gchar *key)
Get a boolean value from GSettings.
const gchar * gnc_gsettings_get_prefix(void)
Get the default gsettings schema prefix.
void gnc_gsettings_remove_cb_by_id(const gchar *schema, guint handlerid)
Remove a function that was registered for a callback when a specific key in the settings schema chang...
gboolean gnc_gsettings_set_value(const gchar *schema, const gchar *key, GVariant *value)
Store an arbitrary combination of values into GSettings.
gulong gnc_gsettings_register_cb(const char *schema, const gchar *key, gpointer func, gpointer user_data)
Register a callback for when a specific key in the settings schema is changed.
GSettings helper routines.
gboolean gnc_gsettings_set_string(const gchar *schema, const gchar *key, const gchar *value)
Store a string into GSettings.
void gnc_gsettings_remove_cb_by_func(const gchar *schema, const gchar *key, gpointer func, gpointer user_data)
Remove a function that was registered for a callback when a specific key in the settings schema chang...