2014-01-01 22:46:29 +01:00
|
|
|
#include "unshieldworker.hpp"
|
2013-12-25 18:52:34 +01:00
|
|
|
|
|
|
|
#include <QDebug>
|
2014-01-27 15:12:02 +01:00
|
|
|
#include <QDir>
|
2013-12-26 18:02:34 +01:00
|
|
|
#include <QFile>
|
2014-01-01 17:15:34 +01:00
|
|
|
#include <QFileInfo>
|
2013-12-26 18:02:34 +01:00
|
|
|
#include <QReadLocker>
|
2013-12-25 18:52:34 +01:00
|
|
|
#include <QStringList>
|
2023-01-19 15:07:12 +04:00
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
|
|
|
|
#include <components/files/qtconversion.hpp>
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2022-10-10 13:41:36 +02:00
|
|
|
#include <apps/wizard/inisettings.hpp>
|
|
|
|
|
2022-04-17 16:28:14 +00:00
|
|
|
Wizard::UnshieldWorker::UnshieldWorker(qint64 expectedMorrowindBsaSize, QObject* parent)
|
2014-01-01 22:46:29 +01:00
|
|
|
: QObject(parent)
|
2022-04-17 16:28:14 +00:00
|
|
|
, mExpectedMorrowindBsaSize(expectedMorrowindBsaSize)
|
2013-12-26 18:02:34 +01:00
|
|
|
, mIniSettings()
|
2013-12-25 18:52:34 +01:00
|
|
|
{
|
|
|
|
unshield_set_log_level(0);
|
|
|
|
|
2013-12-26 18:02:34 +01:00
|
|
|
mPath = QString();
|
|
|
|
mIniPath = QString();
|
2014-03-16 22:09:20 +01:00
|
|
|
mDiskPath = QString();
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
// Default to Latin encoding
|
2023-01-19 15:07:12 +04:00
|
|
|
mIniEncoding = ToUTF8::FromType::WINDOWS_1252;
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2013-12-25 18:52:34 +01:00
|
|
|
mInstallMorrowind = false;
|
|
|
|
mInstallTribunal = false;
|
|
|
|
mInstallBloodmoon = false;
|
2014-01-17 13:21:44 +01:00
|
|
|
|
|
|
|
mMorrowindDone = false;
|
|
|
|
mTribunalDone = false;
|
|
|
|
mBloodmoonDone = false;
|
2014-02-08 00:09:25 +01:00
|
|
|
|
2014-04-18 13:17:37 +02:00
|
|
|
mStopped = false;
|
|
|
|
|
2014-02-08 00:09:25 +01:00
|
|
|
qRegisterMetaType<Wizard::Component>("Wizard::Component");
|
2014-01-01 22:46:29 +01:00
|
|
|
}
|
|
|
|
|
2014-04-18 13:17:37 +02:00
|
|
|
void Wizard::UnshieldWorker::stopWorker()
|
|
|
|
{
|
|
|
|
mStopped = true;
|
2017-06-13 12:00:55 +02:00
|
|
|
mWait.wakeOne();
|
2014-04-18 13:17:37 +02:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
void Wizard::UnshieldWorker::setInstallComponent(Wizard::Component component, bool install)
|
2013-12-25 18:52:34 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QWriteLocker writeLock(&mLock);
|
2014-01-27 22:54:14 +01:00
|
|
|
switch (component)
|
|
|
|
{
|
|
|
|
|
|
|
|
case Wizard::Component_Morrowind:
|
|
|
|
mInstallMorrowind = install;
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Tribunal:
|
|
|
|
mInstallTribunal = install;
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Bloodmoon:
|
|
|
|
mInstallBloodmoon = install;
|
|
|
|
break;
|
|
|
|
}
|
2013-12-25 18:52:34 +01:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
bool Wizard::UnshieldWorker::getInstallComponent(Component component)
|
2014-01-24 22:25:22 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QReadLocker readLock(&mLock);
|
2014-01-27 22:54:14 +01:00
|
|
|
switch (component)
|
|
|
|
{
|
|
|
|
|
|
|
|
case Wizard::Component_Morrowind:
|
|
|
|
return mInstallMorrowind;
|
|
|
|
case Wizard::Component_Tribunal:
|
|
|
|
return mInstallTribunal;
|
|
|
|
case Wizard::Component_Bloodmoon:
|
|
|
|
return mInstallBloodmoon;
|
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
return false;
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
void Wizard::UnshieldWorker::setComponentDone(Component component, bool done)
|
2014-01-24 22:25:22 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QWriteLocker writeLock(&mLock);
|
2014-01-27 22:54:14 +01:00
|
|
|
switch (component)
|
|
|
|
{
|
|
|
|
|
|
|
|
case Wizard::Component_Morrowind:
|
|
|
|
mMorrowindDone = done;
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Tribunal:
|
|
|
|
mTribunalDone = done;
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Bloodmoon:
|
|
|
|
mBloodmoonDone = done;
|
|
|
|
break;
|
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
bool Wizard::UnshieldWorker::getComponentDone(Component component)
|
2014-01-24 22:25:22 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QReadLocker readLock(&mLock);
|
2014-01-27 22:54:14 +01:00
|
|
|
switch (component)
|
|
|
|
{
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
case Wizard::Component_Morrowind:
|
|
|
|
return mMorrowindDone;
|
|
|
|
case Wizard::Component_Tribunal:
|
|
|
|
return mTribunalDone;
|
|
|
|
case Wizard::Component_Bloodmoon:
|
|
|
|
return mBloodmoonDone;
|
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
return false;
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
|
|
|
|
2014-01-01 22:46:29 +01:00
|
|
|
void Wizard::UnshieldWorker::setPath(const QString& path)
|
2013-12-25 18:52:34 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QWriteLocker writeLock(&mLock);
|
2013-12-25 18:52:34 +01:00
|
|
|
mPath = path;
|
|
|
|
}
|
|
|
|
|
2014-01-01 22:46:29 +01:00
|
|
|
void Wizard::UnshieldWorker::setIniPath(const QString& path)
|
2013-12-26 18:02:34 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QWriteLocker writeLock(&mLock);
|
2013-12-26 18:02:34 +01:00
|
|
|
mIniPath = path;
|
|
|
|
}
|
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
void Wizard::UnshieldWorker::setDiskPath(const QString& path)
|
|
|
|
{
|
|
|
|
QWriteLocker writeLock(&mLock);
|
|
|
|
mDiskPath = path;
|
|
|
|
mWait.wakeAll();
|
|
|
|
}
|
|
|
|
|
2014-01-27 15:12:02 +01:00
|
|
|
QString Wizard::UnshieldWorker::getPath()
|
|
|
|
{
|
|
|
|
QReadLocker readLock(&mLock);
|
|
|
|
return mPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Wizard::UnshieldWorker::getIniPath()
|
|
|
|
{
|
|
|
|
QReadLocker readLock(&mLock);
|
|
|
|
return mIniPath;
|
|
|
|
}
|
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
QString Wizard::UnshieldWorker::getDiskPath()
|
|
|
|
{
|
|
|
|
QReadLocker readLock(&mLock);
|
|
|
|
return mDiskPath;
|
|
|
|
}
|
|
|
|
|
2023-01-19 15:07:12 +04:00
|
|
|
void Wizard::UnshieldWorker::setIniEncoding(ToUTF8::FromType encoding)
|
2013-12-26 18:02:34 +01:00
|
|
|
{
|
2014-01-27 15:12:02 +01:00
|
|
|
QWriteLocker writeLock(&mLock);
|
2023-01-19 15:07:12 +04:00
|
|
|
mIniEncoding = encoding;
|
2013-12-26 18:02:34 +01:00
|
|
|
}
|
|
|
|
|
2022-04-17 16:28:14 +00:00
|
|
|
void Wizard::UnshieldWorker::wakeAll()
|
|
|
|
{
|
|
|
|
mWait.wakeAll();
|
|
|
|
}
|
|
|
|
|
2014-03-30 22:58:50 +02:00
|
|
|
bool Wizard::UnshieldWorker::setupSettings()
|
2013-12-26 18:02:34 +01:00
|
|
|
{
|
|
|
|
// Create Morrowind.ini settings map
|
2014-01-27 15:12:02 +01:00
|
|
|
if (getIniPath().isEmpty())
|
2014-03-30 22:58:50 +02:00
|
|
|
return false;
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2023-01-19 15:07:12 +04:00
|
|
|
const auto iniPath = Files::pathFromQString(getIniPath());
|
|
|
|
std::ifstream file(iniPath);
|
|
|
|
if (file.fail())
|
2013-12-26 18:02:34 +01:00
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
emit error(tr("Failed to open Morrowind configuration file!"),
|
2023-01-29 21:56:59 +01:00
|
|
|
tr("Opening %1 failed: %2.").arg(getIniPath(), std::generic_category().message(errno).c_str()));
|
2014-03-30 22:58:50 +02:00
|
|
|
return false;
|
2013-12-26 18:02:34 +01:00
|
|
|
}
|
|
|
|
|
2023-01-19 15:07:12 +04:00
|
|
|
mIniSettings.readFile(file, mIniEncoding);
|
2014-03-30 22:58:50 +02:00
|
|
|
|
|
|
|
return true;
|
2013-12-26 18:02:34 +01:00
|
|
|
}
|
|
|
|
|
2014-03-30 22:58:50 +02:00
|
|
|
bool Wizard::UnshieldWorker::writeSettings()
|
2014-02-18 11:55:26 +01:00
|
|
|
{
|
|
|
|
if (getIniPath().isEmpty())
|
2014-03-30 22:58:50 +02:00
|
|
|
return false;
|
2014-02-18 11:55:26 +01:00
|
|
|
|
2023-01-19 15:07:12 +04:00
|
|
|
const auto iniPath = Files::pathFromQString(getIniPath());
|
|
|
|
std::ifstream file(iniPath);
|
|
|
|
if (file.fail())
|
2014-02-18 11:55:26 +01:00
|
|
|
{
|
|
|
|
emit error(tr("Failed to open Morrowind configuration file!"),
|
2023-01-29 21:56:59 +01:00
|
|
|
tr("Opening %1 failed: %2.").arg(getIniPath(), std::generic_category().message(errno).c_str()));
|
2014-03-30 22:58:50 +02:00
|
|
|
return false;
|
2014-02-18 11:55:26 +01:00
|
|
|
}
|
|
|
|
|
2023-01-19 15:07:12 +04:00
|
|
|
if (!mIniSettings.writeFile(getIniPath(), file, mIniEncoding))
|
2014-02-18 11:55:26 +01:00
|
|
|
{
|
|
|
|
emit error(tr("Failed to write Morrowind configuration file!"),
|
2023-01-29 21:56:59 +01:00
|
|
|
tr("Writing to %1 failed: %2.").arg(getIniPath(), std::generic_category().message(errno).c_str()));
|
2014-03-30 22:58:50 +02:00
|
|
|
return false;
|
2014-02-18 11:55:26 +01:00
|
|
|
}
|
2014-03-30 22:58:50 +02:00
|
|
|
|
|
|
|
return true;
|
2014-02-18 11:55:26 +01:00
|
|
|
}
|
|
|
|
|
2014-01-01 22:46:29 +01:00
|
|
|
bool Wizard::UnshieldWorker::removeDirectory(const QString& dirName)
|
2014-01-01 17:15:34 +01:00
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
bool result = false;
|
2014-01-01 17:15:34 +01:00
|
|
|
QDir dir(dirName);
|
|
|
|
|
|
|
|
if (dir.exists(dirName))
|
|
|
|
{
|
|
|
|
QFileInfoList list(dir.entryInfoList(
|
|
|
|
QDir::NoDotAndDotDot | QDir::System | QDir::Hidden | QDir::AllDirs | QDir::Files, QDir::DirsFirst));
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QFileInfo& info : list)
|
|
|
|
{
|
2014-01-01 17:15:34 +01:00
|
|
|
if (info.isDir())
|
|
|
|
{
|
|
|
|
result = removeDirectory(info.absoluteFilePath());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
result = QFile::remove(info.absoluteFilePath());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!result)
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = dir.rmdir(dirName);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-01-17 13:21:44 +01:00
|
|
|
bool Wizard::UnshieldWorker::copyFile(const QString& source, const QString& destination, bool keepSource)
|
2014-01-01 17:15:34 +01:00
|
|
|
{
|
|
|
|
QDir dir;
|
|
|
|
QFile file;
|
|
|
|
|
2014-01-17 13:21:44 +01:00
|
|
|
QFileInfo info(destination);
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
if (info.exists())
|
|
|
|
{
|
|
|
|
if (!dir.remove(info.absoluteFilePath()))
|
|
|
|
return false;
|
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
|
|
|
|
if (file.copy(source, destination))
|
|
|
|
{
|
|
|
|
if (!keepSource)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
if (!file.remove(source))
|
|
|
|
return false;
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-01-17 13:21:44 +01:00
|
|
|
return true;
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
return false;
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
return true;
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
|
2014-01-17 13:21:44 +01:00
|
|
|
bool Wizard::UnshieldWorker::copyDirectory(const QString& source, const QString& destination, bool keepSource)
|
2014-01-01 17:15:34 +01:00
|
|
|
{
|
|
|
|
QDir sourceDir(source);
|
|
|
|
QDir destDir(destination);
|
2014-03-17 21:33:44 +01:00
|
|
|
bool result = true;
|
2014-01-01 17:15:34 +01:00
|
|
|
|
|
|
|
if (!destDir.exists())
|
|
|
|
{
|
2014-03-17 21:33:44 +01:00
|
|
|
if (!sourceDir.mkpath(destination))
|
2014-03-17 14:31:05 +01:00
|
|
|
return false;
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
|
2014-03-17 21:33:44 +01:00
|
|
|
destDir.refresh();
|
|
|
|
|
2014-01-01 17:15:34 +01:00
|
|
|
if (!destDir.exists())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
QFileInfoList list(sourceDir.entryInfoList(
|
|
|
|
QDir::NoDotAndDotDot | QDir::System | QDir::Hidden | QDir::AllDirs | QDir::Files, QDir::DirsFirst));
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QFileInfo& info : list)
|
|
|
|
{
|
2014-01-01 17:15:34 +01:00
|
|
|
QString relativePath(info.absoluteFilePath());
|
|
|
|
relativePath.remove(source);
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
QString destinationPath(destDir.absolutePath() + relativePath);
|
|
|
|
|
2014-01-01 17:15:34 +01:00
|
|
|
if (info.isSymLink())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (info.isDir())
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
result = copyDirectory(info.absoluteFilePath(), destinationPath);
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
result = copyFile(info.absoluteFilePath(), destinationPath);
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-17 13:21:44 +01:00
|
|
|
if (!keepSource)
|
|
|
|
return result && removeDirectory(sourceDir.absolutePath());
|
|
|
|
|
|
|
|
return result;
|
2014-01-01 17:15:34 +01:00
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
bool Wizard::UnshieldWorker::installFile(
|
|
|
|
const QString& fileName, const QString& path, Qt::MatchFlags flags, bool keepSource)
|
2013-12-25 18:52:34 +01:00
|
|
|
{
|
2016-09-15 19:03:07 +03:00
|
|
|
return installFiles(fileName, path, flags, keepSource, true);
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
2013-12-25 18:52:34 +01:00
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
bool Wizard::UnshieldWorker::installFiles(
|
|
|
|
const QString& fileName, const QString& path, Qt::MatchFlags flags, bool keepSource, bool single)
|
2014-01-17 13:21:44 +01:00
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
QDir dir(path);
|
2013-12-25 18:52:34 +01:00
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!dir.exists())
|
|
|
|
return false;
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
QStringList files(findFiles(fileName, path, flags));
|
2013-12-25 18:52:34 +01:00
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& file : files)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
QFileInfo info(file);
|
|
|
|
emit textChanged(tr("Installing: %1").arg(info.fileName()));
|
|
|
|
|
|
|
|
if (single)
|
|
|
|
{
|
|
|
|
return copyFile(info.absoluteFilePath(), getPath() + QDir::separator() + info.fileName(), keepSource);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!copyFile(info.absoluteFilePath(), getPath() + QDir::separator() + info.fileName(), keepSource))
|
|
|
|
return false;
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
return true;
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
bool Wizard::UnshieldWorker::installDirectories(
|
|
|
|
const QString& dirName, const QString& path, bool recursive, bool keepSource)
|
2014-01-17 13:21:44 +01:00
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
QDir dir(path);
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!dir.exists())
|
|
|
|
return false;
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
QStringList directories(findDirectories(dirName, path, recursive));
|
2014-01-28 01:03:47 +01:00
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& dir : directories)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
QFileInfo info(dir);
|
|
|
|
emit textChanged(tr("Installing: %1 directory").arg(info.fileName()));
|
|
|
|
if (!copyDirectory(info.absoluteFilePath(), getPath() + QDir::separator() + info.fileName(), keepSource))
|
|
|
|
return false;
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
return true;
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
void Wizard::UnshieldWorker::extract()
|
|
|
|
{
|
2014-03-16 22:09:20 +01:00
|
|
|
if (getInstallComponent(Wizard::Component_Morrowind))
|
|
|
|
{
|
|
|
|
if (!getComponentDone(Wizard::Component_Morrowind))
|
|
|
|
if (!setupComponent(Wizard::Component_Morrowind))
|
|
|
|
return;
|
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
if (getInstallComponent(Wizard::Component_Tribunal))
|
|
|
|
{
|
|
|
|
if (!getComponentDone(Wizard::Component_Tribunal))
|
|
|
|
if (!setupComponent(Wizard::Component_Tribunal))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getInstallComponent(Wizard::Component_Bloodmoon))
|
|
|
|
{
|
|
|
|
if (!getComponentDone(Wizard::Component_Bloodmoon))
|
|
|
|
if (!setupComponent(Wizard::Component_Bloodmoon))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update Morrowind configuration
|
|
|
|
if (getInstallComponent(Wizard::Component_Tribunal))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
mIniSettings.setValue(QLatin1String("Archives/Archive 0"), QVariant(QString("Tribunal.bsa")));
|
|
|
|
mIniSettings.setValue(QLatin1String("Game Files/GameFile1"), QVariant(QString("Tribunal.esm")));
|
2014-03-16 22:09:20 +01:00
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
if (getInstallComponent(Wizard::Component_Bloodmoon))
|
2014-01-27 22:54:14 +01:00
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
mIniSettings.setValue(QLatin1String("Archives/Archive 0"), QVariant(QString("Bloodmoon.bsa")));
|
|
|
|
mIniSettings.setValue(QLatin1String("Game Files/GameFile1"), QVariant(QString("Bloodmoon.esm")));
|
2014-03-16 22:09:20 +01:00
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
if (getInstallComponent(Wizard::Component_Tribunal) && getInstallComponent(Wizard::Component_Bloodmoon))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
mIniSettings.setValue(QLatin1String("Archives/Archive 0"), QVariant(QString("Tribunal.bsa")));
|
|
|
|
mIniSettings.setValue(QLatin1String("Archives/Archive 1"), QVariant(QString("Bloodmoon.bsa")));
|
|
|
|
mIniSettings.setValue(QLatin1String("Game Files/GameFile1"), QVariant(QString("Tribunal.esm")));
|
|
|
|
mIniSettings.setValue(QLatin1String("Game Files/GameFile2"), QVariant(QString("Bloodmoon.esm")));
|
2014-03-16 22:09:20 +01:00
|
|
|
}
|
2014-01-27 22:54:14 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
// Write the settings to the Morrowind config file
|
2014-03-30 22:58:50 +02:00
|
|
|
if (!writeSettings())
|
2014-03-30 23:11:38 +02:00
|
|
|
return;
|
2014-01-27 22:54:14 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
// Remove the temporary directory
|
2014-03-17 23:39:21 +01:00
|
|
|
removeDirectory(getPath() + QDir::separator() + QLatin1String("extract-temp"));
|
2014-01-27 22:54:14 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
// Fill the progress bar
|
|
|
|
int total = 0;
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
if (getInstallComponent(Wizard::Component_Morrowind))
|
|
|
|
total = 100;
|
|
|
|
|
|
|
|
if (getInstallComponent(Wizard::Component_Tribunal))
|
|
|
|
total = total + 100;
|
|
|
|
|
|
|
|
if (getInstallComponent(Wizard::Component_Bloodmoon))
|
|
|
|
total = total + 100;
|
|
|
|
|
|
|
|
emit textChanged(tr("Installation finished!"));
|
|
|
|
emit progressChanged(total);
|
|
|
|
emit finished();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Wizard::UnshieldWorker::setupComponent(Component component)
|
|
|
|
{
|
|
|
|
QString name;
|
|
|
|
switch (component)
|
|
|
|
{
|
|
|
|
|
|
|
|
case Wizard::Component_Morrowind:
|
|
|
|
name = QLatin1String("Morrowind");
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Tribunal:
|
|
|
|
name = QLatin1String("Tribunal");
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Bloodmoon:
|
|
|
|
name = QLatin1String("Bloodmoon");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (name.isEmpty())
|
|
|
|
{
|
|
|
|
emit error(tr("Component parameter is invalid!"), tr("An invalid component parameter was supplied."));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
QString cabFile;
|
|
|
|
QDir disk;
|
|
|
|
|
|
|
|
// Keep showing the file dialog until we find the necessary install files
|
|
|
|
while (!found)
|
|
|
|
{
|
|
|
|
if (getDiskPath().isEmpty())
|
|
|
|
{
|
|
|
|
QReadLocker readLock(&mLock);
|
2014-03-16 20:07:54 +01:00
|
|
|
emit requestFileDialog(component);
|
2014-01-27 22:54:14 +01:00
|
|
|
mWait.wait(&mLock);
|
2017-06-13 12:00:55 +02:00
|
|
|
if (mStopped)
|
|
|
|
{
|
|
|
|
qDebug() << "We are asked to stop !!";
|
|
|
|
break;
|
|
|
|
}
|
2014-03-16 22:09:20 +01:00
|
|
|
disk.setPath(getDiskPath());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
disk.setPath(getDiskPath());
|
2014-01-27 22:54:14 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
QStringList list(findFiles(QLatin1String("data1.hdr"), disk.absolutePath()));
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& file : list)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
|
2014-03-17 23:39:21 +01:00
|
|
|
qDebug() << "current archive: " << file;
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
if (component == Wizard::Component_Morrowind)
|
|
|
|
{
|
2014-03-30 23:11:38 +02:00
|
|
|
bool morrowindFound = findInCab(QLatin1String("Morrowind.bsa"), file);
|
|
|
|
bool tribunalFound = findInCab(QLatin1String("Tribunal.bsa"), file);
|
|
|
|
bool bloodmoonFound = findInCab(QLatin1String("Bloodmoon.bsa"), file);
|
2014-03-17 14:31:05 +01:00
|
|
|
|
|
|
|
if (morrowindFound)
|
|
|
|
{
|
|
|
|
// Check if we have correct archive, other archives have Morrowind.bsa too
|
Some PVS-Studio and cppcheck fixes
cppcheck:
[apps/esmtool/record.cpp:697]: (performance) Prefer prefix ++/-- operators for non-primitive types.
[apps/esmtool/record.cpp:1126]: (performance) Prefer prefix ++/-- operators for non-primitive types.
[apps/esmtool/record.cpp:1138]: (performance) Prefer prefix ++/-- operators for non-primitive types.
[apps/niftest/niftest.cpp:36]: (performance) Function parameter 'filename' should be passed by reference.
[apps/niftest/niftest.cpp:41]: (performance) Function parameter 'filename' should be passed by reference.
[apps/opencs/model/prefs/boolsetting.cpp:25]: (warning) Possible leak in public function. The pointer 'mWidget' is not deallocated before it is allocated.
[apps/opencs/model/prefs/shortcuteventhandler.cpp:52]: (warning) Return value of std::remove() ignored. Elements remain in container.
[apps/openmw/mwstate/quicksavemanager.cpp:5]: (performance) Variable 'mSaveName' is assigned in constructor body. Consider performing initialization in initialization list.
PVS-Studio:
apps/opencs/model/filter/parser.cpp 582 warn V560 A part of conditional expression is always true: allowPredefined.
apps/opencs/view/world/referencecreator.cpp 67 warn V547 Expression '!errors.empty()' is always false.
apps/opencs/view/world/referencecreator.cpp 74 warn V547 Expression '!errors.empty()' is always false.
apps/opencs/view/doc/loader.cpp 170 warn V560 A part of conditional expression is always true: !completed.
apps/opencs/view/doc/loader.cpp 170 warn V560 A part of conditional expression is always true: !error.empty().
apps/opencs/model/tools/pathgridcheck.cpp 32 err V517 The use of 'if (A) {...} else if (A) {...}' pattern was detected. There is a probability of logical error presence. Check lines: 32, 34.
apps/opencs/model/world/refidadapterimp.cpp 1376 err V547 Expression 'subColIndex < 3' is always true.
apps/openmw/mwgui/widgets.hpp 318 warn V703 It is odd that the 'mEnableRepeat' field in derived class 'MWScrollBar' overwrites field in base class 'ScrollBar'. Check lines: widgets.hpp:318, MyGUI_ScrollBar.h:179.
apps/openmw/mwgui/widgets.hpp 319 warn V703 It is odd that the 'mRepeatTriggerTime' field in derived class 'MWScrollBar' overwrites field in base class 'ScrollBar'. Check lines: widgets.hpp:319, MyGUI_ScrollBar.h:180.
apps/openmw/mwgui/widgets.hpp 320 warn V703 It is odd that the 'mRepeatStepTime' field in derived class 'MWScrollBar' overwrites field in base class 'ScrollBar'. Check lines: widgets.hpp:320, MyGUI_ScrollBar.h:181
apps/openmw/mwmechanics/actors.cpp 1425 warn V547 Expression '!detected' is always true.
apps/openmw/mwmechanics/character.cpp 2155 err V547 Expression 'mode == 0' is always true.
apps/openmw/mwmechanics/character.cpp 1192 warn V592 The expression was enclosed by parentheses twice: ((expression)). One pair of parentheses is unnecessary or misprint is present.
apps/openmw/mwmechanics/character.cpp 521 warn V560 A part of conditional expression is always true: (idle == mIdleState).
apps/openmw/mwmechanics/pathfinding.cpp 317 err V547 Expression 'mPath.size() >= 2' is always true.
apps/openmw/mwscript/interpretercontext.cpp 409 warn V560 A part of conditional expression is always false: rank > 9.
apps/openmw/mwgui/windowbase.cpp 28 warn V560 A part of conditional expression is always true: !visible.
apps/openmw/mwgui/journalwindow.cpp 561 warn V547 Expression '!mAllQuests' is always false.
apps/openmw/mwgui/referenceinterface.cpp 18 warn V571 Recurring check. The '!mPtr.isEmpty()' condition was already verified in line 16.
apps/openmw/mwworld/scene.cpp 463 warn V547 Expression 'adjustPlayerPos' is always true.
apps/openmw/mwworld/worldimp.cpp 409 err V766 An item with the same key '"sCompanionShare"' has already been added.
apps/openmw/mwworld/cellstore.cpp 691 warn V519 The 'state.mWaterLevel' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 689, 691.
apps/openmw/mwworld/weather.cpp 1125 warn V519 The 'mResult.mParticleEffect' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 1123, 1125.
apps/openmw/mwworld/weather.cpp 1137 warn V519 The 'mResult.mParticleEffect' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 1135, 1137.
apps/wizard/unshield/unshieldworker.cpp 475 warn V728 An excessive check can be simplified. The '(A && B) || (!A && !B)' expression is equivalent to the 'bool(A) == bool(B)' expression.
apps/wizard/installationpage.cpp 163 warn V735 Possibly an incorrect HTML. The "</p" closing tag was encountered, while the "</span" tag was expected.
components/fontloader/fontloader.cpp 427 err V547 Expression 'i == 1' is always true.
components/nifosg/nifloader.cpp 282 warn V519 The 'created' variable is assigned values twice successively. Perhaps this is a mistake. Check lines: 278, 282.
components/esm/loadregn.cpp 119 err V586 The 'clear' function is called twice for deallocation of the same resource. Check lines: 112, 119.
components/esm/cellref.cpp 178 warn V581 The conditional expressions of the 'if' statements situated alongside each other are identical. Check lines: 175, 178.
components/esmterrain/storage.cpp 235 warn V560 A part of conditional expression is always true: colStart == 0.
components/esmterrain/storage.cpp 237 warn V560 A part of conditional expression is always true: rowStart == 0.
2018-04-09 18:55:16 +03:00
|
|
|
if (tribunalFound == bloodmoonFound)
|
|
|
|
{
|
2022-04-17 16:28:14 +00:00
|
|
|
qint64 actualFileSize = getMorrowindBsaFileSize(file);
|
|
|
|
if (actualFileSize != mExpectedMorrowindBsaSize)
|
|
|
|
{
|
|
|
|
QReadLocker readLock(&mLock);
|
|
|
|
emit requestOldVersionDialog();
|
|
|
|
mWait.wait(&mLock);
|
|
|
|
if (mStopped)
|
|
|
|
{
|
|
|
|
qDebug() << "We are asked to stop !!";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-03-17 14:31:05 +01:00
|
|
|
cabFile = file;
|
2014-03-17 23:39:21 +01:00
|
|
|
found = true; // We have a GoTY disk or a Morrowind-only disk
|
2014-03-17 14:31:05 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-03-17 23:39:21 +01:00
|
|
|
|
2014-03-30 23:11:38 +02:00
|
|
|
if (findInCab(name + QLatin1String(".bsa"), file))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
cabFile = file;
|
|
|
|
found = true;
|
|
|
|
}
|
2014-01-27 22:54:14 +01:00
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
|
|
|
|
2022-04-17 16:28:14 +00:00
|
|
|
if (cabFile.isEmpty())
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-01-22 23:39:12 +03:00
|
|
|
if (!found)
|
|
|
|
{
|
|
|
|
emit textChanged(tr("Failed to find a valid archive containing %1.bsa! Retrying.").arg(name));
|
2014-03-16 22:09:20 +01:00
|
|
|
QReadLocker readLock(&mLock);
|
|
|
|
emit requestFileDialog(component);
|
|
|
|
mWait.wait(&mLock);
|
2014-01-27 22:54:14 +01:00
|
|
|
}
|
2014-03-16 22:09:20 +01:00
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
if (installComponent(component, cabFile))
|
|
|
|
{
|
|
|
|
setComponentDone(component, true);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return false;
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
2014-03-16 22:09:20 +01:00
|
|
|
|
|
|
|
return true;
|
2014-01-27 22:54:14 +01:00
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
bool Wizard::UnshieldWorker::installComponent(Component component, const QString& path)
|
2014-01-27 22:54:14 +01:00
|
|
|
{
|
|
|
|
QString name;
|
|
|
|
switch (component)
|
|
|
|
{
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
case Wizard::Component_Morrowind:
|
|
|
|
name = QLatin1String("Morrowind");
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Tribunal:
|
|
|
|
name = QLatin1String("Tribunal");
|
|
|
|
break;
|
|
|
|
case Wizard::Component_Bloodmoon:
|
|
|
|
name = QLatin1String("Bloodmoon");
|
|
|
|
break;
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-28 01:03:47 +01:00
|
|
|
if (name.isEmpty())
|
|
|
|
{
|
|
|
|
emit error(tr("Component parameter is invalid!"), tr("An invalid component parameter was supplied."));
|
2014-01-24 22:25:22 +01:00
|
|
|
return false;
|
2014-01-28 01:03:47 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-02-25 15:33:30 +01:00
|
|
|
emit textChanged(tr("Installing %1").arg(name));
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
QFileInfo info(path);
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
if (!info.exists())
|
|
|
|
{
|
|
|
|
emit error(tr("Installation media path not set!"), tr("The source path for %1 was not set.").arg(name));
|
2014-01-24 22:25:22 +01:00
|
|
|
return false;
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
// Create temporary extract directory
|
|
|
|
// TODO: Use QTemporaryDir in Qt 5.0
|
2014-01-27 15:12:02 +01:00
|
|
|
QString tempPath(getPath() + QDir::separator() + QLatin1String("extract-temp"));
|
2014-01-24 22:25:22 +01:00
|
|
|
QDir temp;
|
2014-01-01 17:15:34 +01:00
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
// Make sure the temporary folder is empty
|
|
|
|
removeDirectory(tempPath);
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
if (!temp.mkpath(tempPath))
|
|
|
|
{
|
2014-02-25 15:33:30 +01:00
|
|
|
emit error(tr("Cannot create temporary directory!"), tr("Failed to create %1.").arg(tempPath));
|
2014-01-24 22:25:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-01-01 22:46:29 +01:00
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
temp.setPath(tempPath);
|
2014-01-01 17:15:34 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
if (!temp.mkdir(name))
|
|
|
|
{
|
2014-02-25 15:33:30 +01:00
|
|
|
emit error(
|
|
|
|
tr("Cannot create temporary directory!"), tr("Failed to create %1.").arg(temp.absoluteFilePath(name)));
|
2014-01-24 22:25:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
if (!temp.cd(name))
|
|
|
|
{
|
2014-02-25 15:33:30 +01:00
|
|
|
emit error(tr("Cannot move into temporary directory!"),
|
|
|
|
tr("Failed to move into %1.").arg(temp.absoluteFilePath(name)));
|
2014-01-24 22:25:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
// Extract the installation files
|
2014-03-16 22:09:20 +01:00
|
|
|
if (!extractCab(info.absoluteFilePath(), temp.absolutePath()))
|
2014-01-28 01:03:47 +01:00
|
|
|
return false;
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
// Move the files from the temporary path to the destination folder
|
2014-01-27 15:12:02 +01:00
|
|
|
emit textChanged(tr("Moving installation files"));
|
2014-03-16 20:07:54 +01:00
|
|
|
|
|
|
|
// Install extracted directories
|
|
|
|
QStringList directories;
|
|
|
|
directories << QLatin1String("BookArt") << QLatin1String("Fonts") << QLatin1String("Icons")
|
|
|
|
<< QLatin1String("Meshes") << QLatin1String("Music") << QLatin1String("Sound")
|
|
|
|
<< QLatin1String("Splash") << QLatin1String("Textures") << QLatin1String("Video");
|
2022-09-22 21:26:05 +03:00
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& dir : directories)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
if (!installDirectories(dir, temp.absolutePath()))
|
|
|
|
{
|
|
|
|
emit error(
|
|
|
|
tr("Could not install directory!"), tr("Installing %1 to %2 failed.").arg(dir, temp.absolutePath()));
|
|
|
|
return false;
|
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
}
|
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
// Install directories from disk
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& dir : directories)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
if (!installDirectories(dir, info.absolutePath(), false, true))
|
|
|
|
{
|
|
|
|
emit error(
|
|
|
|
tr("Could not install directory!"), tr("Installing %1 to %2 failed.").arg(dir, info.absolutePath()));
|
2014-03-17 21:33:44 +01:00
|
|
|
return false;
|
2014-03-17 14:31:05 +01:00
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
// Install translation files
|
|
|
|
QStringList extensions;
|
|
|
|
extensions << QLatin1String(".cel") << QLatin1String(".top") << QLatin1String(".mrk");
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& extension : extensions)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
if (!installFiles(extension, info.absolutePath(), Qt::MatchEndsWith))
|
|
|
|
{
|
|
|
|
emit error(tr("Could not install translation file!"), tr("Failed to install *%1 files.").arg(extension));
|
|
|
|
return false;
|
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
if (component == Wizard::Component_Morrowind)
|
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
QStringList files;
|
|
|
|
files << QLatin1String("Morrowind.esm") << QLatin1String("Morrowind.bsa");
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& file : files)
|
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!installFile(file, temp.absolutePath()))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
emit error(tr("Could not install Morrowind data file!"), tr("Failed to install %1.").arg(file));
|
2014-03-16 20:07:54 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-02-25 17:23:26 +01:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
// Copy Morrowind configuration file
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!installFile(QLatin1String("Morrowind.ini"), temp.absolutePath()))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
emit error(tr("Could not install Morrowind configuration file!"),
|
|
|
|
tr("Failed to install %1.").arg(QLatin1String("Morrowind.ini")));
|
2014-01-27 22:54:14 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-02-18 11:55:26 +01:00
|
|
|
|
|
|
|
// Setup Morrowind configuration
|
|
|
|
setIniPath(getPath() + QDir::separator() + QLatin1String("Morrowind.ini"));
|
2014-03-30 22:58:50 +02:00
|
|
|
|
|
|
|
if (!setupSettings())
|
|
|
|
return false;
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-01-28 13:36:14 +01:00
|
|
|
if (component == Wizard::Component_Tribunal)
|
|
|
|
{
|
|
|
|
QFileInfo sounds(temp.absoluteFilePath(QLatin1String("Sounds")));
|
2014-03-17 14:31:05 +01:00
|
|
|
QString dest(getPath() + QDir::separator() + QLatin1String("Sound"));
|
2014-01-28 13:36:14 +01:00
|
|
|
|
|
|
|
if (sounds.exists())
|
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
emit textChanged(tr("Installing: Sound directory"));
|
2014-03-17 14:31:05 +01:00
|
|
|
if (!copyDirectory(sounds.absoluteFilePath(), dest))
|
|
|
|
{
|
|
|
|
emit error(tr("Could not install directory!"),
|
|
|
|
tr("Installing %1 to %2 failed.").arg(sounds.absoluteFilePath(), dest));
|
2014-03-17 21:33:44 +01:00
|
|
|
return false;
|
2014-03-17 14:31:05 +01:00
|
|
|
}
|
2014-01-28 13:36:14 +01:00
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
|
|
|
|
QStringList files;
|
|
|
|
files << QLatin1String("Tribunal.esm") << QLatin1String("Tribunal.bsa");
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& file : files)
|
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!installFile(file, temp.absolutePath()))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
emit error(tr("Could not find Tribunal data file!"), tr("Failed to find %1.").arg(file));
|
2014-03-16 20:07:54 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2014-01-28 13:36:14 +01:00
|
|
|
}
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
if (component == Wizard::Component_Bloodmoon)
|
|
|
|
{
|
|
|
|
QFileInfo original(getPath() + QDir::separator() + QLatin1String("Tribunal.esm"));
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
if (original.exists())
|
|
|
|
{
|
|
|
|
if (!installFile(QLatin1String("Tribunal.esm"), temp.absolutePath()))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
emit error(tr("Could not find Tribunal patch file!"),
|
|
|
|
tr("Failed to find %1.").arg(QLatin1String("Tribunal.esm")));
|
2014-03-16 20:07:54 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2014-02-25 17:36:55 +01:00
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
QStringList files;
|
|
|
|
files << QLatin1String("Bloodmoon.esm") << QLatin1String("Bloodmoon.bsa");
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& file : files)
|
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!installFile(file, temp.absolutePath()))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
emit error(tr("Could not find Bloodmoon data file!"), tr("Failed to find %1.").arg(file));
|
2014-03-16 20:07:54 +01:00
|
|
|
return false;
|
|
|
|
}
|
2014-01-27 22:54:14 +01:00
|
|
|
}
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-02-18 11:55:26 +01:00
|
|
|
// Load Morrowind configuration settings from the setup script
|
2014-03-16 22:09:20 +01:00
|
|
|
QStringList list(findFiles(QLatin1String("setup.inx"), getDiskPath()));
|
2014-02-18 11:55:26 +01:00
|
|
|
|
2014-03-16 22:09:20 +01:00
|
|
|
emit textChanged(tr("Updating Morrowind configuration file"));
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& inx : list)
|
|
|
|
{
|
2014-03-16 22:09:20 +01:00
|
|
|
mIniSettings.parseInx(inx);
|
2014-02-18 11:55:26 +01:00
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
// Finally, install Data Files directories from temp and disk
|
|
|
|
QStringList datafiles(findDirectories(QLatin1String("Data Files"), temp.absolutePath()));
|
|
|
|
datafiles.append(findDirectories(QLatin1String("Data Files"), info.absolutePath()));
|
|
|
|
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QString& dir : datafiles)
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
QFileInfo info(dir);
|
|
|
|
emit textChanged(tr("Installing: %1 directory").arg(info.fileName()));
|
|
|
|
|
|
|
|
if (!copyDirectory(info.absoluteFilePath(), getPath()))
|
|
|
|
{
|
|
|
|
emit error(tr("Could not install directory!"),
|
|
|
|
tr("Installing %1 to %2 failed.").arg(info.absoluteFilePath(), getPath()));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-25 15:33:30 +01:00
|
|
|
emit textChanged(tr("%1 installation finished!").arg(name));
|
2014-01-24 22:25:22 +01:00
|
|
|
return true;
|
2014-01-27 22:54:14 +01:00
|
|
|
}
|
2014-01-24 22:25:22 +01:00
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
bool Wizard::UnshieldWorker::extractFile(
|
|
|
|
Unshield* unshield, const QString& destination, const QString& prefix, int index, int counter)
|
2013-12-26 18:02:34 +01:00
|
|
|
{
|
2014-03-17 21:33:44 +01:00
|
|
|
bool success = false;
|
2014-03-16 20:07:54 +01:00
|
|
|
QString path(destination);
|
2014-01-01 17:15:34 +01:00
|
|
|
path.append(QDir::separator());
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
int directory = unshield_file_directory(unshield, index);
|
|
|
|
|
|
|
|
if (!prefix.isEmpty())
|
2014-01-01 17:15:34 +01:00
|
|
|
path.append(prefix + QDir::separator());
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
if (directory >= 0)
|
2014-02-24 15:59:44 +01:00
|
|
|
path.append(QString::fromUtf8(unshield_directory_name(unshield, directory)) + QDir::separator());
|
2014-01-01 17:15:34 +01:00
|
|
|
|
|
|
|
// Ensure the path has the right separators
|
|
|
|
path.replace(QLatin1Char('\\'), QDir::separator());
|
|
|
|
path = QDir::toNativeSeparators(path);
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
// Ensure the target path exists
|
|
|
|
QDir dir;
|
2015-01-27 19:00:26 +01:00
|
|
|
if (!dir.mkpath(path))
|
|
|
|
return false;
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
QString fileName(path);
|
2014-02-24 15:59:44 +01:00
|
|
|
fileName.append(QString::fromUtf8(unshield_file_name(unshield, index)));
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
// Calculate the percentage done
|
2014-01-27 15:12:02 +01:00
|
|
|
int progress = (((float)counter / (float)unshield_file_count(unshield)) * 100);
|
2014-01-17 13:21:44 +01:00
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
if (getComponentDone(Wizard::Component_Morrowind))
|
2014-01-17 13:21:44 +01:00
|
|
|
progress = progress + 100;
|
|
|
|
|
2014-01-27 22:54:14 +01:00
|
|
|
if (getComponentDone(Wizard::Component_Tribunal))
|
2014-01-17 13:21:44 +01:00
|
|
|
progress = progress + 100;
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-02-24 15:59:44 +01:00
|
|
|
emit textChanged(tr("Extracting: %1").arg(QString::fromUtf8(unshield_file_name(unshield, index))));
|
2013-12-26 18:02:34 +01:00
|
|
|
emit progressChanged(progress);
|
|
|
|
|
2014-02-24 15:59:44 +01:00
|
|
|
QByteArray array(fileName.toUtf8());
|
|
|
|
success = unshield_file_save(unshield, index, array.constData());
|
2013-12-26 18:02:34 +01:00
|
|
|
|
2014-01-17 13:21:44 +01:00
|
|
|
if (!success)
|
|
|
|
{
|
2014-03-18 00:33:31 +01:00
|
|
|
qDebug() << "error";
|
2013-12-26 18:02:34 +01:00
|
|
|
dir.remove(fileName);
|
2014-01-17 13:21:44 +01:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2014-03-17 23:39:21 +01:00
|
|
|
bool Wizard::UnshieldWorker::extractCab(const QString& cabFile, const QString& destination)
|
|
|
|
{
|
|
|
|
bool success = false;
|
|
|
|
|
|
|
|
QByteArray array(cabFile.toUtf8());
|
|
|
|
|
|
|
|
Unshield* unshield;
|
|
|
|
unshield = unshield_open(array.constData());
|
|
|
|
|
|
|
|
if (!unshield)
|
|
|
|
{
|
|
|
|
emit error(tr("Failed to open InstallShield Cabinet File."), tr("Opening %1 failed.").arg(cabFile));
|
2014-03-30 23:11:38 +02:00
|
|
|
unshield_close(unshield);
|
2014-03-17 23:39:21 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int counter = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < unshield_file_group_count(unshield); ++i)
|
|
|
|
{
|
|
|
|
UnshieldFileGroup* group = unshield_file_group_get(unshield, i);
|
|
|
|
|
|
|
|
for (size_t j = group->first_file; j <= group->last_file; ++j)
|
|
|
|
{
|
2014-04-18 13:17:37 +02:00
|
|
|
if (mStopped)
|
|
|
|
{
|
|
|
|
qDebug() << "We're asked to stop!";
|
|
|
|
|
|
|
|
unshield_close(unshield);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-03-17 23:39:21 +01:00
|
|
|
if (unshield_file_is_valid(unshield, j))
|
|
|
|
{
|
|
|
|
success = extractFile(unshield, destination, group->name, j, counter);
|
2014-03-18 00:33:31 +01:00
|
|
|
|
|
|
|
if (!success)
|
|
|
|
{
|
|
|
|
QString name(QString::fromUtf8(unshield_file_name(unshield, j)));
|
|
|
|
|
|
|
|
emit error(tr("Failed to extract %1.").arg(name),
|
|
|
|
tr("Complete path: %1").arg(destination + QDir::separator() + name));
|
2014-04-18 13:17:37 +02:00
|
|
|
|
|
|
|
unshield_close(unshield);
|
2014-03-18 00:33:31 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-03-17 23:39:21 +01:00
|
|
|
++counter;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unshield_close(unshield);
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
QString Wizard::UnshieldWorker::findFile(const QString& fileName, const QString& path)
|
2014-03-16 20:07:54 +01:00
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
return findFiles(fileName, path).first();
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
QStringList Wizard::UnshieldWorker::findFiles(
|
|
|
|
const QString& fileName, const QString& path, int depth, bool recursive, bool directories, Qt::MatchFlags flags)
|
2014-03-16 20:07:54 +01:00
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
static const int MAXIMUM_DEPTH = 10;
|
2014-03-16 22:09:20 +01:00
|
|
|
|
|
|
|
if (depth >= MAXIMUM_DEPTH)
|
|
|
|
{
|
|
|
|
qWarning("Maximum directory depth limit reached.");
|
|
|
|
return QStringList();
|
|
|
|
}
|
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
QStringList result;
|
2014-03-16 22:09:20 +01:00
|
|
|
QDir dir(path);
|
2014-03-16 20:07:54 +01:00
|
|
|
|
2014-03-17 23:39:21 +01:00
|
|
|
// Prevent parsing over the complete filesystem
|
|
|
|
if (dir == QDir::rootPath())
|
|
|
|
return QStringList();
|
|
|
|
|
2014-03-16 20:07:54 +01:00
|
|
|
if (!dir.exists())
|
|
|
|
return QStringList();
|
|
|
|
|
|
|
|
QFileInfoList list(dir.entryInfoList(QDir::NoDotAndDotDot | QDir::AllDirs | QDir::Files, QDir::DirsFirst));
|
2019-10-06 13:39:27 +02:00
|
|
|
for (const QFileInfo& info : list)
|
|
|
|
{
|
2014-03-16 20:07:54 +01:00
|
|
|
if (info.isSymLink())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (info.isDir())
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
if (directories)
|
|
|
|
{
|
2019-01-15 13:24:59 +01:00
|
|
|
if (!info.fileName().compare(fileName, Qt::CaseInsensitive))
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
result.append(info.absoluteFilePath());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (recursive)
|
|
|
|
result.append(findFiles(fileName, info.absoluteFilePath(), depth + 1, recursive, true));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (recursive)
|
|
|
|
result.append(findFiles(fileName, info.absoluteFilePath(), depth + 1));
|
|
|
|
}
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-03-17 14:31:05 +01:00
|
|
|
if (directories)
|
|
|
|
break;
|
|
|
|
|
|
|
|
switch (flags)
|
|
|
|
{
|
|
|
|
case Qt::MatchExactly:
|
2019-01-15 13:24:59 +01:00
|
|
|
if (!info.fileName().compare(fileName, Qt::CaseInsensitive))
|
2014-03-17 14:31:05 +01:00
|
|
|
result.append(info.absoluteFilePath());
|
|
|
|
break;
|
|
|
|
case Qt::MatchEndsWith:
|
2019-02-17 11:44:45 +04:00
|
|
|
if (info.fileName().endsWith(fileName, Qt::CaseInsensitive))
|
2014-03-17 14:31:05 +01:00
|
|
|
result.append(info.absoluteFilePath());
|
|
|
|
break;
|
2014-03-16 20:07:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-03-17 14:31:05 +01:00
|
|
|
QStringList Wizard::UnshieldWorker::findDirectories(const QString& dirName, const QString& path, bool recursive)
|
|
|
|
{
|
|
|
|
return findFiles(dirName, path, 0, true, true);
|
|
|
|
}
|
|
|
|
|
2014-03-30 23:11:38 +02:00
|
|
|
bool Wizard::UnshieldWorker::findInCab(const QString& fileName, const QString& cabFile)
|
2014-01-24 22:25:22 +01:00
|
|
|
{
|
2014-02-24 15:59:44 +01:00
|
|
|
QByteArray array(cabFile.toUtf8());
|
|
|
|
|
2014-01-24 22:25:22 +01:00
|
|
|
Unshield* unshield;
|
2014-02-24 15:59:44 +01:00
|
|
|
unshield = unshield_open(array.constData());
|
2014-01-24 22:25:22 +01:00
|
|
|
|
|
|
|
if (!unshield)
|
|
|
|
{
|
2014-01-28 01:03:47 +01:00
|
|
|
emit error(tr("Failed to open InstallShield Cabinet File."), tr("Opening %1 failed.").arg(cabFile));
|
2014-03-17 23:39:21 +01:00
|
|
|
unshield_close(unshield);
|
2014-01-28 01:03:47 +01:00
|
|
|
return false;
|
2014-03-30 23:11:38 +02:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
|
|
|
|
for (int i = 0; i < unshield_file_group_count(unshield); ++i)
|
|
|
|
{
|
|
|
|
UnshieldFileGroup* group = unshield_file_group_get(unshield, i);
|
|
|
|
|
|
|
|
for (size_t j = group->first_file; j <= group->last_file; ++j)
|
|
|
|
{
|
2014-03-17 23:39:21 +01:00
|
|
|
|
2013-12-26 18:02:34 +01:00
|
|
|
if (unshield_file_is_valid(unshield, j))
|
|
|
|
{
|
2014-03-17 23:39:21 +01:00
|
|
|
QString current(QString::fromUtf8(unshield_file_name(unshield, j)));
|
2014-04-18 13:17:37 +02:00
|
|
|
if (current.toLower() == fileName.toLower())
|
|
|
|
{
|
|
|
|
unshield_close(unshield);
|
2014-03-17 23:39:21 +01:00
|
|
|
return true; // File is found!
|
2014-04-18 13:17:37 +02:00
|
|
|
}
|
2013-12-26 18:02:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-18 13:17:37 +02:00
|
|
|
unshield_close(unshield);
|
2014-03-17 23:39:21 +01:00
|
|
|
return false;
|
2013-12-25 18:52:34 +01:00
|
|
|
}
|
2022-04-17 16:28:14 +00:00
|
|
|
|
|
|
|
size_t Wizard::UnshieldWorker::getMorrowindBsaFileSize(const QString& cabFile)
|
|
|
|
{
|
|
|
|
QString fileName = QString("Morrowind.bsa");
|
|
|
|
QByteArray array(cabFile.toUtf8());
|
|
|
|
|
|
|
|
Unshield* unshield;
|
|
|
|
unshield = unshield_open(array.constData());
|
|
|
|
|
|
|
|
if (!unshield)
|
|
|
|
{
|
|
|
|
emit error(tr("Failed to open InstallShield Cabinet File."), tr("Opening %1 failed.").arg(cabFile));
|
|
|
|
unshield_close(unshield);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < unshield_file_group_count(unshield); ++i)
|
|
|
|
{
|
|
|
|
UnshieldFileGroup* group = unshield_file_group_get(unshield, i);
|
|
|
|
|
|
|
|
for (size_t j = group->first_file; j <= group->last_file; ++j)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (unshield_file_is_valid(unshield, j))
|
|
|
|
{
|
|
|
|
QString current(QString::fromUtf8(unshield_file_name(unshield, j)));
|
|
|
|
if (current.toLower() == fileName.toLower())
|
|
|
|
{
|
|
|
|
size_t fileSize = unshield_file_size(unshield, j);
|
|
|
|
unshield_close(unshield);
|
|
|
|
return fileSize; // File is found!
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unshield_close(unshield);
|
|
|
|
return 0;
|
|
|
|
}
|