/* * 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 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("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)) / "Data")); QDir outDirSprites(pathToQString(VCMIDirs::get().userDataPath() / "Mods" / "chronicles" / "Mods" / ("chronicles_" + std::to_string(no)) / "Sprites")); CArchiveLoader archive("", tmpDir.filePath("xBitmap.lod").toStdString(), false); for(auto & entry : archive.getEntries()) archive.extractToFolder(outDirData.absolutePath().toStdString(), "", entry.second, true); for(auto & entry : outDirData.entryList()) outDirData.rename(entry, "Hc" + QString::number(no) + "_" + entry); } 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); } }