forked from mirror/openmw-tes3mp
247 lines
6.7 KiB
C++
247 lines
6.7 KiB
C++
#include "usersettings.hpp"
|
|
|
|
#include <QTextStream>
|
|
#include <QDir>
|
|
#include <QString>
|
|
#include <QRegExp>
|
|
#include <QMap>
|
|
#include <QMessageBox>
|
|
#include <QTextCodec>
|
|
|
|
#include <QFile>
|
|
#include <QDebug>
|
|
|
|
#include <components/files/configurationmanager.hpp>
|
|
#include "settingcontainer.hpp"
|
|
#include <boost/version.hpp>
|
|
|
|
/**
|
|
* Workaround for problems with whitespaces in paths in older versions of Boost library
|
|
*/
|
|
#if (BOOST_VERSION <= 104600)
|
|
namespace boost
|
|
{
|
|
|
|
template<>
|
|
inline boost::filesystem::path lexical_cast<boost::filesystem::path, std::string>(const std::string& arg)
|
|
{
|
|
return boost::filesystem::path(arg);
|
|
}
|
|
|
|
} /* namespace boost */
|
|
#endif /* (BOOST_VERSION <= 104600) */
|
|
|
|
CSMSettings::UserSettings *CSMSettings::UserSettings::mUserSettingsInstance = 0;
|
|
|
|
CSMSettings::UserSettings::UserSettings()
|
|
{
|
|
assert(!mUserSettingsInstance);
|
|
mUserSettingsInstance = this;
|
|
|
|
mReadWriteMessage = QObject::tr("<br><b>Could not open or create file for writing</b><br><br> \
|
|
Please make sure you have the right permissions and try again.<br>");
|
|
|
|
mReadOnlyMessage = QObject::tr("<br><b>Could not open file for reading</b><br><br> \
|
|
Please make sure you have the right permissions and try again.<br>");
|
|
}
|
|
|
|
CSMSettings::UserSettings::~UserSettings()
|
|
{
|
|
mUserSettingsInstance = 0;
|
|
}
|
|
|
|
QTextStream *CSMSettings::UserSettings::openFileStream (const QString &filePath, bool isReadOnly) const
|
|
{
|
|
QFile *file = new QFile(filePath);
|
|
|
|
QIODevice::OpenMode openFlags;
|
|
|
|
if (isReadOnly)
|
|
openFlags = QIODevice::ReadOnly | QIODevice::Text;
|
|
else
|
|
openFlags = QIODevice::ReadWrite | QIODevice::Text | QIODevice::Truncate;
|
|
|
|
if (!(file->open(openFlags)))
|
|
{
|
|
// File cannot be opened or created
|
|
QMessageBox msgBox;
|
|
msgBox.setWindowTitle(QObject::tr("OpenCS configuration file I/O error"));
|
|
msgBox.setIcon(QMessageBox::Critical);
|
|
msgBox.setStandardButtons(QMessageBox::Ok);
|
|
|
|
QString fileMessage = QObject::tr("<br> File: %0").arg(file->fileName());
|
|
|
|
if (!isReadOnly)
|
|
msgBox.setText (mReadWriteMessage + fileMessage);
|
|
else
|
|
msgBox.setText (mReadOnlyMessage + fileMessage);
|
|
|
|
msgBox.exec();
|
|
delete file;
|
|
file = 0;
|
|
}
|
|
|
|
QTextStream *stream = 0;
|
|
|
|
if (file)
|
|
{
|
|
stream = new QTextStream(file);
|
|
stream->setCodec(QTextCodec::codecForName("UTF-8"));
|
|
}
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
bool CSMSettings::UserSettings::writeSettings(QMap<QString, CSMSettings::SettingList *> &settings)
|
|
{
|
|
QTextStream *stream = openFileStream(mPaths.back());
|
|
|
|
QList<QString> keyList = settings.keys();
|
|
|
|
foreach (QString key, keyList)
|
|
{
|
|
SettingList *sectionSettings = settings[key];
|
|
|
|
*stream << "[" << key << "]" << '\n';
|
|
|
|
foreach (SettingContainer *item, *sectionSettings)
|
|
*stream << item->objectName() << " = " << item->getValue() << '\n';
|
|
}
|
|
|
|
stream->device()->close();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
const CSMSettings::SectionMap &CSMSettings::UserSettings::getSettings() const
|
|
{
|
|
return mSectionSettings;
|
|
}
|
|
|
|
void CSMSettings::UserSettings::loadFromFile(const QString &filePath)
|
|
{
|
|
if (filePath.isEmpty())
|
|
return;
|
|
|
|
mSectionSettings.clear();
|
|
|
|
QTextStream *stream = openFileStream (filePath, true);
|
|
|
|
if (stream)
|
|
{
|
|
//looks for a square bracket, "'\\["
|
|
//that has one or more "not nothing" in it, "([^]]+)"
|
|
//and is closed with a square bracket, "\\]"
|
|
|
|
QRegExp sectionRe("^\\[([^]]+)\\]");
|
|
|
|
//Find any character(s) that is/are not equal sign(s), "[^=]+"
|
|
//followed by an optional whitespace, an equal sign, and another optional whitespace, "\\s*=\\s*"
|
|
//and one or more periods, "(.+)"
|
|
|
|
QRegExp keyRe("^([^=]+)\\s*=\\s*(.+)$");
|
|
|
|
CSMSettings::SettingMap *settings = 0;
|
|
QString section = "none";
|
|
|
|
while (!stream->atEnd())
|
|
{
|
|
QString line = stream->readLine().simplified();
|
|
|
|
if (line.isEmpty() || line.startsWith("#"))
|
|
continue;
|
|
|
|
//if a section is found, push it onto a new QStringList
|
|
//and push the QStringList onto
|
|
if (sectionRe.exactMatch(line))
|
|
{
|
|
//add the previous section's settings to the member map
|
|
if (settings)
|
|
mSectionSettings.insert(section, settings);
|
|
|
|
//save new section and create a new list
|
|
section = sectionRe.cap(1);
|
|
settings = new SettingMap;
|
|
continue;
|
|
}
|
|
|
|
if (keyRe.indexIn(line) != -1)
|
|
{
|
|
SettingContainer *sc = new SettingContainer (keyRe.cap(2).simplified());
|
|
sc->setObjectName(keyRe.cap(1).simplified());
|
|
(*settings)[keyRe.cap(1).simplified()] = sc;
|
|
}
|
|
|
|
}
|
|
|
|
mSectionSettings.insert(section, settings);
|
|
|
|
stream->device()->close();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
void CSMSettings::UserSettings::loadSettings (const QString &fileName)
|
|
{
|
|
if (mPaths.count() == 0)
|
|
{
|
|
mPaths.append(QString::fromStdString(mCfgMgr.getGlobalPath().string()) + fileName);
|
|
mPaths.append(QString::fromStdString(mCfgMgr.getLocalPath().string()) + fileName);
|
|
mPaths.append(QString::fromStdString(mCfgMgr.getUserPath().string()) + fileName);
|
|
}
|
|
|
|
foreach (const QString &path, mPaths)
|
|
{
|
|
qDebug() << "Loading config file:" << qPrintable(path);
|
|
loadFromFile(path);
|
|
}
|
|
}
|
|
|
|
void CSMSettings::UserSettings::updateSettings (const QString §ionName, const QString &settingName)
|
|
{
|
|
SettingMap *settings = mSectionSettings[sectionName];
|
|
|
|
if (!settings)
|
|
return;
|
|
|
|
SettingContainer *setting = 0;
|
|
|
|
if (settingName.isEmpty())
|
|
{
|
|
foreach (setting, *settings)
|
|
emit signalUpdateEditorSetting (setting->objectName(), setting->getValue());
|
|
}
|
|
else
|
|
{
|
|
if (settings->find(settingName)!=settings->end())
|
|
{
|
|
setting = settings->value(settingName);
|
|
emit signalUpdateEditorSetting (setting->objectName(), setting->getValue());
|
|
}
|
|
}
|
|
}
|
|
|
|
QString CSMSettings::UserSettings::getSetting (const QString §ion, const QString &setting) const
|
|
{
|
|
if(mSectionSettings.find(section) == mSectionSettings.end())
|
|
return QString();
|
|
|
|
CSMSettings::SettingMap *settings = mSectionSettings.value(section);
|
|
|
|
if(settings->find(setting) == settings->end())
|
|
return QString();
|
|
|
|
CSMSettings::SettingContainer *settingContainer = settings->value(setting);
|
|
|
|
return settingContainer->getValue();
|
|
}
|
|
|
|
CSMSettings::UserSettings& CSMSettings::UserSettings::instance()
|
|
{
|
|
assert(mUserSettingsInstance);
|
|
return *mUserSettingsInstance;
|
|
}
|
|
|