#include "bundle.hpp" #include "value.hpp" using options::base_value; namespace options { namespace detail { bundle::bundle(const QString& group_name) : mtx(QMutex::Recursive), group_name(group_name), saved(group_name), transient(saved) { } void bundle::reload() { if (group_name.size()) { QMutexLocker l(&mtx); saved = group(group_name); const bool has_changes = is_modified(false); transient = saved; if (has_changes) { connector::notify_all_values(); emit reloading(); emit changed(); } } } void bundle::store_kv(const QString& name, const QVariant& datum) { QMutexLocker l(&mtx); transient.put(name, datum); if (group_name.size()) connector::notify_values(name); emit changed(); } bool bundle::contains(const QString &name) const { QMutexLocker l(mtx); return transient.contains(name); } void bundle::save_deferred(QSettings& s) { if (group_name.size() == 0) return; bool modified_ = false; { QMutexLocker l(&mtx); if (is_modified(true)) { qDebug() << "bundle" << group_name << "changed, saving"; modified_ = true; saved = transient; saved.save_deferred(s); } } if (modified_) emit saving(); } void bundle::save() { save_deferred(*group::ini_file()); } bool bundle::is_modified(bool logspam) const { QMutexLocker l(mtx); for (const auto& kv : transient.kvs) { const QVariant other = saved.get(kv.first); if (!saved.contains(kv.first) || !is_equal(kv.first, kv.second, other)) { if (logspam) qDebug() << "bundle" << group_name << "modified" << "key" << kv.first << "-" << other << "<>" << kv.second; return true; } } for (const auto& kv : saved.kvs) { const QVariant other = transient.get(kv.first); if (!transient.contains(kv.first) || !is_equal(kv.first, kv.second, other)) { if (logspam) qDebug() << "bundle" << group_name << "modified" << "key" << kv.first << "-" << other << "<>" << kv.second; return true; } } return false; } void bundler::bundle_decf(const bundler::k& key) { QMutexLocker l(&implsgl_mtx); if (--std::get<0>(implsgl_data[key]) == 0) { qDebug() << "bundle -" << key; implsgl_data.erase(key); } } void bundler::after_profile_changed_() { QMutexLocker l(&implsgl_mtx); for (auto& kv : implsgl_data) { tt& tuple = kv.second; std::weak_ptr& bundle = std::get<1>(tuple); mem bundle_ = bundle.lock(); if (bundle_) { qDebug() << "bundle: reverting" << kv.first << "due to profile change"; bundle_->reload(); } } } void bundler::refresh_all_bundles() { singleton().after_profile_changed_(); } bundler::bundler() : implsgl_mtx(QMutex::Recursive) { } bundler::~bundler() { qDebug() << "exit: bundle singleton"; } std::shared_ptr bundler::make_bundle(const bundler::k &key) { QMutexLocker l(&implsgl_mtx); if (implsgl_data.count(key) != 0) { auto shared = std::get<1>(implsgl_data[key]).lock(); if (shared != nullptr) return shared; } qDebug() << "bundle +" << key; std::shared_ptr shr(new v(key), [this](v* val) { bundle_decf(val->name()); }); implsgl_data[key] = tt(1, shr); return shr; } OPENTRACK_OPTIONS_EXPORT bundler& singleton() { static bundler ret; return ret; } } // end options::detail OPENTRACK_OPTIONS_EXPORT std::shared_ptr make_bundle(const QString& name) { if (name.size()) return detail::singleton().make_bundle(name); else return std::make_shared(QStringLiteral("")); } QMutex* options::detail::bundle::get_mtx() const { return mtx; } } // end options