From fcf3870b7c0f30a5991e518ad8a404a9d38c3a45 Mon Sep 17 00:00:00 2001
From: Aqua-sama
Date: Fri, 26 Jan 2018 23:09:31 +0100
Subject: Using boost::program_options instead of libconfig
---
src/browser.cpp | 3 +
src/configuration.cpp | 228 +++++++++++++++---------------------------
src/configuration.h | 54 +++-------
src/forms/aboutdialog.cpp | 11 +-
src/main.cpp | 17 ++--
src/mainwindow/mainwindow.cpp | 6 +-
6 files changed, 117 insertions(+), 202 deletions(-)
(limited to 'src')
diff --git a/src/browser.cpp b/src/browser.cpp
index cd901e3..53ca5d6 100644
--- a/src/browser.cpp
+++ b/src/browser.cpp
@@ -24,12 +24,15 @@ Browser::Browser(int &argc, char *argv[])
Browser::~Browser()
{
+ /* We can't modify config, so no point in saving it
if(m_config) {
//QtConcurrent::run(QThreadPool::globalInstance(), m_config.get(), &Configuration::writeIfNeeded);
QtConcurrent::run([c = m_config.get()]() {
qDebug("Writing configuration: %s", c->writeIfNeeded() ? "ok" : "failed");
});
}
+ */
+
if(m_bookmarksManager) {
QtConcurrent::run(QThreadPool::globalInstance(), m_bookmarksManager.get(), &BookmarksWidget::save);
}
diff --git a/src/configuration.cpp b/src/configuration.cpp
index b281b70..4e72d18 100644
--- a/src/configuration.cpp
+++ b/src/configuration.cpp
@@ -7,163 +7,99 @@
*/
#include "configuration.h"
-#include
-#include
-#include
+#include
+#include
-using namespace libconfig;
+namespace po = boost::program_options;
-Configuration::Configuration(const std::string &path, const std::string &home)
+Configuration::Configuration(const QStringList &options)
{
- m_userCfg = new Config();
- // prettier output
- m_userCfg->setOption(Config::OptionSemicolonSeparators, true);
- m_userCfg->setOption(Config::OptionColonAssignmentForGroups, false);
- m_userCfg->setOption(Config::OptionColonAssignmentForNonGroups, false);
- m_userCfg->setOption(Config::OptionOpenBraceOnSeparateLine, false);
- m_userCfg->setOption(Config::OptionFsync, true);
-
- m_userCfgPath = path;
- m_homePath = home;
-}
-
-Configuration::~Configuration()
-{
- delete m_userCfg;
-}
-
-bool Configuration::read(const QString &path)
-{
- QFile conf(path);
-
- if(!conf.open(QIODevice::ReadOnly)) {
- return false;
- }
-
- try {
- m_userCfg->readString(conf.readAll().toStdString());
- conf.close();
- } catch(const ParseException &e) {
- return false;
- }
-
- return true;
-}
-
-bool Configuration::writeIfNeeded(const std::string &path)
-{
- if(!path.empty())
- m_userCfgPath = path;
-
- if(!changed)
- return true;
-
- try {
- m_userCfg->writeFile(m_userCfgPath.c_str());
- } catch(const FileIOException &e) {
- return false;
- }
-
- changed = false;
- return true;
-}
-
-std::vector Configuration::childrenSettings(const char *name)
-{
- std::vector groupNames;
- const Setting &root = m_userCfg->lookup(name);
-
- for(const Setting &setting : root) {
- if(setting.getType() != Setting::TypeGroup) {
- groupNames.emplace_back(setting.getName());
- //groupNames.push_back(setting.getName());
+ m_homePath = QStandardPaths::writableLocation(QStandardPaths::HomeLocation).toStdString();
+
+ // create description
+ desc.add_options()
+
+ // Browser default settings
+ // default profile name the browser should use; "" is off-the-record
+ ("browser.profile", po::value()->default_value(""))
+
+ // default window size
+ ("browser.window.height", po::value()->default_value(720))
+ ("browser.window.width", po::value()->default_value(1280))
+ ("browser.window.maximized", po::value()->default_value(true))
+ ("browser.window.title", po::value()->default_value("title — smolbote [profile]"))
+
+ // window ui
+ ("browser.ui.navtoolbarMovable", po::value()->default_value(false))
+ ("browser.ui.tabtoolbarMovable", po::value()->default_value(false))
+
+ // browser shortcuts
+
+ // browser menu
+ ("browser.shortcuts.newWindow", po::value()->default_value("Ctrl+N"))
+ ("browser.shortcuts.newTab", po::value()->default_value("Ctrl+T"))
+ ("browser.shortcuts.about", po::value()->default_value("F1"))
+ ("browser.shortcuts.quit", po::value()->default_value("Ctrl+Q"))
+
+ // navigation
+ ("browser.shortcuts.back", po::value()->default_value("Ctrl+Left"))
+ ("browser.shortcuts.forward", po::value()->default_value("Ctrl+Right"))
+ ("browser.shortcuts.refresh", po::value()->default_value("F5"))
+ ("browser.shortcuts.reload", po::value()->default_value("Ctrl+F5"))
+ ("browser.shortcuts.home", po::value()->default_value("Ctrl+Home"))
+
+ // tabs
+ ("browser.shortcuts.tabClose", po::value()->default_value("Ctrl+X"))
+ ("browser.shortcuts.tabLeft", po::value()->default_value("Ctrl+O"))
+ ("browser.shortcuts.tabRight", po::value()->default_value("Ctrl+P"))
+
+ // page
+ ("browser.shortcuts.toggleSearchBox", po::value()->default_value("F3"))
+ ("browser.shortcuts.focusAddress", po::value()->default_value("F4"))
+ ("browser.shortcuts.fullscreen", po::value()->default_value("F11"))
+
+ // Filter settings
+ ("filter.path", po::value()->default_value("~/.config/smolbote/hosts.d"))
+
+ // Plugin settings
+ ("plugins.path", po::value()->default_value("~/.config/smolbote/plugins.d"))
+
+ // Profile settings
+ ("profile.path", po::value()->default_value("~/.config/smolbote/profiles.d"))
+ ("profile.search", po::value()->default_value("https://duckduckgo.com/?q=$term&ia=web"))
+ ("profile.homepage", po::value()->default_value("about:blank"))
+ ("profile.newtab", po::value()->default_value("about:blank"))
+
+ // Bookmark settings
+ ("bookmarks.path", po::value()->default_value("~/.config/smolbote/bookmarks.xbel"))
+ ("bookmarks.shortcut", po::value()->default_value("Ctrl+B"))
+
+ // Downloads settings
+ ("downloads.path", po::value()->default_value("~/Downloads"))
+ ("downloads.shortcut", po::value()->default_value("Ctrl+D"))
+ ;
+
+ // store the defaults into the vm
+ {
+ int argc = options.length();
+ const char* argv[argc];
+ for(int i = 0; i < argc; ++i) {
+ argv[i] = qUtf8Printable(options.at(i));
}
+
+ po::store(po::parse_command_line(argc, argv, desc), vm);
}
- return groupNames;
}
-std::vector Configuration::childrenGroups(const char *name)
+Configuration::~Configuration()
{
- std::vector groupNames;
- const Setting &root = m_userCfg->lookup(name);
-
- for(const Setting &setting : root) {
- if(setting.getType() == Setting::TypeGroup) {
- groupNames.emplace_back(setting.getName());
- //groupNames.push_back(setting.getName());
- }
- }
- return groupNames;
}
-std::string castToString(const libconfig::Setting &v)
+bool Configuration::read(const QString &path)
{
- // cast depending on type
- // type checks are done during compile time
- switch(v.getType()) {
- case Setting::TypeNone:
- return std::string();
-
- case Setting::TypeInt:
- // int, unsigned int, long, unsigned long
- return std::to_string(static_cast(v));
-
- case Setting::TypeInt64:
- // int, unsigned int; long long, unsigned long long
- return std::to_string(static_cast(v));
-
- case Setting::TypeFloat:
- // float, double
- return std::to_string(static_cast(v));
-
- case Setting::TypeString:
- // const char*, std::string
- return std::string(static_cast(v));
-
- case Setting::TypeBoolean:
- // bool
- return std::string(static_cast(v) ? "true" : "false");
-
- case Setting::TypeGroup:
- case Setting::TypeArray:
- case Setting::TypeList:
- return std::string();
- }
+ std::ifstream f(path.toStdString(), std::ifstream::in);
+ po::store(po::parse_config_file(f, desc, false), vm);
+ return true;
}
-void setFromString(libconfig::Setting &setting, const std::string &value)
-{
- switch(setting.getType()) {
- case libconfig::Setting::TypeNone:
- break;
-
- case libconfig::Setting::TypeInt:
- case libconfig::Setting::TypeInt64:
- setting = std::stoi(value);
- break;
-
- case libconfig::Setting::TypeFloat:
- setting = std::stod(value);
- break;
-
- case libconfig::Setting::TypeString:
- setting = value.c_str();
- break;
-
- case libconfig::Setting::TypeBoolean:
- if(value == "true") {
- setting = true;
- } else if(value == "false") {
- setting = false;
- }
- break;
-
- case libconfig::Setting::TypeGroup:
- break;
- case libconfig::Setting::TypeArray:
- break;
- case libconfig::Setting::TypeList:
- break;
- }
-}
diff --git a/src/configuration.h b/src/configuration.h
index 913fa65..27988b9 100644
--- a/src/configuration.h
+++ b/src/configuration.h
@@ -9,39 +9,34 @@
#ifndef CONFIGURATION_H
#define CONFIGURATION_H
-#include
#include
#include
#include
#include
-
-std::string castToString(const libconfig::Setting &v);
-void setFromString(libconfig::Setting &setting, const std::string &value);
+#include
+#include
class Configuration
{
public:
- explicit Configuration(const std::string &path, const std::string &home);
+ explicit Configuration(const QStringList &options = QStringList());
~Configuration();
bool read(const QString &path);
- bool writeIfNeeded(const std::string &path = std::string());
-
- std::vector childrenSettings(const char *name = "");
- std::vector childrenGroups(const char *name = "");
template
std::optional value(const char *path) const
{
- // if setting doesn't exist, give back a nullopt
- if(!m_userCfg->exists(path)) {
- qWarning("Requesting non-existent option %s", path);
- return std::nullopt;
+ // if setting doesn't exist, we crash
+ // in debug builds, check if setting exists
+#ifdef QT_DEBUG
+ if(vm.count(path) == 0) {
+ qWarning("value(%s) does not exist, probably crashing now", path);
}
+#endif
- const libconfig::Setting &v = m_userCfg->lookup(path);
if constexpr(std::is_same_v) {
- std::string r = castToString(v);
+ std::string r = vm[path].as();
// check if it's a path
if(r.front() == '~') {
@@ -50,35 +45,14 @@ public:
return std::optional(r);
} else
- return std::optional(static_cast(v));
- }
-
- template
- bool setValue(std::string path, const T &val)
- {
- if(!m_userCfg->exists(path)) {
- return false;
- }
-
- libconfig::Setting &setting = m_userCfg->lookup(path);
- // compiler complained about operator= not taking unsinged ints, longs and long longs
- if constexpr(std::is_unsigned_v && !std::is_same_v) {
- setting = static_cast>(val);
- } else if constexpr(std::is_same_v) {
- setFromString(setting, val);
- } else {
- setting = val;
- }
-
- changed = true;
- return true;
+ return std::optional(vm[path].as());
}
private:
- bool changed = false;
+ boost::program_options::options_description desc;
+ boost::program_options::variables_map vm;
+
std::string m_homePath;
- std::string m_userCfgPath;
- libconfig::Config *m_userCfg;
};
#endif // CONFIGURATION_H
diff --git a/src/forms/aboutdialog.cpp b/src/forms/aboutdialog.cpp
index d9c75c3..89cb23e 100644
--- a/src/forms/aboutdialog.cpp
+++ b/src/forms/aboutdialog.cpp
@@ -9,11 +9,11 @@
#include "aboutdialog.h"
#include "ui_aboutdialog.h"
#include "version.h"
-#include
+#include
// The extra level of indirection will allow the preprocessor to expand the macros before they are converted to strings.
-#define STR_HELPER(x) #x
-#define STR(x) STR_HELPER(x)
+#define STRINGIFY(x) #x
+#define STR(x) STRINGIFY(x)
// compiler
// clang also defines __GNUC__, so we need to check for clang first
@@ -25,9 +25,6 @@
#define compiler "unknown compiler";
#endif
-// libconfig
-#define LIBCONFIG_VERSION_STR STR(LIBCONFIGXX_VER_MAJOR) "." STR(LIBCONFIGXX_VER_MINOR) "." STR(LIBCONFIGXX_VER_REVISION)
-
AboutDialog::AboutDialog(QWidget *parent)
: QDialog(parent)
, ui(new Ui::AboutDialog)
@@ -64,7 +61,7 @@ AboutDialog::AboutDialog(QWidget *parent)
"Compiled with " compiler "
"
""
"- Qt " QT_VERSION_STR "
"
- "- libconfig " LIBCONFIG_VERSION_STR "
"
+ "- Boost " BOOST_LIB_VERSION "
"
"
"));
ui->toolBox->addItem(libsLabel, tr("Details"));
}
diff --git a/src/main.cpp b/src/main.cpp
index cce2c61..70f22be 100644
--- a/src/main.cpp
+++ b/src/main.cpp
@@ -51,7 +51,6 @@ int main(int argc, char **argv)
// default config, :/poi.cfg
QCommandLineOption defaultConfigOption("default-config", "Set the default configuration file.", "path");
- defaultConfigOption.setDefaultValue(":/poi.cfg");
parser.addOption(defaultConfigOption);
QCommandLineOption profileOption({ "p", "profile" }, "Use this profile.", "PROFILE");
@@ -76,14 +75,20 @@ int main(int argc, char **argv)
qDebug("profile=%s", qUtf8Printable(parser.value(profileOption)));
#endif
- std::shared_ptr config = std::make_shared(
- parser.value(configOption).toStdString(),
- QStandardPaths::writableLocation(QStandardPaths::HomeLocation).toStdString());
+ std::shared_ptr config = std::make_shared();
// first load the default configuration
- config->read(parser.value(defaultConfigOption));
+ if(parser.isSet(defaultConfigOption)) {
+ qDebug("Reading default configuration [%s]: %s",
+ qUtf8Printable(parser.value(defaultConfigOption)),
+ config->read(parser.value(defaultConfigOption)) ? "ok" : "failed");
+ }
// then load in the user configuration, which will overwrite it
- config->read(parser.value(configOption));
+ if(parser.isSet(configOption)) {
+ qDebug("Reading configuration [%s]: %s",
+ qUtf8Printable(parser.value(configOption)),
+ config->read(parser.value(configOption)) ? "ok" : "failed");
+ }
// check for other instances
// if we socket hasn't been disabled (socket is not none)
diff --git a/src/mainwindow/mainwindow.cpp b/src/mainwindow/mainwindow.cpp
index d5efe5c..b0195a5 100644
--- a/src/mainwindow/mainwindow.cpp
+++ b/src/mainwindow/mainwindow.cpp
@@ -16,7 +16,7 @@
#include
#include
#include
-#include
+//#include
MainWindow::MainWindow(std::shared_ptr config, QWidget *parent)
: QMainWindow(parent)
@@ -201,8 +201,8 @@ void MainWindow::about()
void MainWindow::showSettingsDialog()
{
- SettingsDialog *dlg = new SettingsDialog(m_config, this);
- dlg->exec();
+ //SettingsDialog *dlg = new SettingsDialog(m_config, this);
+ //dlg->exec();
}
void MainWindow::setProfile(std::shared_ptr profile)
--
cgit v1.2.1