2017-02-21 11:01:06 +00:00
|
|
|
#include "LaunchConfig.hxx"
|
|
|
|
|
2018-06-25 23:06:20 +01:00
|
|
|
#include <set>
|
|
|
|
|
2017-02-21 11:01:06 +00:00
|
|
|
#include <Main/options.hxx>
|
|
|
|
#include <simgear/misc/sg_path.hxx>
|
|
|
|
|
2017-12-15 15:42:36 +00:00
|
|
|
#include <QSettings>
|
|
|
|
#include <QDebug>
|
2018-06-27 23:06:39 +01:00
|
|
|
#include <QIODevice>
|
|
|
|
#include <QDataStream>
|
2018-09-02 09:18:46 +01:00
|
|
|
#include <QClipboard>
|
|
|
|
#include <QGuiApplication>
|
2017-12-15 15:42:36 +00:00
|
|
|
|
2017-07-19 19:24:33 +01:00
|
|
|
static bool static_enableDownloadDirUI = true;
|
2018-06-27 23:06:39 +01:00
|
|
|
static QSettings::Format static_binaryFormat = QSettings::InvalidFormat;
|
|
|
|
|
|
|
|
static bool binaryReadFunc(QIODevice &device, QSettings::SettingsMap &map)
|
|
|
|
{
|
|
|
|
QDataStream ds(&device);
|
|
|
|
int count;
|
|
|
|
ds >> count;
|
|
|
|
for (int i=0; i < count; ++i) {
|
|
|
|
QString k;
|
|
|
|
QVariant v;
|
|
|
|
ds >> k >> v;
|
|
|
|
map.insert(k, v);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool binaryWriteFunc(QIODevice &device, const QSettings::SettingsMap &map)
|
|
|
|
{
|
|
|
|
QDataStream ds(&device);
|
|
|
|
|
|
|
|
ds << map.size();
|
|
|
|
Q_FOREACH(QString key, map.keys()) {
|
|
|
|
ds << key << map.value(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2017-07-19 19:24:33 +01:00
|
|
|
|
2017-02-21 11:01:06 +00:00
|
|
|
LaunchConfig::LaunchConfig(QObject* parent) :
|
|
|
|
QObject(parent)
|
|
|
|
{
|
2018-06-27 23:06:39 +01:00
|
|
|
if (static_binaryFormat == QSettings::InvalidFormat) {
|
|
|
|
static_binaryFormat = QSettings::registerFormat("fglaunch",
|
|
|
|
&binaryReadFunc,
|
|
|
|
&binaryWriteFunc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LaunchConfig::~LaunchConfig()
|
|
|
|
{
|
|
|
|
|
2017-02-21 11:01:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::reset()
|
|
|
|
{
|
|
|
|
m_values.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::applyToOptions() const
|
|
|
|
{
|
2018-06-26 11:08:01 +01:00
|
|
|
const auto extraArgs = extraArgNames();
|
2017-02-21 11:01:06 +00:00
|
|
|
flightgear::Options* options = flightgear::Options::sharedInstance();
|
2018-06-25 23:06:20 +01:00
|
|
|
std::for_each(m_values.begin(), m_values.end(),
|
2018-06-26 11:08:01 +01:00
|
|
|
[options, &extraArgs](const Arg& arg)
|
2018-06-25 23:06:20 +01:00
|
|
|
{
|
|
|
|
const auto name = arg.arg.toStdString();
|
|
|
|
if (arg.origin == Launcher) {
|
2018-06-26 11:08:01 +01:00
|
|
|
auto it = extraArgs.find(name);
|
|
|
|
if (it != extraArgs.end()) {
|
2018-06-25 23:06:20 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
options->addOption(name, arg.value.toStdString());
|
|
|
|
});
|
2017-02-21 11:01:06 +00:00
|
|
|
}
|
|
|
|
|
2018-06-26 11:08:01 +01:00
|
|
|
std::set<std::string> LaunchConfig::extraArgNames() const
|
|
|
|
{
|
|
|
|
// build a set of all the extra args we have defined
|
|
|
|
std::set<std::string> r;
|
|
|
|
for (auto arg : m_values) {
|
|
|
|
// don't override prop: arguments
|
|
|
|
if (arg.arg == "prop") continue;
|
|
|
|
|
2018-07-03 09:35:26 +01:00
|
|
|
// allow some multi-valued arguments
|
|
|
|
if (arg.arg == "fg-scenery")
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-06-26 11:08:01 +01:00
|
|
|
if (arg.origin == ExtraArgs)
|
|
|
|
r.insert(arg.arg.toStdString());
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2018-06-25 23:06:20 +01:00
|
|
|
void LaunchConfig::setArg(QString name, QString value, Origin origin)
|
2017-02-21 11:01:06 +00:00
|
|
|
{
|
2018-06-25 23:06:20 +01:00
|
|
|
m_values.push_back(Arg(name, value, origin));
|
2017-02-21 11:01:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::setArg(const std::string &name, const std::string &value)
|
|
|
|
{
|
2018-06-25 23:06:20 +01:00
|
|
|
setArg(QString::fromStdString(name), QString::fromStdString(value), Launcher);
|
2017-02-21 11:01:06 +00:00
|
|
|
}
|
|
|
|
|
2018-06-25 23:06:20 +01:00
|
|
|
void LaunchConfig::setProperty(QString path, QVariant value, Origin origin)
|
2017-02-21 11:01:06 +00:00
|
|
|
{
|
2018-06-25 23:06:20 +01:00
|
|
|
m_values.push_back(Arg("prop", path + "=" + value.toString(), origin));
|
2017-02-21 11:01:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::setEnableDisableOption(QString name, bool value)
|
|
|
|
{
|
2018-06-25 23:06:20 +01:00
|
|
|
m_values.push_back(Arg((value ? "enable-" : "disable-") + name, "", Launcher));
|
2017-02-21 11:01:06 +00:00
|
|
|
}
|
|
|
|
|
2018-06-22 12:09:35 +01:00
|
|
|
QString LaunchConfig::htmlForCommandLine()
|
|
|
|
{
|
|
|
|
QString html;
|
|
|
|
string_list commandLineOpts = flightgear::Options::sharedInstance()->extractOptions();
|
|
|
|
if (!commandLineOpts.empty()) {
|
|
|
|
html += tr("<p>Options passed on the command line:</p>\n");
|
|
|
|
html += "<ul>\n";
|
|
|
|
for (auto opt : commandLineOpts) {
|
|
|
|
html += QString("<li>--") + QString::fromStdString(opt) + "</li>\n";
|
|
|
|
}
|
|
|
|
html += "</ul>\n";
|
|
|
|
}
|
2018-06-25 23:06:20 +01:00
|
|
|
|
2018-06-22 12:09:35 +01:00
|
|
|
reset();
|
|
|
|
collect();
|
|
|
|
|
2018-06-26 11:08:01 +01:00
|
|
|
const auto extraArgs = extraArgNames();
|
|
|
|
|
2018-06-22 12:09:35 +01:00
|
|
|
html += tr("<p>Options set in the launcher:</p>\n");
|
|
|
|
html += "<ul>\n";
|
2018-06-25 23:06:20 +01:00
|
|
|
for (auto arg : valuesFromLauncher()) {
|
2018-06-26 11:08:01 +01:00
|
|
|
auto it = extraArgs.find(arg.arg.toStdString());
|
|
|
|
html += "<li>";
|
|
|
|
bool strikeThrough = (it != extraArgs.end());
|
|
|
|
if (strikeThrough) {
|
|
|
|
html += "<i>";
|
|
|
|
}
|
2018-06-25 23:06:20 +01:00
|
|
|
if (arg.value.isEmpty()) {
|
2018-06-26 11:08:01 +01:00
|
|
|
html += QString("--") + arg.arg;
|
2018-06-25 23:06:20 +01:00
|
|
|
} else if (arg.arg == "prop") {
|
2018-06-26 11:08:01 +01:00
|
|
|
html += QString("--") + arg.arg + ":" + arg.value;
|
2018-06-25 23:06:20 +01:00
|
|
|
} else {
|
2018-06-26 11:08:01 +01:00
|
|
|
html += QString("--") + arg.arg + "=" + arg.value;
|
2018-06-25 23:06:20 +01:00
|
|
|
}
|
2018-06-26 11:08:01 +01:00
|
|
|
if (strikeThrough) {
|
|
|
|
html += tr(" (will be skipped due to being specified as an additional argument)") + "</i> ";
|
|
|
|
}
|
|
|
|
html += "</li>\n";
|
2018-06-25 23:06:20 +01:00
|
|
|
}
|
|
|
|
html += "</ul>\n";
|
|
|
|
|
|
|
|
html += tr("<p>Options set as additional arguments:</p>\n");
|
|
|
|
html += "<ul>\n";
|
|
|
|
for (auto arg : valuesFromExtraArgs()) {
|
2018-06-22 12:09:35 +01:00
|
|
|
if (arg.value.isEmpty()) {
|
|
|
|
html += QString("<li>--") + arg.arg + "</li>\n";
|
|
|
|
} else if (arg.arg == "prop") {
|
|
|
|
html += QString("<li>--") + arg.arg + ":" + arg.value + "</li>\n";
|
|
|
|
} else {
|
|
|
|
html += QString("<li>--") + arg.arg + "=" + arg.value + "</li>\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
html += "</ul>\n";
|
|
|
|
|
|
|
|
return html;
|
|
|
|
}
|
|
|
|
|
2018-09-02 09:18:46 +01:00
|
|
|
static QString formatArgForClipboard(const LaunchConfig::Arg& a)
|
|
|
|
{
|
|
|
|
if (a.value.isEmpty()) {
|
|
|
|
return "--" + a.arg + " ";
|
|
|
|
} else if (a.arg == "prop") {
|
|
|
|
return "--" + a.arg + ":" + a.value + " ";
|
|
|
|
} else {
|
|
|
|
return "--" + a.arg + "=" + a.value + " ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::copyCommandLine()
|
|
|
|
{
|
|
|
|
QString r;
|
|
|
|
|
|
|
|
for (auto opt : flightgear::Options::sharedInstance()->extractOptions()) {
|
|
|
|
r += "--" + QString::fromStdString(opt) + " ";
|
|
|
|
}
|
|
|
|
|
|
|
|
reset();
|
|
|
|
collect();
|
|
|
|
const auto extraArgs = extraArgNames();
|
|
|
|
|
|
|
|
for (auto arg : valuesFromLauncher()) {
|
|
|
|
auto it = extraArgs.find(arg.arg.toStdString());
|
|
|
|
if (it != extraArgs.end()) {
|
|
|
|
continue; // skipped due to extra arg overriding
|
|
|
|
}
|
|
|
|
|
|
|
|
r += formatArgForClipboard(arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto arg : valuesFromExtraArgs()) {
|
|
|
|
r += formatArgForClipboard(arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
QClipboard *clipboard = QGuiApplication::clipboard();
|
|
|
|
clipboard->setText(r);
|
|
|
|
}
|
|
|
|
|
2018-06-27 23:06:39 +01:00
|
|
|
bool LaunchConfig::saveConfigToINI()
|
|
|
|
{
|
|
|
|
// create settings using default type (INI) and path (inside FG_HOME),
|
|
|
|
// as setup in initQSettings()
|
|
|
|
m_loadSaveSettings.reset(new QSettings);
|
|
|
|
emit save();
|
|
|
|
m_loadSaveSettings->sync();
|
|
|
|
m_loadSaveSettings.reset();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LaunchConfig::loadConfigFromINI()
|
|
|
|
{
|
|
|
|
// create settings using default type (INI) and path (inside FG_HOME),
|
|
|
|
// as setup in initQSettings()
|
|
|
|
m_loadSaveSettings.reset(new QSettings);
|
|
|
|
emit restore();
|
|
|
|
emit postRestore();
|
|
|
|
m_loadSaveSettings.reset();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LaunchConfig::saveConfigToFile(QString path)
|
|
|
|
{
|
|
|
|
m_loadSaveSettings.reset(new QSettings(path, static_binaryFormat));
|
|
|
|
emit save();
|
|
|
|
m_loadSaveSettings.reset();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool LaunchConfig::loadConfigFromFile(QString path)
|
|
|
|
{
|
|
|
|
m_loadSaveSettings.reset(new QSettings(path, static_binaryFormat));
|
|
|
|
emit restore();
|
|
|
|
// some things have an ordering dependency, give them a chance to run
|
|
|
|
// after other settings have been restored (eg, location or aircraft)
|
|
|
|
emit postRestore();
|
|
|
|
m_loadSaveSettings.reset();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-12-15 15:42:36 +00:00
|
|
|
QVariant LaunchConfig::getValueForKey(QString group, QString key, QVariant defaultValue) const
|
|
|
|
{
|
2018-06-27 23:06:39 +01:00
|
|
|
if (!m_loadSaveSettings) {
|
|
|
|
// becuase we load settings on component completion, we need
|
|
|
|
// to create the default implementation (using the INI file)
|
|
|
|
// on demand
|
|
|
|
m_loadSaveSettings.reset(new QSettings);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_loadSaveSettings->beginGroup(group);
|
|
|
|
auto v = m_loadSaveSettings->value(key, defaultValue);
|
2018-09-02 09:18:46 +01:00
|
|
|
bool convertedOk = v.convert(static_cast<int>(defaultValue.type()));
|
2017-12-15 15:42:36 +00:00
|
|
|
if (!convertedOk) {
|
|
|
|
qWarning() << "type forcing on loaded value failed:" << key << v << v.typeName() << defaultValue;
|
|
|
|
}
|
|
|
|
// qInfo() << Q_FUNC_INFO << key << "value" << v << v.typeName() << convertedOk;
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::setValueForKey(QString group, QString key, QVariant var)
|
|
|
|
{
|
2018-06-27 23:06:39 +01:00
|
|
|
Q_ASSERT(m_loadSaveSettings);
|
|
|
|
m_loadSaveSettings->beginGroup(group);
|
2017-12-15 15:42:36 +00:00
|
|
|
// qInfo() << "saving" << key << "with value" << var << var.typeName();
|
2018-06-27 23:06:39 +01:00
|
|
|
m_loadSaveSettings->setValue(key, var);
|
|
|
|
m_loadSaveSettings->endGroup();
|
2017-12-15 15:42:36 +00:00
|
|
|
}
|
|
|
|
|
2017-02-21 11:01:06 +00:00
|
|
|
QString LaunchConfig::defaultDownloadDir() const
|
|
|
|
{
|
|
|
|
return QString::fromStdString(flightgear::defaultDownloadDir().utf8Str());
|
|
|
|
}
|
|
|
|
|
2017-07-19 19:24:33 +01:00
|
|
|
bool LaunchConfig::enableDownloadDirUI() const
|
|
|
|
{
|
|
|
|
return static_enableDownloadDirUI;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LaunchConfig::setEnableDownloadDirUI(bool enableDownloadDirUI)
|
|
|
|
{
|
|
|
|
static_enableDownloadDirUI = enableDownloadDirUI;
|
|
|
|
}
|
|
|
|
|
2017-02-21 11:01:06 +00:00
|
|
|
auto LaunchConfig::values() const -> std::vector<Arg>
|
|
|
|
{
|
|
|
|
return m_values;
|
|
|
|
}
|
2018-06-25 23:06:20 +01:00
|
|
|
|
|
|
|
auto LaunchConfig::valuesFromLauncher() const -> std::vector<Arg>
|
|
|
|
{
|
|
|
|
std::vector<Arg> result;
|
|
|
|
std::copy_if(m_values.begin(), m_values.end(), std::back_inserter(result), [](const Arg& a)
|
|
|
|
{ return a.origin == Launcher; });
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto LaunchConfig::valuesFromExtraArgs() const -> std::vector<Arg>
|
|
|
|
{
|
|
|
|
std::vector<Arg> result;
|
|
|
|
std::copy_if(m_values.begin(), m_values.end(), std::back_inserter(result), [](const Arg& a)
|
|
|
|
{ return a.origin == ExtraArgs; });
|
|
|
|
return result;
|
|
|
|
}
|