#include "DataSpec/DNACommon/FONT.hpp" #include "DataSpec/DNACommon/PAK.hpp" #include <logvisor/logvisor.hpp> namespace DataSpec::DNAFont { logvisor::Module LogModule("urde::DNAFont"); template <class IDType> void FONT<IDType>::_read(athena::io::IStreamReader& __dna_reader) { /* magic */ DNAFourCC magic; magic.read(__dna_reader); if (magic != SBIG('FONT')) { LogModule.report(logvisor::Fatal, FMT_STRING("Invalid FONT magic '{}'"), magic); return; } /* version */ version = __dna_reader.readUint32Big(); /* unknown1 */ unknown1 = __dna_reader.readUint32Big(); /* lineHeight */ lineHeight = __dna_reader.readInt32Big(); /* verticalOffset */ verticalOffset = __dna_reader.readInt32Big(); /* lineMargin */ lineMargin = __dna_reader.readInt32Big(); /* unknown2 */ unknown2 = __dna_reader.readBool(); /* unknown3 */ unknown3 = __dna_reader.readBool(); /* unknown4 */ unknown4 = __dna_reader.readUint32Big(); /* fontSize */ fontSize = __dna_reader.readUint32Big(); /* name */ name = __dna_reader.readString(-1); /* textureId */ textureId.read(__dna_reader); /* textureFormat */ textureFormat = __dna_reader.readUint32Big(); /* glyphCount */ glyphCount = __dna_reader.readUint32Big(); /* glyphs */ for (atUint32 i = 0; i < glyphCount; i++) { if (version < 4) glyphs.emplace_back(new GlyphMP1); else glyphs.emplace_back(new GlyphMP2); glyphs.back()->read(__dna_reader); } /* kerningInfoCount */ kerningInfoCount = __dna_reader.readUint32Big(); /* kerningInfo */ __dna_reader.enumerate(kerningInfo, kerningInfoCount); } template <class IDType> void FONT<IDType>::_write(athena::io::IStreamWriter& __dna_writer) const { /* magic */ __dna_writer.writeBytes((atInt8*)"FONT", 4); /* version */ __dna_writer.writeUint32Big(version); /* unknown1 */ __dna_writer.writeUint32Big(unknown1); /* lineHeight */ __dna_writer.writeInt32Big(lineHeight); /* verticalOffset */ __dna_writer.writeInt32Big(verticalOffset); /* lineMargin */ __dna_writer.writeInt32Big(lineMargin); /* unknown2 */ __dna_writer.writeBool(unknown2); /* unknown3 */ __dna_writer.writeBool(unknown3); /* unknown4 */ __dna_writer.writeUint32Big(unknown4); /* fontSize */ __dna_writer.writeUint32Big(fontSize); /* name */ __dna_writer.writeString(name, -1); /* textureId */ textureId.write(__dna_writer); /* textureFormat */ __dna_writer.writeUint32Big(textureFormat); /* glyphCount */ __dna_writer.writeUint32Big(glyphCount); /* glyphs */ for (const std::unique_ptr<IGlyph>& glyph : glyphs) glyph->write(__dna_writer); /* kerningInfoCount */ __dna_writer.writeUint32Big(kerningInfoCount); /* kerningInfo */ __dna_writer.enumerate(kerningInfo); } template <class IDType> void FONT<IDType>::_read(athena::io::YAMLDocReader& __dna_docin) { /* version */ version = __dna_docin.readUint32("version"); /* unknown1 */ unknown1 = __dna_docin.readUint32("unknown1"); /* lineHeight */ lineHeight = __dna_docin.readInt32("lineHeight"); /* verticalOffset */ verticalOffset = __dna_docin.readInt32("verticalOffset"); /* lineMargin */ lineMargin = __dna_docin.readInt32("lineMargin"); /* unknown2 */ unknown2 = __dna_docin.readBool("unknown2"); /* unknown3 */ unknown3 = __dna_docin.readBool("unknown3"); /* unknown4 */ unknown4 = __dna_docin.readUint32("unknown4"); /* fontSize */ fontSize = __dna_docin.readUint32("fontSize"); /* name */ name = __dna_docin.readString("name"); /* textureId */ __dna_docin.enumerate("textureId", textureId); /* textureFormat */ textureFormat = __dna_docin.readUint32("textureFormat"); /* glyphCount */ /* glyphs */ size_t count; if (auto v = __dna_docin.enterSubVector("glyphs", count)) { glyphCount = count; for (atUint32 i = 0; i < glyphCount; i++) { if (version < 4) glyphs.emplace_back(new GlyphMP1); else glyphs.emplace_back(new GlyphMP2); if (auto rec = __dna_docin.enterSubRecord()) glyphs.back()->read(__dna_docin); } } /* kerningInfoCount squelched */ /* kerningInfo */ kerningInfoCount = __dna_docin.enumerate("kerningInfo", kerningInfo); } template <class IDType> void FONT<IDType>::_write(athena::io::YAMLDocWriter& __dna_docout) const { /* version */ __dna_docout.writeUint32("version", version); /* unknown1 */ __dna_docout.writeUint32("unknown1", unknown1); /* lineHeight */ __dna_docout.writeInt32("lineHeight", lineHeight); /* verticalOffset */ __dna_docout.writeInt32("verticalOffset", verticalOffset); /* lineMargin */ __dna_docout.writeInt32("lineMargin", lineMargin); /* unknown2 */ __dna_docout.writeBool("unknown2", unknown2); /* unknown3 */ __dna_docout.writeBool("unknown3", unknown3); /* unknown4 */ __dna_docout.writeUint32("unknown4", unknown4); /* fontSize */ __dna_docout.writeUint32("fontSize", fontSize); /* name */ __dna_docout.writeString("name", name); /* textureId */ __dna_docout.enumerate("textureId", textureId); /* textureFormat */ __dna_docout.writeUint32("textureFormat", textureFormat); /* glyphCount squelched */ /* glyphs */ if (auto v = __dna_docout.enterSubVector("glyphs")) for (const std::unique_ptr<IGlyph>& glyph : glyphs) if (auto rec = __dna_docout.enterSubRecord()) glyph->write(__dna_docout); /* kerningInfoCount squelched */ /* kerningInfo */ __dna_docout.enumerate("kerningInfo", kerningInfo); } template <> std::string_view FONT<UniqueID32>::DNAType() { return "FONT<UniqueID32>"sv; } template <> std::string_view FONT<UniqueID64>::DNAType() { return "FONT<UniqueID64>"sv; } template <class IDType> void FONT<IDType>::_binarySize(size_t& __isz) const { __isz += name.size() + 1; textureId.binarySize(__isz); for (const std::unique_ptr<IGlyph>& glyph : glyphs) glyph->binarySize(__isz); for (const KerningInfo& k : kerningInfo) k.binarySize(__isz); __isz += 46; } AT_SUBSPECIALIZE_DNA_YAML(FONT<UniqueID32>) AT_SUBSPECIALIZE_DNA_YAML(FONT<UniqueID64>) template <class IDType> bool ExtractFONT(PAKEntryReadStream& rs, const hecl::ProjectPath& outPath) { athena::io::FileWriter writer(outPath.getAbsolutePath()); if (writer.isOpen()) { FONT<IDType> font; font.read(rs); athena::io::ToYAMLStream(font, writer); return true; } return false; } template bool ExtractFONT<UniqueID32>(PAKEntryReadStream& rs, const hecl::ProjectPath& outPath); template bool ExtractFONT<UniqueID64>(PAKEntryReadStream& rs, const hecl::ProjectPath& outPath); template <class IDType> bool WriteFONT(const FONT<IDType>& font, const hecl::ProjectPath& outPath) { athena::io::FileWriter w(outPath.getAbsolutePath(), true, false); if (w.hasError()) return false; font.write(w); int64_t rem = w.position() % 32; if (rem) for (int64_t i = 0; i < 32 - rem; ++i) w.writeUByte(0xff); return true; } template bool WriteFONT<UniqueID32>(const FONT<UniqueID32>& font, const hecl::ProjectPath& outPath); template bool WriteFONT<UniqueID64>(const FONT<UniqueID64>& font, const hecl::ProjectPath& outPath); } // namespace DataSpec::DNAFont