mirror of
https://github.com/HarbourMasters/Shipwright.git
synced 2025-02-09 11:50:33 -05:00
![briaguya](/assets/img/avatar_default.png)
Co-authored-by: kenix3 <kenixwhisperwind@gmail.com> Co-authored-by: David Chavez <david@dcvz.io> Co-authored-by: KiritoDv <kiritodev01@gmail.com> Co-authored-by: Christopher Leggett <chris@leggett.dev>
292 lines
9.9 KiB
C++
292 lines
9.9 KiB
C++
#include "Main.h"
|
|
#include "BackgroundExporter.h"
|
|
#include "TextureExporter.h"
|
|
#include "RoomExporter.h"
|
|
#include "CollisionExporter.h"
|
|
#include "DisplayListExporter.h"
|
|
#include "PlayerAnimationExporter.h"
|
|
#include "SkeletonExporter.h"
|
|
#include "SkeletonLimbExporter.h"
|
|
#include "ArrayExporter.h"
|
|
#include "VtxExporter.h"
|
|
#include "AnimationExporter.h"
|
|
#include "CutsceneExporter.h"
|
|
#include "PathExporter.h"
|
|
#include "TextExporter.h"
|
|
#include "BlobExporter.h"
|
|
#include "MtxExporter.h"
|
|
#include "AudioExporter.h"
|
|
#include <Globals.h>
|
|
#include <Utils/File.h>
|
|
#include <Utils/Directory.h>
|
|
#include <Utils/MemoryStream.h>
|
|
#include <Utils/BinaryWriter.h>
|
|
#include <bit>
|
|
|
|
std::string otrFileName = "oot.otr";
|
|
std::shared_ptr<Ship::Archive> otrArchive;
|
|
BinaryWriter* fileWriter;
|
|
std::chrono::steady_clock::time_point fileStart, resStart;
|
|
std::map<std::string, std::vector<char>> files;
|
|
std::mutex fileMutex;
|
|
|
|
void InitVersionInfo();
|
|
|
|
enum class ExporterFileMode
|
|
{
|
|
BuildOTR = (int)ZFileMode::Custom + 1,
|
|
};
|
|
|
|
static void ExporterParseFileMode(const std::string& buildMode, ZFileMode& fileMode)
|
|
{
|
|
if (buildMode == "botr")
|
|
{
|
|
fileMode = (ZFileMode)ExporterFileMode::BuildOTR;
|
|
|
|
printf("BOTR: Generating OTR Archive...\n");
|
|
|
|
if (File::Exists(otrFileName))
|
|
otrArchive = std::shared_ptr<Ship::Archive>(new Ship::Archive(otrFileName, true));
|
|
else
|
|
otrArchive = Ship::Archive::CreateArchive(otrFileName, 40000);
|
|
|
|
auto lst = Directory::ListFiles("Extract");
|
|
|
|
for (auto item : lst)
|
|
{
|
|
auto fileData = File::ReadAllBytes(item);
|
|
otrArchive->AddFile(StringHelper::Split(item, "Extract/")[1], (uintptr_t)fileData.data(), fileData.size());
|
|
}
|
|
}
|
|
}
|
|
|
|
static void ExporterProgramEnd()
|
|
{
|
|
if (Globals::Instance->fileMode == ZFileMode::ExtractDirectory)
|
|
{
|
|
printf("Creating version file...\n");
|
|
|
|
// Get crc from rom
|
|
std::string romPath = Globals::Instance->baseRomPath.string();
|
|
std::vector<uint8_t> romData = File::ReadAllBytes(romPath);
|
|
uint32_t crc = BitConverter::ToUInt32BE(romData, 0x10);
|
|
uint8_t endianness = (uint8_t)Endianness::Big;
|
|
|
|
// Write crc to version file
|
|
fs::path versionPath("Extract/version");
|
|
MemoryStream* versionStream = new MemoryStream();
|
|
BinaryWriter writer(versionStream);
|
|
writer.SetEndianness(Endianness::Big);
|
|
writer.Write(endianness);
|
|
writer.Write(crc);
|
|
std::ofstream versionFile(versionPath.c_str(), std::ios::out | std::ios::binary);
|
|
versionFile.write(versionStream->ToVector().data(), versionStream->GetLength());
|
|
versionFile.flush();
|
|
versionFile.close();
|
|
writer.Close();
|
|
|
|
printf("Created version file.\n");
|
|
|
|
printf("Generating OTR Archive...\n");
|
|
otrArchive = Ship::Archive::CreateArchive(otrFileName, 40000);
|
|
|
|
for (auto item : files) {
|
|
std::string fName = item.first;
|
|
if (fName.find("gTitleZeldaShieldLogoMQTex") != std::string::npos && !ZRom(romPath).IsMQ())
|
|
{
|
|
size_t pos = 0;
|
|
if ((pos = fName.find("gTitleZeldaShieldLogoMQTex", 0)) != std::string::npos)
|
|
{
|
|
fName.replace(pos, 27, "gTitleZeldaShieldLogoTex");
|
|
}
|
|
}
|
|
auto fileData = item.second;
|
|
otrArchive->AddFile(fName, (uintptr_t)fileData.data(),
|
|
fileData.size());
|
|
}
|
|
|
|
// Add any additional files that need to be manually copied...
|
|
auto lst = Directory::ListFiles("Extract");
|
|
|
|
for (auto item : lst)
|
|
{
|
|
std::vector<std::string> splitPath = StringHelper::Split(item, ".");
|
|
|
|
if (splitPath.size() >= 3) {
|
|
std::string extension = splitPath.at(splitPath.size() - 1);
|
|
std::string format = splitPath.at(splitPath.size() - 2);
|
|
splitPath.pop_back();
|
|
splitPath.pop_back();
|
|
std::string afterPath = std::accumulate(splitPath.begin(), splitPath.end(), std::string(""));
|
|
if (extension == "png" && (format == "rgba32" || format == "rgb5a1" || format == "i4" || format == "i8" || format == "ia4" || format == "ia8" || format == "ia16" || format == "ci4" || format == "ci8")) {
|
|
Globals::Instance->buildRawTexture = true;
|
|
Globals::Instance->BuildAssetTexture(item, ZTexture::GetTextureTypeFromString(format), afterPath);
|
|
Globals::Instance->buildRawTexture = false;
|
|
|
|
auto fileData = File::ReadAllBytes(afterPath);
|
|
printf("otrArchive->AddFile(%s)\n", StringHelper::Split(afterPath, "Extract/")[1].c_str());
|
|
otrArchive->AddFile(StringHelper::Split(afterPath, "Extract/")[1], (uintptr_t)fileData.data(), fileData.size());
|
|
}
|
|
}
|
|
|
|
auto fileData = File::ReadAllBytes(item);
|
|
printf("otrArchive->AddFile(%s)\n", StringHelper::Split(item, "Extract/")[1].c_str());
|
|
otrArchive->AddFile(StringHelper::Split(item, "Extract/")[1], (uintptr_t)fileData.data(), fileData.size());
|
|
}
|
|
|
|
//otrArchive->AddFile("Audiobank", (uintptr_t)Globals::Instance->GetBaseromFile("Audiobank").data(), Globals::Instance->GetBaseromFile("Audiobank").size());
|
|
//otrArchive->AddFile("Audioseq", (uintptr_t)Globals::Instance->GetBaseromFile("Audioseq").data(), Globals::Instance->GetBaseromFile("Audioseq").size());
|
|
//otrArchive->AddFile("Audiotable", (uintptr_t)Globals::Instance->GetBaseromFile("Audiotable").data(), Globals::Instance->GetBaseromFile("Audiotable").size());
|
|
}
|
|
}
|
|
|
|
|
|
static void ExporterParseArgs(int argc, char* argv[], int& i)
|
|
{
|
|
std::string arg = argv[i];
|
|
|
|
if (arg == "--otrfile")
|
|
{
|
|
otrFileName = argv[i + 1];
|
|
i++;
|
|
}
|
|
}
|
|
|
|
static bool ExporterProcessFileMode(ZFileMode fileMode)
|
|
{
|
|
// Do whatever work is associated with these custom file modes...
|
|
// Return true to indicate one of our own file modes is being processed
|
|
if (fileMode == (ZFileMode)ExporterFileMode::BuildOTR)
|
|
return true;
|
|
|
|
return false;
|
|
}
|
|
|
|
static void ExporterFileBegin(ZFile* file)
|
|
{
|
|
fileStart = std::chrono::steady_clock::now();
|
|
|
|
MemoryStream* stream = new MemoryStream();
|
|
fileWriter = new BinaryWriter(stream);
|
|
}
|
|
|
|
static void ExporterFileEnd(ZFile* file)
|
|
{
|
|
// delete fileWriter;
|
|
}
|
|
|
|
static void ExporterResourceEnd(ZResource* res, BinaryWriter& writer)
|
|
{
|
|
auto streamShared = writer.GetStream();
|
|
MemoryStream* strem = (MemoryStream*)streamShared.get();
|
|
|
|
auto start = std::chrono::steady_clock::now();
|
|
|
|
if (res->GetName() != "")
|
|
{
|
|
std::string oName = res->parent->GetOutName();
|
|
std::string rName = res->GetName();
|
|
std::string prefix = OTRExporter_DisplayList::GetPrefix(res);
|
|
|
|
//auto xmlFilePath = res->parent->GetXmlFilePath();
|
|
//prefix = StringHelper::Split(StringHelper::Split(xmlFilePath.string(), "xml\\")[1], ".xml")[0];
|
|
|
|
if (StringHelper::Contains(oName, "_scene"))
|
|
{
|
|
auto split = StringHelper::Split(oName, "_");
|
|
oName = "";
|
|
for (size_t i = 0; i < split.size() - 1; i++)
|
|
oName += split[i] + "_";
|
|
|
|
oName += "scene";
|
|
}
|
|
else if (StringHelper::Contains(oName, "_room"))
|
|
{
|
|
oName = StringHelper::Split(oName, "_room")[0] + "_scene";
|
|
}
|
|
|
|
std::string fName = "";
|
|
|
|
if (prefix != "")
|
|
fName = StringHelper::Sprintf("%s/%s/%s", prefix.c_str(), oName.c_str(), rName.c_str());
|
|
else
|
|
fName = StringHelper::Sprintf("%s/%s", oName.c_str(), rName.c_str());
|
|
|
|
if (Globals::Instance->fileMode == ZFileMode::ExtractDirectory)
|
|
{
|
|
std::unique_lock Lock(fileMutex);
|
|
files[fName] = strem->ToVector();
|
|
}
|
|
else
|
|
File::WriteAllBytes("Extract/" + fName, strem->ToVector());
|
|
}
|
|
|
|
auto end = std::chrono::steady_clock::now();
|
|
size_t diff = std::chrono::duration_cast<std::chrono::milliseconds>(end - start).count();
|
|
|
|
//if (diff > 10)
|
|
//printf("Exported Resource End %s in %zums\n", res->GetName().c_str(), diff);
|
|
}
|
|
|
|
static void ExporterXMLBegin()
|
|
{
|
|
}
|
|
|
|
static void ExporterXMLEnd()
|
|
{
|
|
}
|
|
|
|
void AddFile(std::string fName, std::vector<char> data)
|
|
{
|
|
if (Globals::Instance->fileMode != ZFileMode::ExtractDirectory)
|
|
File::WriteAllBytes("Extract/" + fName, data);
|
|
else
|
|
{
|
|
std::unique_lock Lock(fileMutex);
|
|
files[fName] = data;
|
|
}
|
|
}
|
|
|
|
static void ImportExporters()
|
|
{
|
|
// In this example we set up a new exporter called "EXAMPLE".
|
|
// By running ZAPD with the argument -se EXAMPLE, we tell it that we want to use this exporter for our resources.
|
|
ExporterSet* exporterSet = new ExporterSet();
|
|
exporterSet->processFileModeFunc = ExporterProcessFileMode;
|
|
exporterSet->parseFileModeFunc = ExporterParseFileMode;
|
|
exporterSet->parseArgsFunc = ExporterParseArgs;
|
|
exporterSet->beginFileFunc = ExporterFileBegin;
|
|
exporterSet->endFileFunc = ExporterFileEnd;
|
|
exporterSet->beginXMLFunc = ExporterXMLBegin;
|
|
exporterSet->endXMLFunc = ExporterXMLEnd;
|
|
exporterSet->resSaveFunc = ExporterResourceEnd;
|
|
exporterSet->endProgramFunc = ExporterProgramEnd;
|
|
|
|
exporterSet->exporters[ZResourceType::Background] = new OTRExporter_Background();
|
|
exporterSet->exporters[ZResourceType::Texture] = new OTRExporter_Texture();
|
|
exporterSet->exporters[ZResourceType::Room] = new OTRExporter_Room();
|
|
exporterSet->exporters[ZResourceType::AltHeader] = new OTRExporter_Room();
|
|
exporterSet->exporters[ZResourceType::Scene] = new OTRExporter_Room();
|
|
exporterSet->exporters[ZResourceType::CollisionHeader] = new OTRExporter_Collision();
|
|
exporterSet->exporters[ZResourceType::DisplayList] = new OTRExporter_DisplayList();
|
|
exporterSet->exporters[ZResourceType::PlayerAnimationData] = new OTRExporter_PlayerAnimationExporter();
|
|
exporterSet->exporters[ZResourceType::Skeleton] = new OTRExporter_Skeleton();
|
|
exporterSet->exporters[ZResourceType::Limb] = new OTRExporter_SkeletonLimb();
|
|
exporterSet->exporters[ZResourceType::Animation] = new OTRExporter_Animation();
|
|
exporterSet->exporters[ZResourceType::Cutscene] = new OTRExporter_Cutscene();
|
|
exporterSet->exporters[ZResourceType::Vertex] = new OTRExporter_Vtx();
|
|
exporterSet->exporters[ZResourceType::Array] = new OTRExporter_Array();
|
|
exporterSet->exporters[ZResourceType::Path] = new OTRExporter_Path();
|
|
exporterSet->exporters[ZResourceType::Text] = new OTRExporter_Text();
|
|
exporterSet->exporters[ZResourceType::Blob] = new OTRExporter_Blob();
|
|
exporterSet->exporters[ZResourceType::Mtx] = new OTRExporter_MtxExporter();
|
|
exporterSet->exporters[ZResourceType::Audio] = new OTRExporter_Audio();
|
|
|
|
Globals::AddExporter("OTR", exporterSet);
|
|
|
|
InitVersionInfo();
|
|
}
|
|
|
|
// When ZAPD starts up, it will automatically call the below function, which in turn sets up our exporters.
|
|
REGISTER_EXPORTER(ImportExporters);
|