mirror of
https://github.com/OpenMW/openmw.git
synced 2025-10-23 19:56:36 +00:00
244 lines
7.2 KiB
C++
244 lines
7.2 KiB
C++
#include "inisettings.hpp"
|
|
|
|
#include <QDebug>
|
|
#include <QFile>
|
|
#include <QRegularExpression>
|
|
#include <QString>
|
|
#include <QStringList>
|
|
|
|
#include <fstream>
|
|
|
|
#include <components/files/qtconversion.hpp>
|
|
|
|
QStringList Wizard::IniSettings::findKeys(const QString& text)
|
|
{
|
|
QStringList result;
|
|
|
|
for (const QString& key : mSettings.keys())
|
|
{
|
|
|
|
if (key.startsWith(text))
|
|
result << key;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
bool Wizard::IniSettings::readFile(std::ifstream& stream, ToUTF8::FromType encoding)
|
|
{
|
|
// Look for a square bracket, "'\\["
|
|
// that has one or more "not nothing" in it, "([^]]+)"
|
|
// and is closed with a square bracket, "\\]"
|
|
QRegularExpression 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, "(.+)"
|
|
QRegularExpression keyRe(QLatin1String("^([^=]+)\\s*=\\s*(.+)$"));
|
|
|
|
QString currentSection;
|
|
|
|
ToUTF8::Utf8Encoder encoder(encoding);
|
|
|
|
std::string legacyEncLine;
|
|
while (std::getline(stream, legacyEncLine))
|
|
{
|
|
std::string_view lineBuffer = encoder.getUtf8(legacyEncLine);
|
|
|
|
// unify Unix-style and Windows file ending
|
|
if (!(lineBuffer.empty()) && (lineBuffer[lineBuffer.length() - 1]) == '\r')
|
|
{
|
|
lineBuffer = lineBuffer.substr(0, lineBuffer.length() - 1);
|
|
}
|
|
|
|
const QString line = QString::fromStdString(std::string(lineBuffer));
|
|
if (line.isEmpty() || line.startsWith(QLatin1Char(';')))
|
|
continue;
|
|
|
|
QRegularExpressionMatch sectionMatch = sectionRe.match(line);
|
|
if (sectionMatch.hasMatch())
|
|
{
|
|
currentSection = sectionMatch.captured(1);
|
|
continue;
|
|
}
|
|
|
|
QRegularExpressionMatch match = keyRe.match(line);
|
|
if (match.hasMatch())
|
|
{
|
|
QString key = match.captured(1).trimmed();
|
|
QString value = match.captured(2).trimmed();
|
|
|
|
// Append the section, but only if there is one
|
|
if (!currentSection.isEmpty())
|
|
key = currentSection + QLatin1Char('/') + key;
|
|
|
|
mSettings[key] = QVariant(value);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Wizard::IniSettings::writeFile(const QString& path, std::ifstream& stream, ToUTF8::FromType encoding)
|
|
{
|
|
// Look for a square bracket, "'\\["
|
|
// that has one or more "not nothing" in it, "([^]]+)"
|
|
// and is closed with a square bracket, "\\]"
|
|
QRegularExpression 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, "(.+)"
|
|
QRegularExpression keyRe(QLatin1String("^([^=]+)\\s*=\\s*(.+)$"));
|
|
|
|
const QStringList keys(mSettings.keys());
|
|
|
|
QString currentSection;
|
|
QString buffer;
|
|
|
|
ToUTF8::Utf8Encoder encoder(encoding);
|
|
|
|
std::string legacyEncLine;
|
|
while (std::getline(stream, legacyEncLine))
|
|
{
|
|
std::string_view lineBuffer = encoder.getUtf8(legacyEncLine);
|
|
// unify Unix-style and Windows file ending
|
|
if (!(lineBuffer.empty()) && (lineBuffer[lineBuffer.length() - 1]) == '\r')
|
|
{
|
|
lineBuffer = lineBuffer.substr(0, lineBuffer.length() - 1);
|
|
}
|
|
|
|
const QString line = QString::fromStdString(std::string(lineBuffer));
|
|
if (line.isEmpty() || line.startsWith(QLatin1Char(';')))
|
|
{
|
|
buffer.append(line + QLatin1String("\n"));
|
|
continue;
|
|
}
|
|
|
|
QRegularExpressionMatch sectionMatch = sectionRe.match(line);
|
|
if (sectionMatch.hasMatch())
|
|
{
|
|
buffer.append(line + QLatin1String("\n"));
|
|
currentSection = sectionMatch.captured(1);
|
|
continue;
|
|
}
|
|
|
|
QRegularExpressionMatch match = keyRe.match(line);
|
|
if (match.hasMatch())
|
|
{
|
|
QString key(match.captured(1).trimmed());
|
|
QString lookupKey(key);
|
|
|
|
// Append the section, but only if there is one
|
|
if (!currentSection.isEmpty())
|
|
lookupKey = currentSection + QLatin1Char('/') + key;
|
|
|
|
buffer.append(key + QLatin1Char('=') + mSettings[lookupKey].toString() + QLatin1String("\n"));
|
|
mSettings.remove(lookupKey);
|
|
}
|
|
}
|
|
|
|
// Add the new settings to the buffer
|
|
QHashIterator<QString, QVariant> i(mSettings);
|
|
while (i.hasNext())
|
|
{
|
|
i.next();
|
|
|
|
QStringList fullKey(i.key().split(QLatin1Char('/')));
|
|
QString section(fullKey.at(0));
|
|
section.prepend(QLatin1Char('['));
|
|
section.append(QLatin1Char(']'));
|
|
const QString& key(fullKey.at(1));
|
|
|
|
int index = buffer.lastIndexOf(section);
|
|
if (index == -1)
|
|
{
|
|
// Add the section to the end of the file, because it's not found
|
|
buffer.append(QString("\n%1\n").arg(section));
|
|
index = buffer.lastIndexOf(section);
|
|
}
|
|
|
|
// Look for the next section
|
|
index = buffer.indexOf(QLatin1Char('['), index + 1);
|
|
|
|
if (index == -1)
|
|
{
|
|
// We are at the last section, append it to the bottom of the file
|
|
buffer.append(QString("\n%1=%2").arg(key, i.value().toString()));
|
|
mSettings.remove(i.key());
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
// Not at last section, add the key at the index
|
|
buffer.insert(index - 1, QString("\n%1=%2").arg(key, i.value().toString()));
|
|
mSettings.remove(i.key());
|
|
}
|
|
}
|
|
|
|
const auto iniPath = Files::pathFromQString(path);
|
|
std::ofstream file(iniPath, std::ios::out);
|
|
if (file.fail())
|
|
return false;
|
|
|
|
file << encoder.getLegacyEnc(buffer.toStdString());
|
|
file.close();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Wizard::IniSettings::parseInx(const QString& path)
|
|
{
|
|
QFile file(path);
|
|
|
|
if (file.open(QIODevice::ReadOnly))
|
|
{
|
|
|
|
const QByteArray data(file.readAll());
|
|
const QByteArray pattern("\x21\x00\x1A\x01\x04\x00\x04\x97\xFF\x06", 10);
|
|
|
|
int i = 0;
|
|
while ((i = data.indexOf(pattern, i)) != -1)
|
|
{
|
|
|
|
int next = data.indexOf(pattern, i + 1);
|
|
if (next == -1)
|
|
break;
|
|
|
|
QByteArray array(data.mid(i, (next - i)));
|
|
|
|
// Skip some invalid entries
|
|
if (array.contains("\x04\x96\xFF"))
|
|
{
|
|
++i;
|
|
continue;
|
|
}
|
|
|
|
// Remove the pattern from the beginning
|
|
array.remove(0, 12);
|
|
|
|
int index = array.indexOf("\x06");
|
|
const QString section(array.left(index));
|
|
|
|
// Figure how many characters to read for the key
|
|
int length = array.indexOf("\x06", section.length() + 3) - (section.length() + 3);
|
|
const QString key(array.mid(section.length() + 3, length));
|
|
|
|
QString value(array.mid(section.length() + key.length() + 6));
|
|
|
|
// Add the value
|
|
setValue(section + QLatin1Char('/') + key, QVariant(value));
|
|
|
|
++i;
|
|
}
|
|
|
|
file.close();
|
|
}
|
|
else
|
|
{
|
|
qDebug() << "Failed to open INX file: " << path;
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|