/* Copyright (c) 2013 yvt This file is part of OpenSpades. OpenSpades is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. OpenSpades is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with OpenSpades. If not, see . */ #include #include #include FT_FREETYPE_H #include FT_BITMAP_H #include "FTFont.h" #include #include #include #include #include #include namespace spades { namespace ngclient { namespace { struct FreeType { FT_Library library; FreeType() { FT_Init_FreeType(&library); } ~FreeType() { FT_Done_FreeType(library); } operator FT_Library() const { return library; } }; AutoDeletedThreadLocalStorage ft; FT_Library GetFreeType() { if (ft == nullptr) { ft = new FreeType(); } return *ft; } }; // namespace struct FTFaceWrapper { FT_Face face; std::string buffer; FTFaceWrapper(FT_Face f) : face(f) {} ~FTFaceWrapper() { FT_Done_Face(face); } operator FT_Face() const { return face; } }; struct FTBitmapWrapper { FT_Bitmap b; FTBitmapWrapper() { FT_Bitmap_New(&b); } ~FTBitmapWrapper() { FT_Bitmap_Done(GetFreeType(), &b); } operator FT_Bitmap *() { return &b; } FT_Bitmap *operator->() { return &b; } }; FTFontSet::FTFontSet() { SPADES_MARK_FUNCTION(); } FTFontSet::~FTFontSet() { SPADES_MARK_FUNCTION(); } void FTFontSet::AddFace(const std::string &fileName) { FT_Face face; std::string data = FileManager::ReadAllBytes(fileName.c_str()); auto ret = FT_New_Memory_Face( GetFreeType(), reinterpret_cast(data.data()), data.size(), 0, &face); if (ret) { SPRaise("Failed to load font %s: FreeType error %d", fileName.c_str(), ret); } auto wr = stmp::make_unique(face); wr->buffer = std::move(data); faces.emplace_back(std::move(wr)); } struct BinPlaceResult { std::reference_wrapper image; int x, y; BinPlaceResult(client::IImage &image, int x, int y) : image(image), x(x), y(y) {} }; struct FTFont::Bin { int const width, height; Handle image; std::list> skyline; Bin(int width, int height, client::IRenderer &r) : width(width), height(height) { auto tmpbmp = Handle::New(width, height); memset(tmpbmp->GetPixels(), 0, tmpbmp->GetWidth() * tmpbmp->GetHeight() * 4); image = r.CreateImage(*tmpbmp); skyline.emplace_back(0, 0); skyline.emplace_back(width, 0); } stmp::optional Place(Bitmap &bmp) { int bw = bmp.GetWidth(), bh = bmp.GetHeight(); if (bw > width || bh > height) { SPRaise("Impossible to allocate bitmap of " "%dx%d on %dx%d bin.", bw, bh, width, height); } auto it = skyline.begin(); auto it2 = it; ++it2; auto bestIt1 = it, bestIt2 = it2; int bestWasted = std::numeric_limits::max(); int bestMaxY = 0; while (it2 != skyline.end() && it != skyline.end()) { while (it2 != skyline.end() && it2->first < it->first + bw) { ++it2; } if (it2 == skyline.end()) { break; } auto maxYIter = std::max_element( it, it2, [](const std::pair &a, const std::pair &b) { return a.second < b.second; }); auto maxY = maxYIter->second; if (maxY + bh < height) { int right = it->first + bw; int wasted = 0; for (auto it1 = it; it1 != it2;) { auto cur = it1++; auto segRight = it1->first; wasted += std::min(segRight, right) * (maxY - cur->second); } if (wasted < bestWasted) { bestIt1 = it; bestIt2 = it2; bestWasted = wasted; bestMaxY = maxY; } } ++it; } if (bestWasted == std::numeric_limits::max()) { // failed return stmp::optional(); } SPAssert(bestIt1 != bestIt2); int right = bestIt1->first + bw; BinPlaceResult result(*image, bestIt1->first, bestMaxY); if (bestIt2->first == right) { it = bestIt1; ++it; skyline.erase(it, bestIt2); bestIt1->second = bestMaxY + bh; } else { it = bestIt2; --it; if (it == bestIt1) { it = skyline.emplace(bestIt2, right, bestIt1->second); --bestIt2; } else { --bestIt2; bestIt2->first = right; } it = bestIt1; ++it; skyline.erase(it, bestIt2); bestIt1->second = bestMaxY + bh; } image->Update(bmp, result.x, result.y); return result; } }; FTFont::FTFont(client::IRenderer *renderer, std::shared_ptr _fontSet, float height, float lineHeight) : client::IFont(renderer), renderer(renderer), lineHeight(lineHeight), height(height), fontSet(std::move(_fontSet)) { SPADES_MARK_FUNCTION(); SPAssert(renderer); SPAssert(fontSet); binSize = 256; int targetSize = static_cast(std::min(height * 4.f, 4000.f)); while (binSize < targetSize) { binSize <<= 1; } baselineY = std::floor(height * 1.f); rendererIsLowQuality = dynamic_cast(renderer); bins.emplace_back(binSize, binSize, *renderer); } FTFont::~FTFont() { SPADES_MARK_FUNCTION(); } stmp::optional FTFont::GetGlyph(uint32_t code) { auto it = glyphMap.find(code); if (it != glyphMap.end()) { auto ref = it->second; return ref.get(); } for (const auto &face : fontSet->faces) { auto cId = FT_Get_Char_Index(*face, code); if (cId != 0) { auto it2 = glyphs.find(std::make_pair(*face, cId)); if (it2 == glyphs.end()) { FT_Set_Char_Size(*face, 0, static_cast(height * 64.f), 72, 72); Glyph g; g.face = *face; g.charIndex = cId; FT_Load_Glyph(*face, cId, FT_LOAD_NO_HINTING); const auto &adv = g.face->glyph->advance; g.advance = Vector2(adv.x, adv.y) / (64.f); auto it3 = glyphs.emplace(std::make_pair(*face, cId), std::move(g)); glyphMap.emplace(code, it3.first->second); return it3.first->second; } else { return it2->second; } } } return {}; } template void FTFont::SplitTextIntoGlyphs(const std::string &str, T onGlyph, T3 onFallback, T2 onLineBreak) { // FIXME: ligatures? std::size_t i = 0; uint32_t breakCode = 0; while (i < str.size()) { std::size_t advance; auto code = GetCodePointFromUTF8String(str, i, &advance); i += advance; if (code == 10 || code == 13) { if (breakCode == 0) breakCode = code; if (code == breakCode) { onLineBreak(); } continue; } auto g = GetGlyph(code); if (g) { onGlyph(*g); } else { onFallback(code); } } } Vector2 FTFont::Measure(const std::string &str) { SPADES_MARK_FUNCTION(); float maxWidth = 0.f; float x = 0.f; int lines = 1; SplitTextIntoGlyphs( str, [&](Glyph &g) { x += g.advance.x; maxWidth = std::max(x, maxWidth); }, [&](uint32_t codepoint) { x += MeasureFallback(codepoint, height); maxWidth = std::max(x, maxWidth); }, [&]() { ++lines; x = 0.f; }); return Vector2(maxWidth, lines * lineHeight); } void FTFont::RenderGlyph(Glyph &g) { if (g.image) return; FT_Set_Char_Size(g.face, 0, static_cast(height * 64.f), 72, 72); FT_Load_Glyph(g.face, g.charIndex, FT_LOAD_NO_HINTING); FT_Render_Glyph(g.face->glyph, FT_RENDER_MODE_NORMAL); auto &bmp = g.face->glyph->bitmap; FTBitmapWrapper outbmp; FT_Bitmap_Convert(GetFreeType(), &bmp, outbmp, 1); SPAssert(outbmp->pixel_mode == FT_PIXEL_MODE_GRAY); auto spbmp = Handle::New(outbmp->width + 1, outbmp->rows + 1); memset(spbmp->GetPixels(), 0, 4 * spbmp->GetWidth() * spbmp->GetHeight()); for (int y = 0; y < outbmp->rows; ++y) { const auto *inpixs = outbmp->buffer + y * outbmp->width; auto *outpixs = spbmp->GetPixels(); outpixs += y * spbmp->GetWidth(); for (int x = 0; x < outbmp->width; ++x) { uint32_t v = *inpixs; v = (v << 24) | 0xffffff; *outpixs = v; outpixs++; inpixs++; } } // allocate bin auto result = bins.back().Place(*spbmp); if (!result) { // bin full bins.emplace_back(binSize, binSize, *renderer); result = bins.back().Place(*spbmp); SPAssert(result); } AABB2 bounds((*result).x, (*result).y, spbmp->GetWidth() - 1, spbmp->GetHeight() - 1); Vector2 offs(g.face->glyph->bitmap_left, baselineY - g.face->glyph->bitmap_top); g.image.reset((*result).image, bounds, offs); g.bmp = spbmp; } void FTFont::RenderBlurGlyph(Glyph &g) { RenderGlyph(g); if (g.blurImage) return; enum { KernelSize = 6 }; auto &orig = *g.bmp; auto newbmp = Handle::New(orig.GetWidth() + KernelSize, orig.GetHeight() + KernelSize); int const origW = orig.GetWidth(); int const origH = orig.GetHeight(); int const newW = newbmp->GetWidth(); int const newH = newbmp->GetHeight(); std::vector buf; buf.resize(newW + KernelSize); for (int y = 0; y < newH; ++y) { int iy = y - (KernelSize >> 1); auto *pixels = newbmp->GetPixels() + y * newW; if (iy >= 0 && iy < origH) { { auto *inp = orig.GetPixels() + iy * origW; auto *outp = buf.data() + KernelSize; for (int x = 0; x < origW; ++x) { *(outp++) = *(inp++) >> 24; } } for (int x = 0; x < newW; ++x) { uint32_t sum; sum = static_cast(buf[x]) * 3590U; sum += static_cast(buf[x + 1]) * 8121U; sum += static_cast(buf[x + 2]) * 13254U; sum += static_cast(buf[x + 3]) * 15606U; sum += static_cast(buf[x + 4]) * 13254U; sum += static_cast(buf[x + 5]) * 8121U; sum += static_cast(buf[x + 6]) * 3590U; *(pixels++) = (sum >> 16 << 24) | 0xffffff; } } else { memset(pixels, 0, newW * 4); } } buf.resize(newH + KernelSize); for (auto &e : buf) e = 0; for (int x = 0; x < newW; ++x) { auto *pixels = newbmp->GetPixels() + x; { auto *inp = pixels; auto *outp = buf.data() + (KernelSize >> 1); for (int y = 0; y < newH; ++y) { *(outp++) = *(inp) >> 24; inp += newW; } } for (int y = 0; y < newH; ++y) { uint32_t sum; sum = static_cast(buf[y]) * 3590U; sum += static_cast(buf[y + 1]) * 8121U; sum += static_cast(buf[y + 2]) * 13254U; sum += static_cast(buf[y + 3]) * 15606U; sum += static_cast(buf[y + 4]) * 13254U; sum += static_cast(buf[y + 5]) * 8121U; sum += static_cast(buf[y + 6]) * 3590U; *(pixels) = (sum >> 16 << 24) | 0xffffff; pixels += newW; } } // allocate bin auto result = bins.back().Place(*newbmp); if (!result) { // bin full bins.emplace_back(binSize, binSize, *renderer); result = bins.back().Place(*newbmp); SPAssert(result); } AABB2 bounds((*result).x, (*result).y, newbmp->GetWidth(), newbmp->GetHeight()); Vector2 offs = (*g.image).offset - Vector2(1, 1) * (KernelSize * 0.5f); g.blurImage.reset((*result).image, bounds, offs); } void FTFont::Draw(const std::string &str, Vector2 offset, float scale, Vector4 color) { SPADES_MARK_FUNCTION(); auto lines = SplitIntoLines(str); float x = 0.f; float y = 0.f; color = Vector4(color.x * color.w, color.y * color.w, color.z * color.w, color.w); renderer->SetColorAlphaPremultiplied(color); SplitTextIntoGlyphs( str, [&](Glyph &g) { RenderGlyph(g); auto &img = *g.image; auto srcBounds = img.bounds; auto target = offset + (Vector2(x, y) + img.offset) * scale; target = (target + .5f).Floor(); // for sharper rendering AABB2 destBounds(target.x, target.y, srcBounds.GetWidth() * scale, srcBounds.GetHeight() * scale); if (!rendererIsLowQuality) { srcBounds = srcBounds.Inflate(.5f); destBounds = destBounds.Inflate(.5f * scale); } renderer->DrawImage(&img.img, destBounds, srcBounds); x += g.advance.x; y += g.advance.y; }, [&](uint32_t codepoint) { DrawFallback(codepoint, offset + Vector2(x, y) * scale, height * scale, color); x += MeasureFallback(codepoint, height); }, [&]() { x = 0.f; y += lineHeight; }); } void FTFont::DrawBlurred(const std::string &str, Vector2 offset, float scale, Vector4 color) { SPADES_MARK_FUNCTION(); auto lines = SplitIntoLines(str); float x = 0.f; float y = 0.f; color = Vector4(color.x * color.w, color.y * color.w, color.z * color.w, color.w); renderer->SetColorAlphaPremultiplied(color); SplitTextIntoGlyphs( str, [&](Glyph &g) { RenderBlurGlyph(g); auto &img = *g.blurImage; auto srcBounds = img.bounds; auto target = offset + (Vector2(x, y) + img.offset) * scale; target = (target + .5f).Floor(); // for sharper rendering AABB2 destBounds(target.x, target.y, srcBounds.GetWidth() * scale, srcBounds.GetHeight() * scale); if (!rendererIsLowQuality) { srcBounds = srcBounds.Inflate(.5f); destBounds = destBounds.Inflate(.5f * scale); } renderer->DrawImage(&img.img, destBounds, srcBounds); x += g.advance.x; y += g.advance.y; }, [&](uint32_t codepoint) { DrawFallback(codepoint, offset + Vector2(x, y) * scale, height * scale, color); x += MeasureFallback(codepoint, height); }, [&]() { x = 0.f; y += lineHeight; }); } void FTFont::DrawShadow(const std::string &text, const Vector2 &offset, float scale, const Vector4 &color, const Vector4 &shadowColor) { DrawBlurred(text, offset, scale, shadowColor); Draw(text, offset, scale, color); } } // namespace ngclient } // namespace spades