#include #include "base/display.h" #include "base/logging.h" #include "base/stringutil.h" #include "thin3d/thin3d.h" #include "util/hash/hash.h" #include "util/text/wrap_text.h" #include "util/text/utf8.h" #include "gfx_es2/draw_text.h" #include "gfx_es2/draw_text_qt.h" #if defined(USING_QT_UI) #include #include #include #include TextDrawerQt::TextDrawerQt(Draw::DrawContext *draw) : TextDrawer(draw) { } TextDrawerQt::~TextDrawerQt() { ClearCache(); } uint32_t TextDrawerQt::SetFont(const char *fontName, int size, int flags) { uint32_t fontHash = fontName && strlen(fontName) ? hash::Adler32((const uint8_t *)fontName, strlen(fontName)) : 0; fontHash ^= size; fontHash ^= flags << 10; auto iter = fontMap_.find(fontHash); if (iter != fontMap_.end()) { fontHash_ = fontHash; return fontHash; } QFont *font = fontName ? new QFont(fontName) : new QFont(); font->setPixelSize((int)((size + 6) / dpiScale_)); fontMap_[fontHash] = font; fontHash_ = fontHash; return fontHash; } void TextDrawerQt::SetFont(uint32_t fontHandle) { uint32_t fontHash = fontHandle; auto iter = fontMap_.find(fontHash); if (iter != fontMap_.end()) { fontHash_ = fontHandle; } else { ELOG("Invalid font handle %08x", fontHandle); } } void TextDrawerQt::MeasureString(const char *str, size_t len, float *w, float *h) { CacheKey key{ std::string(str, len), fontHash_ }; TextMeasureEntry *entry; auto iter = sizeCache_.find(key); if (iter != sizeCache_.end()) { entry = iter->second.get(); } else { QFont* font = fontMap_.find(fontHash_)->second; QFontMetrics fm(*font); QSize size = fm.size(0, QString::fromUtf8(str, (int)len)); entry = new TextMeasureEntry(); entry->width = size.width(); entry->height = size.height(); sizeCache_[key] = std::unique_ptr(entry); } entry->lastUsedFrame = frameCount_; *w = entry->width * fontScaleX_ * dpiScale_; *h = entry->height * fontScaleY_ * dpiScale_; } void TextDrawerQt::MeasureStringRect(const char *str, size_t len, const Bounds &bounds, float *w, float *h, int align) { std::string toMeasure = std::string(str, len); int wrap = align & (FLAG_WRAP_TEXT | FLAG_ELLIPSIZE_TEXT); if (wrap) { bool rotated = (align & (ROTATE_90DEG_LEFT | ROTATE_90DEG_RIGHT)) != 0; WrapString(toMeasure, toMeasure.c_str(), rotated ? bounds.h : bounds.w, wrap); } QFont* font = fontMap_.find(fontHash_)->second; QFontMetrics fm(*font); QSize size = fm.size(0, QString::fromUtf8(toMeasure.c_str(), (int)toMeasure.size())); *w = (float)size.width() * fontScaleX_ * dpiScale_; *h = (float)size.height() * fontScaleY_ * dpiScale_; } void TextDrawerQt::DrawStringBitmap(std::vector &bitmapData, TextStringEntry &entry, Draw::DataFormat texFormat, const char *str, int align) { if (!strlen(str)) { bitmapData.clear(); return; } QFont *font = fontMap_.find(fontHash_)->second; QFontMetrics fm(*font); QSize size = fm.size(0, QString::fromUtf8(str)); QImage image((size.width() + 3) & ~3, (size.height() + 3) & ~3, QImage::Format_ARGB32_Premultiplied); if (image.isNull()) { bitmapData.clear(); return; } image.fill(0); QPainter painter; painter.begin(&image); painter.setFont(*font); painter.setPen(0xFFFFFFFF); // TODO: Involve ALIGN_HCENTER (bounds etc.) painter.drawText(image.rect(), Qt::AlignTop | Qt::AlignLeft, QString::fromUtf8(str).replace("&&", "&")); painter.end(); entry.texture = nullptr; entry.bmWidth = entry.width = image.width(); entry.bmHeight = entry.height = image.height(); entry.lastUsedFrame = frameCount_; if (texFormat == Draw::DataFormat::B4G4R4A4_UNORM_PACK16 || texFormat == Draw::DataFormat::R4G4B4A4_UNORM_PACK16) { bitmapData.resize(entry.bmWidth * entry.bmHeight * sizeof(uint16_t)); uint16_t *bitmapData16 = (uint16_t *)&bitmapData[0]; for (int x = 0; x < entry.bmWidth; x++) { for (int y = 0; y < entry.bmHeight; y++) { bitmapData16[entry.bmWidth * y + x] = 0xfff0 | (image.pixel(x, y) >> 28); } } } else if (texFormat == Draw::DataFormat::R8_UNORM) { bitmapData.resize(entry.bmWidth * entry.bmHeight); for (int x = 0; x < entry.bmWidth; x++) { for (int y = 0; y < entry.bmHeight; y++) { bitmapData[entry.bmWidth * y + x] = image.pixel(x, y) >> 24; } } } else { ELOG("Bad TextDrawer format"); assert(false); } } void TextDrawerQt::DrawString(DrawBuffer &target, const char *str, float x, float y, uint32_t color, int align) { using namespace Draw; if (!strlen(str)) return; CacheKey key{ std::string(str), fontHash_ }; target.Flush(true); TextStringEntry *entry; auto iter = cache_.find(key); if (iter != cache_.end()) { entry = iter->second.get(); entry->lastUsedFrame = frameCount_; } else { DataFormat texFormat = Draw::DataFormat::R4G4B4A4_UNORM_PACK16; entry = new TextStringEntry(); TextureDesc desc{}; std::vector bitmapData; DrawStringBitmap(bitmapData, *entry, texFormat, str, align); desc.initData.push_back(&bitmapData[0]); desc.type = TextureType::LINEAR2D; desc.format = texFormat; desc.width = entry->bmWidth; desc.height = entry->bmHeight; desc.depth = 1; desc.mipLevels = 1; desc.tag = "TextDrawer"; entry->texture = draw_->CreateTexture(desc); cache_[key] = std::unique_ptr(entry); } if (entry->texture) { draw_->BindTexture(0, entry->texture); } float w = entry->bmWidth * fontScaleX_ * dpiScale_; float h = entry->bmHeight * fontScaleY_ * dpiScale_; DrawBuffer::DoAlign(align, &x, &y, &w, &h); if (entry->texture) { target.DrawTexRect(x, y, x + w, y + h, 0.0f, 0.0f, 1.0f, 1.0f, color); target.Flush(true); } } void TextDrawerQt::ClearCache() { for (auto &iter : cache_) { if (iter.second->texture) iter.second->texture->Release(); } cache_.clear(); sizeCache_.clear(); // Also wipe the font map. for (auto iter : fontMap_) { delete iter.second; } fontMap_.clear(); fontHash_ = 0; } void TextDrawerQt::OncePerFrame() { frameCount_++; // If DPI changed (small-mode, future proper monitor DPI support), drop everything. float newDpiScale = CalculateDPIScale(); if (newDpiScale != dpiScale_) { dpiScale_ = newDpiScale; ClearCache(); } // Drop old strings. Use a prime number to reduce clashing with other rhythms if (frameCount_ % 23 == 0) { for (auto iter = cache_.begin(); iter != cache_.end();) { if (frameCount_ - iter->second->lastUsedFrame > 100) { if (iter->second->texture) iter->second->texture->Release(); cache_.erase(iter++); } else { iter++; } } for (auto iter = sizeCache_.begin(); iter != sizeCache_.end(); ) { if (frameCount_ - iter->second->lastUsedFrame > 100) { sizeCache_.erase(iter++); } else { iter++; } } } } #endif