mirror of
https://github.com/vcmi/vcmi.git
synced 2025-01-12 02:28:11 +02:00
260 lines
8.5 KiB
C++
260 lines
8.5 KiB
C++
/*
|
|
* chroniclesextractor.cpp, part of VCMI engine
|
|
*
|
|
* Authors: listed in file AUTHORS in main folder
|
|
*
|
|
* License: GNU General Public License v2.0 or later
|
|
* Full text of license available in license.txt file, in main folder
|
|
*
|
|
*/
|
|
#include "StdInc.h"
|
|
|
|
#include "chroniclesextractor.h"
|
|
|
|
#include "../../lib/VCMIDirs.h"
|
|
#include "../../lib/filesystem/CArchiveLoader.h"
|
|
|
|
#include "../innoextract.h"
|
|
|
|
ChroniclesExtractor::ChroniclesExtractor(QWidget *p, std::function<void(float percent)> cb) :
|
|
parent(p), cb(cb)
|
|
{
|
|
}
|
|
|
|
bool ChroniclesExtractor::createTempDir()
|
|
{
|
|
tempDir = QDir(pathToQString(VCMIDirs::get().userDataPath()));
|
|
if(tempDir.cd("tmp"))
|
|
{
|
|
tempDir.removeRecursively(); // remove if already exists (e.g. previous run)
|
|
tempDir.cdUp();
|
|
}
|
|
tempDir.mkdir("tmp");
|
|
if(!tempDir.cd("tmp"))
|
|
return false; // should not happen - but avoid deleting wrong folder in any case
|
|
|
|
return true;
|
|
}
|
|
|
|
void ChroniclesExtractor::removeTempDir()
|
|
{
|
|
tempDir.removeRecursively();
|
|
}
|
|
|
|
int ChroniclesExtractor::getChronicleNo()
|
|
{
|
|
QStringList appDirCandidates = tempDir.entryList({"app"}, QDir::Filter::Dirs);
|
|
|
|
if (!appDirCandidates.empty())
|
|
{
|
|
QDir appDir = tempDir.filePath(appDirCandidates.front());
|
|
|
|
for (size_t i = 1; i < chronicles.size(); ++i)
|
|
{
|
|
QString chronicleName = chronicles.at(i);
|
|
QStringList chroniclesDirCandidates = appDir.entryList({chronicleName}, QDir::Filter::Dirs);
|
|
|
|
if (!chroniclesDirCandidates.empty())
|
|
return i;
|
|
}
|
|
}
|
|
QMessageBox::critical(parent, tr("Invalid file selected"), tr("You have to select a Heroes Chronicles installer file!"));
|
|
return 0;
|
|
}
|
|
|
|
bool ChroniclesExtractor::extractGogInstaller(QString file)
|
|
{
|
|
QString errorText = Innoextract::extract(file, tempDir.path(), [this](float progress) {
|
|
float overallProgress = ((1.0 / static_cast<float>(fileCount)) * static_cast<float>(extractionFile)) + (progress / static_cast<float>(fileCount));
|
|
if(cb)
|
|
cb(overallProgress);
|
|
});
|
|
|
|
if(!errorText.isEmpty())
|
|
{
|
|
QString hashError = Innoextract::getHashError(file, {}, {}, {});
|
|
QMessageBox::critical(parent, tr("Extracting error!"), errorText);
|
|
if(!hashError.isEmpty())
|
|
QMessageBox::critical(parent, tr("Hash error!"), hashError, QMessageBox::Ok, QMessageBox::Ok);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ChroniclesExtractor::createBaseMod() const
|
|
{
|
|
QDir dir(pathToQString(VCMIDirs::get().userDataPath() / "Mods"));
|
|
dir.mkdir("chronicles");
|
|
dir.cd("chronicles");
|
|
dir.mkdir("Mods");
|
|
|
|
QJsonObject mod
|
|
{
|
|
{ "modType", "Expansion" },
|
|
{ "name", tr("Heroes Chronicles") },
|
|
{ "description", tr("Heroes Chronicles") },
|
|
{ "author", "3DO" },
|
|
{ "version", "1.0" },
|
|
{ "contact", "vcmi.eu" },
|
|
{ "heroes", QJsonArray({"config/portraitsChronicles.json"}) },
|
|
{ "settings", QJsonObject({{"mapFormat", QJsonObject({{"chronicles", QJsonObject({{
|
|
{"supported", true},
|
|
{"portraits", QJsonObject({
|
|
{"portraitTarnumBarbarian", 163},
|
|
{"portraitTarnumKnight", 164},
|
|
{"portraitTarnumWizard", 165},
|
|
{"portraitTarnumRanger", 166},
|
|
{"portraitTarnumOverlord", 167},
|
|
{"portraitTarnumBeastmaster", 168},
|
|
})},
|
|
}})}})}})},
|
|
};
|
|
|
|
QFile jsonFile(dir.filePath("mod.json"));
|
|
jsonFile.open(QFile::WriteOnly);
|
|
jsonFile.write(QJsonDocument(mod).toJson());
|
|
|
|
for(auto & dataPath : VCMIDirs::get().dataPaths())
|
|
{
|
|
auto file = pathToQString(dataPath / "config" / "heroes" / "portraitsChronicles.json");
|
|
auto destFolder = VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "content" / "config";
|
|
auto destFile = pathToQString(destFolder / "portraitsChronicles.json");
|
|
if(QFile::exists(file))
|
|
{
|
|
QDir().mkpath(pathToQString(destFolder));
|
|
QFile::remove(destFile);
|
|
QFile::copy(file, destFile);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ChroniclesExtractor::createChronicleMod(int no)
|
|
{
|
|
QDir dir(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / ("chronicles_" + std::to_string(no))));
|
|
dir.removeRecursively();
|
|
dir.mkpath(".");
|
|
|
|
QString tmpChronicles = chronicles.at(no);
|
|
|
|
QJsonObject mod
|
|
{
|
|
{ "modType", "Expansion" },
|
|
{ "name", QString("%1 - %2").arg(no).arg(tmpChronicles) },
|
|
{ "description", tr("Heroes Chronicles %1 - %2").arg(no).arg(tmpChronicles) },
|
|
{ "author", "3DO" },
|
|
{ "version", "1.0" },
|
|
{ "contact", "vcmi.eu" },
|
|
};
|
|
|
|
QFile jsonFile(dir.filePath("mod.json"));
|
|
jsonFile.open(QFile::WriteOnly);
|
|
jsonFile.write(QJsonDocument(mod).toJson());
|
|
|
|
dir.cd("content");
|
|
|
|
extractFiles(no);
|
|
}
|
|
|
|
void ChroniclesExtractor::extractFiles(int no) const
|
|
{
|
|
QString tmpChronicles = chronicles.at(no);
|
|
|
|
std::string chroniclesDir = "chronicles_" + std::to_string(no);
|
|
QDir tmpDir = tempDir.filePath(tempDir.entryList({"app"}, QDir::Filter::Dirs).front());
|
|
tmpDir.setPath(tmpDir.filePath(tmpDir.entryList({QString(tmpChronicles)}, QDir::Filter::Dirs).front()));
|
|
tmpDir.setPath(tmpDir.filePath(tmpDir.entryList({"data"}, QDir::Filter::Dirs).front()));
|
|
auto basePath = VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / chroniclesDir / "content";
|
|
QDir outDirDataPortraits(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "content" / "Data"));
|
|
QDir outDirData(pathToQString(basePath / "Data" / chroniclesDir));
|
|
QDir outDirSprites(pathToQString(basePath / "Sprites" / chroniclesDir));
|
|
QDir outDirVideo(pathToQString(basePath / "Video" / chroniclesDir));
|
|
QDir outDirSounds(pathToQString(basePath / "Sounds" / chroniclesDir));
|
|
QDir outDirMaps(pathToQString(basePath / "Maps" / "Chronicles"));
|
|
|
|
auto extract = [](QDir scrDir, QDir dest, QString file, std::vector<std::string> files = {}){
|
|
CArchiveLoader archive("", scrDir.filePath(scrDir.entryList({file}).front()).toStdString(), false);
|
|
for(auto & entry : archive.getEntries())
|
|
if(files.empty())
|
|
archive.extractToFolder(dest.absolutePath().toStdString(), "", entry.second, true);
|
|
else
|
|
{
|
|
for(const auto & item : files)
|
|
if(boost::algorithm::to_lower_copy(entry.second.name).find(boost::algorithm::to_lower_copy(item)) != std::string::npos)
|
|
archive.extractToFolder(dest.absolutePath().toStdString(), "", entry.second, true);
|
|
}
|
|
};
|
|
|
|
extract(tmpDir, outDirData, "xBitmap.lod");
|
|
extract(tmpDir, outDirData, "xlBitmap.lod");
|
|
extract(tmpDir, outDirSprites, "xSprite.lod");
|
|
extract(tmpDir, outDirSprites, "xlSprite.lod");
|
|
extract(tmpDir, outDirVideo, "xVideo.vid");
|
|
extract(tmpDir, outDirSounds, "xSound.snd");
|
|
|
|
tmpDir.cdUp();
|
|
if(tmpDir.entryList({"maps"}, QDir::Filter::Dirs).size()) // special case for "The World Tree": the map is in the "Maps" folder instead of inside the lod
|
|
{
|
|
QDir tmpDirMaps = tmpDir.filePath(tmpDir.entryList({"maps"}, QDir::Filter::Dirs).front());
|
|
for(const auto & entry : tmpDirMaps.entryList())
|
|
QFile(tmpDirMaps.filePath(entry)).copy(outDirData.filePath(entry));
|
|
}
|
|
|
|
tmpDir.cdUp();
|
|
QDir tmpDirData = tmpDir.filePath(tmpDir.entryList({"data"}, QDir::Filter::Dirs).front());
|
|
auto tarnumPortraits = std::vector<std::string>{"HPS137", "HPS138", "HPS139", "HPS140", "HPS141", "HPS142", "HPL137", "HPL138", "HPL139", "HPL140", "HPL141", "HPL142"};
|
|
extract(tmpDirData, outDirDataPortraits, "bitmap.lod", tarnumPortraits);
|
|
extract(tmpDirData, outDirData, "lbitmap.lod", std::vector<std::string>{"INTRORIM"});
|
|
|
|
if(!outDirMaps.exists())
|
|
outDirMaps.mkpath(".");
|
|
QString campaignFileName = "Hc" + QString::number(no) + "_Main.h3c";
|
|
QFile(outDirData.filePath(outDirData.entryList({"Main.h3c"}).front())).copy(outDirMaps.filePath(campaignFileName));
|
|
}
|
|
|
|
void ChroniclesExtractor::installChronicles(QStringList exe)
|
|
{
|
|
logGlobal->info("Installing Chronicles");
|
|
|
|
extractionFile = -1;
|
|
fileCount = exe.size();
|
|
for(QString f : exe)
|
|
{
|
|
extractionFile++;
|
|
|
|
logGlobal->info("Creating temporary directory");
|
|
if(!createTempDir())
|
|
continue;
|
|
|
|
logGlobal->info("Copying offline installer");
|
|
// FIXME: this is required at the moment for Android (and possibly iOS)
|
|
// Incoming file names are in content URI form, e.g. content://media/internal/chronicles.exe
|
|
// Qt can handle those like it does regular files
|
|
// however, innoextract fails to open such files
|
|
// so make a copy in directory to which vcmi always has full access and operate on it
|
|
QString filepath = tempDir.filePath("chr.exe");
|
|
QFile(f).copy(filepath);
|
|
QFile file(filepath);
|
|
|
|
logGlobal->info("Extracting offline installer");
|
|
if(!extractGogInstaller(filepath))
|
|
continue;
|
|
|
|
logGlobal->info("Detecting Chronicle");
|
|
int chronicleNo = getChronicleNo();
|
|
if(!chronicleNo)
|
|
continue;
|
|
|
|
logGlobal->info("Creating base Chronicle mod");
|
|
createBaseMod();
|
|
|
|
logGlobal->info("Creating Chronicle mod");
|
|
createChronicleMod(chronicleNo);
|
|
|
|
logGlobal->info("Removing temporary directory");
|
|
removeTempDir();
|
|
}
|
|
|
|
logGlobal->info("Chronicles installed");
|
|
}
|