mirror of
https://github.com/vcmi/vcmi.git
synced 2024-12-18 17:40:48 +02:00
254 lines
6.8 KiB
C++
254 lines
6.8 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"
|
|
|
|
#ifdef ENABLE_INNOEXTRACT
|
|
#include "cli/extract.hpp"
|
|
#include "setup/version.hpp"
|
|
#endif
|
|
|
|
ChroniclesExtractor::ChroniclesExtractor(QWidget *p, std::function<void(float percent)> cb) :
|
|
parent(p), cb(cb)
|
|
{
|
|
}
|
|
|
|
bool ChroniclesExtractor::handleTempDir(bool create)
|
|
{
|
|
if(create)
|
|
{
|
|
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
|
|
}
|
|
else
|
|
tempDir.removeRecursively();
|
|
|
|
return true;
|
|
}
|
|
|
|
int ChroniclesExtractor::getChronicleNo(QFile & file)
|
|
{
|
|
if(!file.open(QIODevice::ReadOnly))
|
|
{
|
|
QMessageBox::critical(parent, tr("File cannot opened"), file.errorString());
|
|
return 0;
|
|
}
|
|
|
|
QByteArray magic{"MZ"};
|
|
QByteArray magicFile = file.read(magic.length());
|
|
if(!magicFile.startsWith(magic))
|
|
{
|
|
QMessageBox::critical(parent, tr("Invalid file selected"), tr("You have to select an gog installer file!"));
|
|
return 0;
|
|
}
|
|
|
|
QByteArray dataBegin = file.read(1'000'000);
|
|
int chronicle = 0;
|
|
for (const auto& kv : chronicles) {
|
|
if(dataBegin.contains(kv.second))
|
|
{
|
|
chronicle = kv.first;
|
|
break;
|
|
}
|
|
}
|
|
if(!chronicle)
|
|
{
|
|
QMessageBox::critical(parent, tr("Invalid file selected"), tr("You have to select an chronicle installer file!"));
|
|
return 0;
|
|
}
|
|
return chronicle;
|
|
}
|
|
|
|
bool ChroniclesExtractor::extractGogInstaller(QString file)
|
|
{
|
|
#ifndef ENABLE_INNOEXTRACT
|
|
QMessageBox::critical(parent, tr("Innoextract functionality missing"), "VCMI was compiled without innoextract support, which is needed to extract chroncles!");
|
|
return false;
|
|
#else
|
|
::extract_options o;
|
|
o.extract = true;
|
|
|
|
// standard settings
|
|
o.gog_galaxy = true;
|
|
o.codepage = 0U;
|
|
o.output_dir = tempDir.path().toStdString();
|
|
o.extract_temp = true;
|
|
o.extract_unknown = true;
|
|
o.filenames.set_expand(true);
|
|
|
|
o.preserve_file_times = true; // also correctly closes file -> without it: on Windows the files are not written completely
|
|
|
|
QString errorText = "";
|
|
try
|
|
{
|
|
process_file(file.toStdString(), o, [this](float progress) {
|
|
float overallProgress = ((1.0 / float(fileCount)) * float(extractionFile)) + (progress / float(fileCount));
|
|
if(cb)
|
|
cb(overallProgress);
|
|
});
|
|
}
|
|
catch(const std::ios_base::failure & e)
|
|
{
|
|
errorText = tr("Stream error while extracting files!\nerror reason: ");
|
|
errorText += e.what();
|
|
}
|
|
catch(const format_error & e)
|
|
{
|
|
errorText = e.what();
|
|
}
|
|
catch(const std::runtime_error & e)
|
|
{
|
|
errorText = e.what();
|
|
}
|
|
catch(const setup::version_error &)
|
|
{
|
|
errorText = tr("Not a supported Inno Setup installer!");
|
|
}
|
|
|
|
if(!errorText.isEmpty())
|
|
{
|
|
QMessageBox::critical(parent, tr("Extracting error!"), errorText);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
void ChroniclesExtractor::createBaseMod()
|
|
{
|
|
QDir dir(pathToQString(VCMIDirs::get().userDataPath() / "Mods"));
|
|
dir.mkdir("chronicles");
|
|
dir.cd("chronicles");
|
|
dir.mkdir("Mods");
|
|
|
|
QJsonObject mod
|
|
{
|
|
{ "modType", "Extension" },
|
|
{ "name", "Heroes Chronicles" },
|
|
{ "description", "" },
|
|
{ "author", "3DO" },
|
|
{ "version", "1.0" },
|
|
};
|
|
|
|
QFile jsonFile(dir.filePath("mod.json"));
|
|
jsonFile.open(QFile::WriteOnly);
|
|
jsonFile.write(QJsonDocument(mod).toJson());
|
|
}
|
|
|
|
void ChroniclesExtractor::createChronicleMod(int no)
|
|
{
|
|
QDir dir(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods"));
|
|
dir.mkdir("chronicles_" + QString::number(no));
|
|
dir.cd("chronicles_" + QString::number(no));
|
|
|
|
QJsonObject mod
|
|
{
|
|
{ "modType", "Extension" },
|
|
{ "name", "Heroes Chronicles - " + QString::number(no) },
|
|
{ "description", "" },
|
|
{ "author", "3DO" },
|
|
{ "version", "1.0" },
|
|
};
|
|
|
|
QFile jsonFile(dir.filePath("mod.json"));
|
|
jsonFile.open(QFile::WriteOnly);
|
|
jsonFile.write(QJsonDocument(mod).toJson());
|
|
|
|
dir.mkdir("content");
|
|
dir.cd("content");
|
|
|
|
dir.mkdir("Data");
|
|
dir.mkdir("Sprites");
|
|
extractFiles(no);
|
|
}
|
|
|
|
void ChroniclesExtractor::extractFiles(int no)
|
|
{
|
|
QByteArray tmpChronicles = chronicles.at(no);
|
|
tmpChronicles.replace('\0', "");
|
|
|
|
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()));
|
|
QDir outDirData(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / ("chronicles_" + std::to_string(no)) / "content" / "Data"));
|
|
QDir outDirSprites(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / ("chronicles_" + std::to_string(no)) / "content" / "Sprites"));
|
|
QDir outDirVideo(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / ("chronicles_" + std::to_string(no)) / "content" / "Video"));
|
|
QDir outDirSounds(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / ("chronicles_" + std::to_string(no)) / "content" / "Sounds"));
|
|
|
|
auto extract = [tmpDir, no](QDir dest, QString file){
|
|
CArchiveLoader archive("", tmpDir.filePath(tmpDir.entryList({file}).front()).toStdString(), false);
|
|
for(auto & entry : archive.getEntries())
|
|
archive.extractToFolder(dest.absolutePath().toStdString(), "", entry.second, true);
|
|
};
|
|
auto rename = [tmpDir, no](QDir dest){
|
|
dest.refresh();
|
|
for(auto & entry : dest.entryList())
|
|
if(!entry.startsWith("Hc"))
|
|
dest.rename(entry, "Hc" + QString::number(no) + "_" + entry);
|
|
};
|
|
|
|
extract(outDirData, "xBitmap.lod");
|
|
extract(outDirData, "xlBitmap.lod");
|
|
extract(outDirSprites, "xSprite.lod");
|
|
extract(outDirSprites, "xlSprite.lod");
|
|
extract(outDirVideo, "xVideo.vid");
|
|
extract(outDirSounds, "xSound.snd");
|
|
|
|
tmpDir.cdUp();
|
|
if(tmpDir.entryList({"maps"}, QDir::Filter::Dirs).size())
|
|
{
|
|
QDir tmpDirMaps = tmpDir.filePath(tmpDir.entryList({"maps"}, QDir::Filter::Dirs).front());
|
|
for(auto & entry : tmpDirMaps.entryList())
|
|
QFile(tmpDirMaps.filePath(entry)).copy(outDirData.filePath(entry));
|
|
}
|
|
|
|
rename(outDirData);
|
|
rename(outDirSprites);
|
|
rename(outDirVideo);
|
|
rename(outDirSounds);
|
|
}
|
|
|
|
void ChroniclesExtractor::installChronicles(QStringList exe)
|
|
{
|
|
extractionFile = -1;
|
|
fileCount = exe.size();
|
|
for(QString f : exe)
|
|
{
|
|
extractionFile++;
|
|
QFile file(f);
|
|
|
|
int chronicleNo = getChronicleNo(file);
|
|
if(!chronicleNo)
|
|
continue;
|
|
|
|
if(!handleTempDir(true))
|
|
continue;
|
|
|
|
if(!extractGogInstaller(f))
|
|
continue;
|
|
|
|
createBaseMod();
|
|
createChronicleMod(chronicleNo);
|
|
|
|
handleTempDir(false);
|
|
}
|
|
} |