/*******************************************************************************
**
** smolbote: yet another qute browser
** Copyright (C) 2017 Xian Nox
**
** 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 3 of the License, or
** (at your option) any later version.
**
** This program is distributed in the hope that it will be useful,
** but WITHOUT ANY WARRANTY; without even the implied warranty of
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
** GNU General Public License for more details.
**
** You should have received a copy of the GNU General Public License
** along with this program. If not, see .
**
******************************************************************************/
#include "settings.h"
#include
#include
#include
#include
#include
#include
#include
Settings::Settings(const QString &configFile, const QString &defaultsFile)
{
m_configurationPath = configFile;
m_defaultsPath = defaultsFile;
// homeLocation is the user's home folder
homeLocation = QStandardPaths::writableLocation(QStandardPaths::HomeLocation);
// settingsLocation is the location of the configFile
if(QFile::exists(configFile)) {
settingsLocation = QFileInfo(configFile).dir().absolutePath();
} else {
// if file doesn't exist, use the generic location
settingsLocation = QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation);
}
// cacheLocation
cacheLocation = QStandardPaths::writableLocation(QStandardPaths::CacheLocation);
userValues = parse(m_configurationPath);
defaultValues = parse(m_defaultsPath);
m_watcher = new QFileSystemWatcher();
m_watcher->addPath(configFile);
QObject::connect(m_watcher, &QFileSystemWatcher::fileChanged, [this](const QString &path) {
if(path == m_configurationPath) {
#ifdef QT_DEBUG
qDebug("Reloading user configuration");
#endif
userValues = parse(m_configurationPath);
}
});
#ifdef QT_DEBUG
qDebug(">> Configuration");
qDebug("- userconf: [%s]", qUtf8Printable(m_configurationPath));
qDebug("- defaults: [%s]", qUtf8Printable(m_defaultsPath));
qDebug("- $home [%s]", qUtf8Printable(homeLocation));
qDebug("- $settings [%s]", qUtf8Printable(settingsLocation));
qDebug("- $cache [%s]", qUtf8Printable(cacheLocation));
#endif
}
Settings::~Settings()
{
m_watcher->deleteLater();
}
QString Settings::configurationPath() const
{
return m_configurationPath;
}
QString Settings::defaultsPath() const
{
return m_defaultsPath;
}
bool Settings::isEmpty() const
{
return userValues.empty();
}
bool Settings::contains(const QString &key)
{
const toml::Value *x = userValues.find(key.toStdString());
const toml::Value *y = defaultValues.find(key.toStdString());
if(x || y) {
return true;
} else {
return false;
}
}
QVariant Settings::value(const QString &key) const
{
const toml::Value *cValue = userValues.find(key.toStdString());
const toml::Value *dValue = defaultValues.find(key.toStdString());
QVariant r;
if(userValues.has(key.toStdString())) {
r = valueToVariant(cValue);
} else {
if(defaultValues.has(key.toStdString())) {
r = valueToVariant(dValue);
}
}
// check if key is a path, in which case replace '~' with the home location
if(key.endsWith(QLatin1String("path"), Qt::CaseInsensitive)) {
QString value = r.toString();
while(value.contains('$')) {
value.replace("$settings", settingsLocation);
value.replace("$cache", cacheLocation);
value.replace("$home", homeLocation);
}
r = QVariant(value);
}
return r;
}
toml::Value Settings::parse(const QString &filename)
{
toml::Value r;
if(filename.isEmpty()) {
qWarning("Empty configuration path.");
} else {
QFile file(filename);
if(file.open(QIODevice::ReadOnly)) {
std::stringstream d(file.readAll().toStdString());
file.close();
toml::ParseResult result = toml::parse(d);
if(!result.valid()) {
qWarning("Invalid configuration: %s", result.errorReason.c_str());
}
r = result.value;
} else {
qWarning("Cannot open configuration: %s", qUtf8Printable(filename));
}
}
return r;
}
QVariant Settings::fromList(const toml::Value *list) const
{
QStringList l;
for(const toml::Value &v : list->as()) {
// TODO check value type
l.append(QString::fromStdString(v.as()));
}
return QVariant(l);
}
QVariant Settings::valueToVariant(const toml::Value *value) const
{
QVariant r;
switch (value->type()) {
case toml::Value::NULL_TYPE:
break;
case toml::Value::BOOL_TYPE:
r = QVariant(value->as());
break;
case toml::Value::INT_TYPE:
r = QVariant(value->as());
break;
case toml::Value::DOUBLE_TYPE:
r = QVariant(value->as());
break;
case toml::Value::STRING_TYPE:
r = QVariant(QString::fromStdString(value->as()));
break;
case toml::Value::ARRAY_TYPE:
r = fromList(value);
break;
default:
qWarning("Unhandled type in configuration");
break;
}
return r;
}