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