osb/source/core/StarFont.cpp

169 lines
4.5 KiB
C++
Raw Normal View History

2023-06-20 14:33:09 +10:00
#include "StarFont.hpp"
#include "StarFile.hpp"
#include "StarFormat.hpp"
#include <ft2build.h>
#include FT_FREETYPE_H
namespace Star {
constexpr int FontLoadFlags = FT_LOAD_NO_SVG | FT_LOAD_COLOR;
2023-06-20 14:33:09 +10:00
struct FTContext {
FT_Library library;
FTContext() {
library = nullptr;
if (FT_Init_FreeType(&library))
throw FontException("Could not initialize freetype library.");
}
~FTContext() {
if (library) {
FT_Done_FreeType(library);
library = nullptr;
}
}
};
FTContext ftContext;
struct FontImpl {
FT_Face face;
2024-04-27 06:46:20 +10:00
unsigned loadedPixelSize = 0;
String::Char loadedChar = 0;
2023-06-20 14:33:09 +10:00
};
2024-03-25 12:49:18 +11:00
FontPtr Font::loadFont(String const& fileName, unsigned pixelSize) {
return loadFont(make_shared<ByteArray>(File::readFile(fileName)), pixelSize);
2023-06-20 14:33:09 +10:00
}
2024-03-25 12:49:18 +11:00
FontPtr Font::loadFont(ByteArrayConstPtr const& bytes, unsigned pixelSize) {
2023-06-20 14:33:09 +10:00
FontPtr font = make_shared<Font>();
font->m_fontBuffer = bytes;
shared_ptr<FontImpl> fontImpl = make_shared<FontImpl>();
if (FT_New_Memory_Face(
ftContext.library, (FT_Byte const*)font->m_fontBuffer->ptr(), font->m_fontBuffer->size(), 0, &fontImpl->face))
throw FontException("Could not load font from buffer");
font->m_fontImpl = fontImpl;
font->setPixelSize(pixelSize);
return font;
}
Font::Font() : m_pixelSize(0), m_alphaThreshold(0) {}
2023-06-20 14:33:09 +10:00
FontPtr Font::clone() const {
2024-03-25 12:49:18 +11:00
return Font::loadFont(m_fontBuffer, m_pixelSize);
2023-06-20 14:33:09 +10:00
}
void Font::setPixelSize(unsigned pixelSize) {
if (pixelSize == 0) {
pixelSize = 1;
}
if (m_pixelSize == pixelSize)
return;
if (FT_Set_Pixel_Sizes(m_fontImpl->face, pixelSize, 0))
2023-06-27 20:23:44 +10:00
throw FontException(strf("Cannot set font pixel size to: {}", pixelSize));
2023-06-20 14:33:09 +10:00
m_pixelSize = pixelSize;
}
void Font::setAlphaThreshold(uint8_t alphaThreshold) {
m_alphaThreshold = alphaThreshold;
}
2023-06-20 14:33:09 +10:00
unsigned Font::height() const {
return m_pixelSize;
}
unsigned Font::width(String::Char c) {
if (auto width = m_widthCache.maybe({c, m_pixelSize})) {
return *width;
} else {
FT_Load_Char(m_fontImpl->face, c, FontLoadFlags);
2023-08-04 00:21:24 +10:00
unsigned newWidth = (m_fontImpl->face->glyph->linearHoriAdvance + 32768) / 65536;
2023-06-20 14:33:09 +10:00
m_widthCache.insert({c, m_pixelSize}, newWidth);
return newWidth;
}
}
tuple<Image, Vec2I, bool> Font::render(String::Char c) {
2023-06-20 14:33:09 +10:00
if (!m_fontImpl)
throw FontException("Font::render called on uninitialized font.");
2023-06-20 14:33:09 +10:00
FT_Face face = m_fontImpl->face;
2024-04-27 06:46:20 +10:00
if (m_fontImpl->loadedPixelSize != m_pixelSize || m_fontImpl->loadedChar != c) {
FT_UInt glyph_index = FT_Get_Char_Index(face, c);
if (FT_Load_Glyph(face, glyph_index, FontLoadFlags) != 0)
return {};
2024-04-27 06:46:20 +10:00
// convert to an anti-aliased bitmap
if (FT_Render_Glyph(face->glyph, FT_RENDER_MODE_NORMAL) != 0)
return {};
}
2023-06-20 14:33:09 +10:00
2024-04-27 06:46:20 +10:00
m_fontImpl->loadedPixelSize = m_pixelSize;
m_fontImpl->loadedChar = c;
2023-06-20 14:33:09 +10:00
FT_GlyphSlot slot = face->glyph;
2024-04-27 06:46:20 +10:00
if (!slot->bitmap.buffer)
return {};
2023-06-20 14:33:09 +10:00
unsigned width = slot->bitmap.width;
unsigned height = slot->bitmap.rows;
bool colored = false;
Image image(width + 2, height + 2, PixelFormat::BGRA32);
if (slot->bitmap.pixel_mode == FT_PIXEL_MODE_GRAY) {
Vec4B white(255, 255, 255, 0);
image.fill(white);
for (unsigned y = 0; y != height; ++y) {
uint8_t* p = slot->bitmap.buffer + y * slot->bitmap.pitch;
for (unsigned x = 0; x != width; ++x) {
if (x < width && y < height) {
white[3] = m_alphaThreshold
? (*(p + x) >= m_alphaThreshold ? 255 : 0)
: *(p + x);
image.set(x + 1, height - y, white);
}
2023-06-20 14:33:09 +10:00
}
}
} else if (colored = slot->bitmap.pixel_mode == FT_PIXEL_MODE_BGRA) {
unsigned bpp = image.bytesPerPixel();
uint8_t* data = image.data() + bpp + ((image.width() * (image.height() - 2)) * bpp); // offset by 1 pixel as it's padded
for (size_t y = 0; y != height; ++y) {
memcpy(data - (y * image.width() * bpp),
slot->bitmap.buffer + y * slot->bitmap.pitch,
slot->bitmap.pitch);
}
// unfortunately FreeType pre-multiplied the color channels :(
image.forEachPixel([](unsigned, unsigned, Vec4B& pixel) {
if (pixel[3] != 0 && pixel[3] != 255) {
float a = byteToFloat(pixel[3]);
pixel[0] /= a;
pixel[1] /= a;
pixel[2] /= a;
}
});
} else {
return {};
2023-06-20 14:33:09 +10:00
}
return make_tuple(
std::move(image),
Vec2I(slot->bitmap_left - 1, (slot->bitmap_top - height) + (m_pixelSize / 4) - 1),
colored);
2023-06-20 14:33:09 +10:00
}
bool Font::exists(String::Char c) {
return FT_Get_Char_Index(m_fontImpl->face, c);
}
2023-06-20 14:33:09 +10:00
}