mirror of https://github.com/AxioDL/metaforce.git
668 lines
22 KiB
C++
668 lines
22 KiB
C++
#ifndef NOMINMAX
|
|
#define NOMINMAX 1
|
|
#endif
|
|
|
|
#include "specter/FontCache.hpp"
|
|
|
|
#include <cstddef>
|
|
#include <cstdint>
|
|
#include <memory>
|
|
#include <vector>
|
|
|
|
#include <athena/FileReader.hpp>
|
|
#include <athena/FileWriter.hpp>
|
|
#include <athena/MemoryReader.hpp>
|
|
|
|
#include <boo/System.hpp>
|
|
|
|
#include FT_GZIP_H
|
|
#include FT_SYSTEM_H
|
|
#include FT_OUTLINE_H
|
|
#include <freetype/internal/internal.h>
|
|
#include <freetype/internal/ftstream.h>
|
|
#include <freetype/internal/tttypes.h>
|
|
|
|
#include <hecl/hecl.hpp>
|
|
#include <hecl/Runtime.hpp>
|
|
|
|
#include <logvisor/logvisor.hpp>
|
|
#include <xxhash/xxhash.h>
|
|
#include <zlib.h>
|
|
|
|
extern "C" const uint8_t DROIDSANS_PERMISSIVE[];
|
|
extern "C" size_t DROIDSANS_PERMISSIVE_SZ;
|
|
|
|
extern "C" const uint8_t BMONOFONT[];
|
|
extern "C" size_t BMONOFONT_SZ;
|
|
|
|
extern "C" const uint8_t SPECTERCURVES[];
|
|
extern "C" size_t SPECTERCURVES_SZ;
|
|
|
|
extern "C" const FT_Driver_ClassRec tt_driver_class;
|
|
|
|
namespace specter {
|
|
static logvisor::Module Log("specter::FontCache");
|
|
|
|
const FCharFilter AllCharFilter{"all-glyphs", [](uint32_t) { return true; }};
|
|
|
|
const FCharFilter LatinCharFilter{"latin-glyphs",
|
|
[](uint32_t c) { return c <= 0xff || (c - 0x2200) <= (0x23FF - 0x2200); }};
|
|
|
|
const FCharFilter LatinAndJapaneseCharFilter{"latin-and-jp-glyphs", [](uint32_t c) {
|
|
return LatinCharFilter.second(c) || (c - 0x2E00) <= (0x30FF - 0x2E00) ||
|
|
(c - 0x4E00) <= (0x9FFF - 0x4E00) ||
|
|
(c - 0xFF00) <= (0xFFEF - 0xFF00);
|
|
}};
|
|
|
|
FontTag::FontTag(std::string_view name, bool subpixel, float points, uint32_t dpi) {
|
|
XXH64_state_t st;
|
|
XXH64_reset(&st, 0);
|
|
XXH64_update(&st, name.data(), name.size());
|
|
XXH64_update(&st, &subpixel, 1);
|
|
XXH64_update(&st, &points, 4);
|
|
XXH64_update(&st, &dpi, 4);
|
|
m_hash = XXH64_digest(&st);
|
|
}
|
|
|
|
FreeTypeGZipMemFace::FreeTypeGZipMemFace(FT_Library lib, const uint8_t* data, size_t sz) : m_lib(lib) {
|
|
m_comp.base = (unsigned char*)data;
|
|
m_comp.size = sz;
|
|
m_comp.memory = lib->memory;
|
|
}
|
|
|
|
void FreeTypeGZipMemFace::open() {
|
|
if (m_face)
|
|
return;
|
|
|
|
if (FT_Stream_OpenGzip(&m_decomp, &m_comp))
|
|
Log.report(logvisor::Fatal, FMT_STRING("unable to open FreeType gzip stream"));
|
|
|
|
FT_Open_Args args = {FT_OPEN_STREAM, nullptr, 0, nullptr, &m_decomp};
|
|
|
|
if (FT_Open_Face(m_lib, &args, 0, &m_face))
|
|
Log.report(logvisor::Fatal, FMT_STRING("unable to open FreeType gzip face"));
|
|
}
|
|
|
|
void FreeTypeGZipMemFace::close() {
|
|
if (!m_face)
|
|
return;
|
|
FT_Done_Face(m_face);
|
|
m_face = nullptr;
|
|
}
|
|
|
|
#define TEXMAP_DIM 1024
|
|
|
|
static unsigned RoundUpPow2(unsigned v) {
|
|
v--;
|
|
v |= v >> 1;
|
|
v |= v >> 2;
|
|
v |= v >> 4;
|
|
v |= v >> 8;
|
|
v |= v >> 16;
|
|
v++;
|
|
return v;
|
|
}
|
|
|
|
using GreyPixel = uint8_t;
|
|
static void MemcpyRect(GreyPixel* img, const FT_Bitmap* bmp, unsigned slice, unsigned x, unsigned y) {
|
|
unsigned sy = TEXMAP_DIM * slice + y;
|
|
for (unsigned i = 0; i < bmp->rows; ++i) {
|
|
const unsigned char* s = &bmp->buffer[bmp->pitch * i];
|
|
GreyPixel* t = &img[TEXMAP_DIM * (sy + i) + x];
|
|
memcpy(t, s, bmp->width);
|
|
}
|
|
}
|
|
|
|
union RgbaPixel {
|
|
uint8_t rgba[4];
|
|
uint32_t pixel;
|
|
};
|
|
static void MemcpyRect(RgbaPixel* img, const FT_Bitmap* bmp, unsigned slice, unsigned x, unsigned y) {
|
|
unsigned sy = TEXMAP_DIM * slice + y;
|
|
for (unsigned i = 0; i < bmp->rows; ++i) {
|
|
const unsigned char* s = &bmp->buffer[bmp->pitch * i];
|
|
RgbaPixel* t = &img[TEXMAP_DIM * (sy + i) + x];
|
|
for (unsigned j = 0; j < bmp->width / 3; ++j) {
|
|
t[j].rgba[0] = s[j * 3];
|
|
t[j].rgba[1] = s[j * 3 + 1];
|
|
t[j].rgba[2] = s[j * 3 + 2];
|
|
t[j].rgba[3] = 0xff;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void GridFitGlyph(FT_GlyphSlot slot, FT_UInt& width, FT_UInt& height) {
|
|
width = slot->metrics.width >> 6;
|
|
height = slot->metrics.height >> 6;
|
|
}
|
|
|
|
void FontAtlas::buildKernTable(FT_Face face) {
|
|
if (face->driver->clazz == &tt_driver_class) {
|
|
TT_Face ttface = reinterpret_cast<TT_Face>(face);
|
|
if (!ttface->kern_table)
|
|
return;
|
|
athena::io::MemoryReader r(ttface->kern_table, ttface->kern_table_size);
|
|
auto it = m_kernAdjs.end();
|
|
atUint32 nSubs = r.readUint32Big();
|
|
for (atUint32 i = 0; i < nSubs; ++i) {
|
|
TT_KernHead kernHead;
|
|
kernHead.read(r);
|
|
if (kernHead.coverage >> 8 != 0) {
|
|
r.seek(kernHead.length - 6, athena::SeekOrigin::Current);
|
|
continue;
|
|
}
|
|
|
|
TT_KernSubHead subHead;
|
|
subHead.read(r);
|
|
|
|
for (atUint16 p = 0; p < subHead.nPairs; ++p) {
|
|
TT_KernPair pair;
|
|
pair.read(r);
|
|
if (it == m_kernAdjs.end() || it->first != pair.left)
|
|
if ((it = m_kernAdjs.find(pair.left)) == m_kernAdjs.end())
|
|
it = m_kernAdjs.insert(std::make_pair(pair.left, std::vector<std::pair<atUint16, atInt16>>())).first;
|
|
it->second.emplace_back(pair.right, pair.value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#define NO_ZLIB 0
|
|
#define ZLIB_BUF_SZ 32768
|
|
|
|
static void WriteCompressed(athena::io::FileWriter& writer, const atUint8* data, size_t sz) {
|
|
#if NO_ZLIB
|
|
writer.writeUBytes(data, sz);
|
|
return;
|
|
#endif
|
|
|
|
atUint8 compBuf[ZLIB_BUF_SZ];
|
|
z_stream z = {};
|
|
deflateInit(&z, Z_DEFAULT_COMPRESSION);
|
|
z.next_in = (Bytef*)data;
|
|
z.avail_in = sz;
|
|
writer.writeUint32Big(sz);
|
|
atUint64 adlerPos = writer.position();
|
|
writer.writeUint32Big(0); /* Space for adler32 */
|
|
while (z.avail_in) {
|
|
z.next_out = compBuf;
|
|
z.avail_out = ZLIB_BUF_SZ;
|
|
deflate(&z, Z_NO_FLUSH);
|
|
writer.writeUBytes(compBuf, ZLIB_BUF_SZ - z.avail_out);
|
|
}
|
|
|
|
int finishCycle = Z_OK;
|
|
while (finishCycle != Z_STREAM_END) {
|
|
z.next_out = compBuf;
|
|
z.avail_out = ZLIB_BUF_SZ;
|
|
finishCycle = deflate(&z, Z_FINISH);
|
|
writer.writeUBytes(compBuf, ZLIB_BUF_SZ - z.avail_out);
|
|
}
|
|
|
|
writer.seek(adlerPos, athena::SeekOrigin::Begin);
|
|
writer.writeUint32Big(z.adler);
|
|
|
|
deflateEnd(&z);
|
|
}
|
|
|
|
static bool ReadDecompressed(athena::io::FileReader& reader, atUint8* data, size_t sz) {
|
|
#if NO_ZLIB
|
|
reader.readUBytesToBuf(data, sz);
|
|
return true;
|
|
#endif
|
|
|
|
atUint8 compBuf[ZLIB_BUF_SZ];
|
|
z_stream z = {};
|
|
inflateInit(&z);
|
|
z.next_out = data;
|
|
atUint32 targetSz = reader.readUint32Big();
|
|
atUint32 adler32 = reader.readUint32Big();
|
|
z.avail_out = std::min(sz, size_t(targetSz));
|
|
size_t readSz;
|
|
while ((readSz = reader.readUBytesToBuf(compBuf, ZLIB_BUF_SZ))) {
|
|
z.next_in = compBuf;
|
|
z.avail_in = readSz;
|
|
if (inflate(&z, Z_NO_FLUSH) == Z_STREAM_END)
|
|
break;
|
|
}
|
|
|
|
inflateEnd(&z);
|
|
return adler32 == z.adler;
|
|
}
|
|
|
|
FontAtlas::FontAtlas(FT_Face face, uint32_t dpi, bool subpixel, FCharFilter& filter, athena::io::FileWriter& writer)
|
|
: m_dpi(dpi)
|
|
, m_ftXscale(face->size->metrics.x_scale)
|
|
, m_ftXPpem(face->size->metrics.x_ppem)
|
|
, m_lineHeight(face->size->metrics.height)
|
|
, m_subpixel(subpixel) {
|
|
FT_Int32 baseFlags = FT_LOAD_NO_BITMAP;
|
|
if (subpixel)
|
|
baseFlags |= FT_LOAD_TARGET_LCD;
|
|
else
|
|
baseFlags |= FT_LOAD_TARGET_NORMAL;
|
|
|
|
/* First count glyphs exposed by unicode charmap and tally required area */
|
|
size_t glyphCount = 0;
|
|
FT_UInt gindex;
|
|
FT_ULong charcode = FT_Get_First_Char(face, &gindex);
|
|
unsigned curLineWidth = 1;
|
|
unsigned curLineHeight = 0;
|
|
unsigned totalHeight = 1;
|
|
m_fullTexmapLayers = 0;
|
|
while (gindex != 0) {
|
|
if (!filter.second(charcode)) {
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
continue;
|
|
}
|
|
++glyphCount;
|
|
FT_Load_Glyph(face, gindex, baseFlags);
|
|
FT_UInt width, height;
|
|
GridFitGlyph(face->glyph, width, height);
|
|
if (curLineWidth + width + 1 > TEXMAP_DIM) {
|
|
totalHeight += curLineHeight + 1;
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
curLineHeight = std::max(curLineHeight, height);
|
|
if (totalHeight + curLineHeight + 1 > TEXMAP_DIM) {
|
|
totalHeight = 1;
|
|
++m_fullTexmapLayers;
|
|
// printf("StagedB: %u\n", gindex);
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
curLineWidth += width + 1;
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
}
|
|
if (curLineHeight)
|
|
totalHeight += curLineHeight + 1;
|
|
m_glyphs.reserve(glyphCount);
|
|
m_glyphLookup.reserve(glyphCount);
|
|
|
|
totalHeight = RoundUpPow2(totalHeight);
|
|
m_finalHeight = m_fullTexmapLayers ? TEXMAP_DIM : totalHeight;
|
|
writer.writeUint32Big(m_fullTexmapLayers + 1);
|
|
writer.writeUint32Big(TEXMAP_DIM);
|
|
writer.writeUint32Big(m_finalHeight);
|
|
|
|
if (subpixel) {
|
|
/* Allocate texmap */
|
|
if (m_fullTexmapLayers) {
|
|
// printf("ALLOC: %u\n", fullTexmapLayers + 1);
|
|
size_t count = TEXMAP_DIM * TEXMAP_DIM * (m_fullTexmapLayers + 1);
|
|
m_texmap.resize(count * sizeof(RgbaPixel));
|
|
} else {
|
|
size_t count = TEXMAP_DIM * totalHeight;
|
|
m_texmap.resize(count * sizeof(RgbaPixel));
|
|
}
|
|
|
|
/* Assemble glyph texmaps and internal data structures */
|
|
charcode = FT_Get_First_Char(face, &gindex);
|
|
curLineWidth = 1;
|
|
curLineHeight = 0;
|
|
totalHeight = 1;
|
|
m_fullTexmapLayers = 0;
|
|
while (gindex != 0) {
|
|
if (!filter.second(charcode)) {
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
continue;
|
|
}
|
|
|
|
FT_Load_Glyph(face, gindex, FT_LOAD_RENDER | baseFlags);
|
|
FT_UInt width, height;
|
|
GridFitGlyph(face->glyph, width, height);
|
|
if (curLineWidth + width + 1 > TEXMAP_DIM) {
|
|
totalHeight += curLineHeight + 1;
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
curLineHeight = std::max(curLineHeight, height);
|
|
if (totalHeight + curLineHeight + 1 > TEXMAP_DIM) {
|
|
totalHeight = 1;
|
|
++m_fullTexmapLayers;
|
|
// printf("RealB: %u\n", gindex);
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
|
|
m_glyphLookup.insert_or_assign(charcode, m_glyphs.size());
|
|
Glyph& g = m_glyphs.emplace_back();
|
|
g.m_unicodePoint = charcode;
|
|
g.m_glyphIdx = gindex;
|
|
g.m_layerIdx = m_fullTexmapLayers;
|
|
g.m_layerFloat = float(g.m_layerIdx);
|
|
g.m_width = face->glyph->bitmap.width / 3;
|
|
g.m_height = face->glyph->bitmap.rows;
|
|
g.m_uv[0] = curLineWidth / float(TEXMAP_DIM);
|
|
g.m_uv[1] = totalHeight / float(m_finalHeight);
|
|
g.m_uv[2] = g.m_uv[0] + g.m_width / float(TEXMAP_DIM);
|
|
g.m_uv[3] = g.m_uv[1] + g.m_height / float(m_finalHeight);
|
|
g.m_leftPadding = face->glyph->metrics.horiBearingX >> 6;
|
|
g.m_advance = face->glyph->advance.x >> 6;
|
|
g.m_verticalOffset = (face->glyph->metrics.horiBearingY - face->glyph->metrics.height) >> 6;
|
|
MemcpyRect((RgbaPixel*)m_texmap.data(), &face->glyph->bitmap, m_fullTexmapLayers, curLineWidth, totalHeight);
|
|
curLineWidth += width + 1;
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
}
|
|
|
|
WriteCompressed(writer, m_texmap.data(), m_texmap.size());
|
|
} else {
|
|
/* Allocate texmap */
|
|
if (m_fullTexmapLayers) {
|
|
// printf("ALLOC: %u\n", fullTexmapLayers + 1);
|
|
size_t count = TEXMAP_DIM * TEXMAP_DIM * (m_fullTexmapLayers + 1);
|
|
m_texmap.resize(count * sizeof(GreyPixel));
|
|
} else {
|
|
size_t count = TEXMAP_DIM * totalHeight;
|
|
m_texmap.resize(count * sizeof(GreyPixel));
|
|
}
|
|
|
|
/* Assemble glyph texmaps and internal data structures */
|
|
charcode = FT_Get_First_Char(face, &gindex);
|
|
curLineWidth = 1;
|
|
curLineHeight = 0;
|
|
totalHeight = 1;
|
|
m_fullTexmapLayers = 0;
|
|
while (gindex != 0) {
|
|
if (!filter.second(charcode)) {
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
continue;
|
|
}
|
|
|
|
FT_Load_Glyph(face, gindex, FT_LOAD_RENDER | baseFlags);
|
|
FT_UInt width, height;
|
|
GridFitGlyph(face->glyph, width, height);
|
|
if (curLineWidth + width + 1 > TEXMAP_DIM) {
|
|
totalHeight += curLineHeight + 1;
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
curLineHeight = std::max(curLineHeight, height);
|
|
if (totalHeight + curLineHeight + 1 > TEXMAP_DIM) {
|
|
totalHeight = 1;
|
|
++m_fullTexmapLayers;
|
|
// printf("RealB: %u\n", gindex);
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
|
|
m_glyphLookup.insert_or_assign(charcode, m_glyphs.size());
|
|
Glyph& g = m_glyphs.emplace_back();
|
|
g.m_unicodePoint = charcode;
|
|
g.m_glyphIdx = gindex;
|
|
g.m_layerIdx = m_fullTexmapLayers;
|
|
g.m_layerFloat = float(g.m_layerIdx);
|
|
g.m_width = face->glyph->bitmap.width;
|
|
g.m_height = face->glyph->bitmap.rows;
|
|
g.m_uv[0] = curLineWidth / float(TEXMAP_DIM);
|
|
g.m_uv[1] = totalHeight / float(m_finalHeight);
|
|
g.m_uv[2] = g.m_uv[0] + g.m_width / float(TEXMAP_DIM);
|
|
g.m_uv[3] = g.m_uv[1] + g.m_height / float(m_finalHeight);
|
|
g.m_leftPadding = face->glyph->metrics.horiBearingX >> 6;
|
|
g.m_advance = face->glyph->advance.x >> 6;
|
|
g.m_verticalOffset = (face->glyph->metrics.horiBearingY - face->glyph->metrics.height) >> 6;
|
|
MemcpyRect(m_texmap.data(), &face->glyph->bitmap, m_fullTexmapLayers, curLineWidth, totalHeight);
|
|
curLineWidth += width + 1;
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
}
|
|
|
|
WriteCompressed(writer, m_texmap.data(), m_texmap.size());
|
|
}
|
|
|
|
buildKernTable(face);
|
|
m_ready = true;
|
|
}
|
|
|
|
FontAtlas::FontAtlas(FT_Face face, uint32_t dpi, bool subpixel, FCharFilter& filter, athena::io::FileReader& reader)
|
|
: m_dpi(dpi)
|
|
, m_ftXscale(face->size->metrics.x_scale)
|
|
, m_ftXPpem(face->size->metrics.x_ppem)
|
|
, m_lineHeight(face->size->metrics.height)
|
|
, m_subpixel(subpixel) {
|
|
FT_Int32 baseFlags = FT_LOAD_NO_BITMAP;
|
|
if (subpixel)
|
|
baseFlags |= FT_LOAD_TARGET_LCD;
|
|
else
|
|
baseFlags |= FT_LOAD_TARGET_NORMAL;
|
|
|
|
/* First count glyphs exposed by unicode charmap */
|
|
size_t glyphCount = 0;
|
|
FT_UInt gindex;
|
|
FT_ULong charcode = FT_Get_First_Char(face, &gindex);
|
|
while (gindex != 0) {
|
|
if (!filter.second(charcode)) {
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
continue;
|
|
}
|
|
++glyphCount;
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
}
|
|
m_glyphs.reserve(glyphCount);
|
|
m_glyphLookup.reserve(glyphCount);
|
|
|
|
m_fullTexmapLayers = reader.readUint32Big() - 1;
|
|
reader.readUint32Big();
|
|
m_finalHeight = reader.readUint32Big();
|
|
|
|
if (subpixel) {
|
|
/* Allocate texmap */
|
|
if (m_fullTexmapLayers) {
|
|
// printf("ALLOC: %u\n", fullTexmapLayers + 1);
|
|
size_t count = TEXMAP_DIM * TEXMAP_DIM * (m_fullTexmapLayers + 1);
|
|
m_texmap.resize(count * sizeof(RgbaPixel));
|
|
} else {
|
|
size_t count = TEXMAP_DIM * m_finalHeight;
|
|
m_texmap.resize(count * sizeof(RgbaPixel));
|
|
}
|
|
|
|
/* Assemble glyph texmaps and internal data structures */
|
|
charcode = FT_Get_First_Char(face, &gindex);
|
|
unsigned curLineWidth = 1;
|
|
unsigned curLineHeight = 0;
|
|
unsigned totalHeight = 1;
|
|
m_fullTexmapLayers = 0;
|
|
while (gindex != 0) {
|
|
if (!filter.second(charcode)) {
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
continue;
|
|
}
|
|
|
|
FT_Load_Glyph(face, gindex, baseFlags);
|
|
FT_UInt width, height;
|
|
GridFitGlyph(face->glyph, width, height);
|
|
if (curLineWidth + width + 1 > TEXMAP_DIM) {
|
|
totalHeight += curLineHeight + 1;
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
curLineHeight = std::max(curLineHeight, height);
|
|
if (totalHeight + curLineHeight + 1 > TEXMAP_DIM) {
|
|
totalHeight = 1;
|
|
++m_fullTexmapLayers;
|
|
// printf("RealB: %u\n", gindex);
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
|
|
m_glyphLookup.insert_or_assign(charcode, m_glyphs.size());
|
|
Glyph& g = m_glyphs.emplace_back();
|
|
g.m_unicodePoint = charcode;
|
|
g.m_glyphIdx = gindex;
|
|
g.m_layerIdx = m_fullTexmapLayers;
|
|
g.m_layerFloat = float(g.m_layerIdx);
|
|
g.m_width = width;
|
|
g.m_height = height;
|
|
g.m_uv[0] = curLineWidth / float(TEXMAP_DIM);
|
|
g.m_uv[1] = totalHeight / float(m_finalHeight);
|
|
g.m_uv[2] = g.m_uv[0] + g.m_width / float(TEXMAP_DIM);
|
|
g.m_uv[3] = g.m_uv[1] + g.m_height / float(m_finalHeight);
|
|
g.m_leftPadding = face->glyph->metrics.horiBearingX >> 6;
|
|
g.m_advance = face->glyph->advance.x >> 6;
|
|
g.m_verticalOffset = (face->glyph->metrics.horiBearingY - face->glyph->metrics.height) >> 6;
|
|
curLineWidth += width + 1;
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
}
|
|
|
|
if (!ReadDecompressed(reader, m_texmap.data(), m_texmap.size()))
|
|
return;
|
|
} else {
|
|
/* Allocate texmap */
|
|
if (m_fullTexmapLayers) {
|
|
// printf("ALLOC: %u\n", fullTexmapLayers + 1);
|
|
size_t count = TEXMAP_DIM * TEXMAP_DIM * (m_fullTexmapLayers + 1);
|
|
m_texmap.resize(count * sizeof(GreyPixel));
|
|
} else {
|
|
size_t count = TEXMAP_DIM * m_finalHeight;
|
|
m_texmap.resize(count * sizeof(GreyPixel));
|
|
}
|
|
|
|
/* Assemble glyph texmaps and internal data structures */
|
|
charcode = FT_Get_First_Char(face, &gindex);
|
|
unsigned curLineWidth = 1;
|
|
unsigned curLineHeight = 0;
|
|
unsigned totalHeight = 1;
|
|
m_fullTexmapLayers = 0;
|
|
while (gindex != 0) {
|
|
if (!filter.second(charcode)) {
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
continue;
|
|
}
|
|
|
|
FT_Load_Glyph(face, gindex, baseFlags);
|
|
FT_UInt width, height;
|
|
GridFitGlyph(face->glyph, width, height);
|
|
if (curLineWidth + width + 1 > TEXMAP_DIM) {
|
|
totalHeight += curLineHeight + 1;
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
curLineHeight = std::max(curLineHeight, height);
|
|
if (totalHeight + curLineHeight + 1 > TEXMAP_DIM) {
|
|
totalHeight = 1;
|
|
++m_fullTexmapLayers;
|
|
// printf("RealB: %u\n", gindex);
|
|
curLineHeight = 0;
|
|
curLineWidth = 1;
|
|
}
|
|
|
|
m_glyphLookup.insert_or_assign(charcode, m_glyphs.size());
|
|
Glyph& g = m_glyphs.emplace_back();
|
|
g.m_unicodePoint = charcode;
|
|
g.m_glyphIdx = gindex;
|
|
g.m_layerIdx = m_fullTexmapLayers;
|
|
g.m_layerFloat = float(g.m_layerIdx);
|
|
g.m_width = width;
|
|
g.m_height = height;
|
|
g.m_uv[0] = curLineWidth / float(TEXMAP_DIM);
|
|
g.m_uv[1] = totalHeight / float(m_finalHeight);
|
|
g.m_uv[2] = g.m_uv[0] + g.m_width / float(TEXMAP_DIM);
|
|
g.m_uv[3] = g.m_uv[1] + g.m_height / float(m_finalHeight);
|
|
g.m_leftPadding = face->glyph->metrics.horiBearingX >> 6;
|
|
g.m_advance = face->glyph->advance.x >> 6;
|
|
g.m_verticalOffset = (face->glyph->metrics.horiBearingY - face->glyph->metrics.height) >> 6;
|
|
curLineWidth += width + 1;
|
|
charcode = FT_Get_Next_Char(face, charcode, &gindex);
|
|
}
|
|
|
|
if (!ReadDecompressed(reader, m_texmap.data(), m_texmap.size()))
|
|
return;
|
|
}
|
|
|
|
buildKernTable(face);
|
|
m_ready = true;
|
|
}
|
|
|
|
boo::ObjToken<boo::ITextureSA> FontAtlas::texture(boo::IGraphicsDataFactory* gf) const {
|
|
if (!m_ready)
|
|
return {};
|
|
if (m_tex)
|
|
return m_tex;
|
|
gf->commitTransaction([&](boo::IGraphicsDataFactory::Context& ctx) {
|
|
if (m_subpixel)
|
|
const_cast<boo::ObjToken<boo::ITextureSA>&>(m_tex) =
|
|
ctx.newStaticArrayTexture(TEXMAP_DIM, m_finalHeight, m_fullTexmapLayers + 1, 1, boo::TextureFormat::RGBA8,
|
|
boo::TextureClampMode::Repeat, m_texmap.data(), m_texmap.size());
|
|
else
|
|
const_cast<boo::ObjToken<boo::ITextureSA>&>(m_tex) =
|
|
ctx.newStaticArrayTexture(TEXMAP_DIM, m_finalHeight, m_fullTexmapLayers + 1, 1, boo::TextureFormat::I8,
|
|
boo::TextureClampMode::Repeat, m_texmap.data(), m_texmap.size());
|
|
const_cast<std::vector<uint8_t>&>(m_texmap) = std::vector<uint8_t>();
|
|
return true;
|
|
} BooTrace);
|
|
return m_tex;
|
|
}
|
|
|
|
FontCache::Library::Library() {
|
|
FT_Error err = FT_Init_FreeType(&m_lib);
|
|
if (err)
|
|
Log.report(logvisor::Fatal, FMT_STRING("unable to FT_Init_FreeType"));
|
|
}
|
|
|
|
FontCache::Library::~Library() { FT_Done_FreeType(m_lib); }
|
|
|
|
FontCache::FontCache(const hecl::Runtime::FileStoreManager& fileMgr)
|
|
: m_fileMgr(fileMgr)
|
|
, m_cacheRoot(hecl::SystemString(m_fileMgr.getStoreRoot()) + _SYS_STR("/fontcache"))
|
|
, m_regFace(m_fontLib, DROIDSANS_PERMISSIVE, DROIDSANS_PERMISSIVE_SZ)
|
|
, m_monoFace(m_fontLib, BMONOFONT, BMONOFONT_SZ)
|
|
, m_curvesFace(m_fontLib, SPECTERCURVES, SPECTERCURVES_SZ) {
|
|
hecl::MakeDir(m_cacheRoot.c_str());
|
|
}
|
|
|
|
FontCache::~FontCache() = default;
|
|
|
|
FontTag FontCache::prepCustomFont(std::string_view name, FT_Face face, FCharFilter filter, bool subpixel, float points,
|
|
uint32_t dpi) {
|
|
/* Quick validation */
|
|
if (!face)
|
|
Log.report(logvisor::Fatal, FMT_STRING("invalid freetype face"));
|
|
|
|
if (!face->charmap || face->charmap->encoding != FT_ENCODING_UNICODE)
|
|
Log.report(logvisor::Fatal, FMT_STRING("font does not contain a unicode char map"));
|
|
|
|
/* Set size with FreeType */
|
|
FT_Set_Char_Size(face, 0, points * 64.0, 0, dpi);
|
|
|
|
/* Make tag and search for cached version */
|
|
const FontTag tag(std::string(name).append(1, '_').append(filter.first), subpixel, points, dpi);
|
|
const auto search = m_cachedAtlases.find(tag);
|
|
if (search != m_cachedAtlases.end()) {
|
|
return tag;
|
|
}
|
|
|
|
/* Now check filesystem cache */
|
|
hecl::SystemString cachePath = m_cacheRoot + _SYS_STR('/') + fmt::format(FMT_STRING(_SYS_STR("{:x}")), tag.hash());
|
|
hecl::Sstat st;
|
|
if (!hecl::Stat(cachePath.c_str(), &st) && S_ISREG(st.st_mode)) {
|
|
athena::io::FileReader r(cachePath);
|
|
if (!r.hasError()) {
|
|
atUint32 magic = r.readUint32Big();
|
|
if (r.position() == 4 && magic == 'FONT') {
|
|
std::unique_ptr<FontAtlas> fa = std::make_unique<FontAtlas>(face, dpi, subpixel, filter, r);
|
|
if (fa->isReady()) {
|
|
m_cachedAtlases.emplace(tag, std::move(fa));
|
|
return tag;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Nada, build and cache now */
|
|
athena::io::FileWriter w(cachePath);
|
|
if (w.hasError())
|
|
Log.report(logvisor::Fatal, FMT_STRING(_SYS_STR("unable to open '{}' for writing")), cachePath);
|
|
w.writeUint32Big('FONT');
|
|
m_cachedAtlases.emplace(tag, std::make_unique<FontAtlas>(face, dpi, subpixel, filter, w));
|
|
return tag;
|
|
}
|
|
|
|
const FontAtlas& FontCache::lookupAtlas(FontTag tag) const {
|
|
auto search = m_cachedAtlases.find(tag);
|
|
if (search == m_cachedAtlases.cend())
|
|
Log.report(logvisor::Fatal, FMT_STRING("invalid font"));
|
|
return *search->second.get();
|
|
}
|
|
|
|
} // namespace specter
|