#include "fontloader.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace { unsigned long utf8ToUnicode(std::string_view utf8) { if (utf8.empty()) return 0; size_t i = 0; unsigned long unicode; size_t numbytes; unsigned char ch = utf8[i++]; if (ch <= 0x7F) { unicode = ch; numbytes = 0; } else if (ch <= 0xBF) { throw std::logic_error("not a UTF-8 string"); } else if (ch <= 0xDF) { unicode = ch&0x1F; numbytes = 1; } else if (ch <= 0xEF) { unicode = ch&0x0F; numbytes = 2; } else if (ch <= 0xF7) { unicode = ch&0x07; numbytes = 3; } else { throw std::logic_error("not a UTF-8 string"); } for (size_t j = 0; j < numbytes; ++j) { ch = utf8[i++]; if (ch < 0x80 || ch > 0xBF) throw std::logic_error("not a UTF-8 string"); unicode <<= 6; unicode += ch & 0x3F; } if (unicode >= 0xD800 && unicode <= 0xDFFF) throw std::logic_error("not a UTF-8 string"); if (unicode > 0x10FFFF) throw std::logic_error("not a UTF-8 string"); return unicode; } /// This is a hack for Polish font unsigned char mapUtf8Char(unsigned char c) { switch(c){ case 0x80: return 0xc6; case 0x81: return 0x9c; case 0x82: return 0xe6; case 0x83: return 0xb3; case 0x84: return 0xf1; case 0x85: return 0xb9; case 0x86: return 0xbf; case 0x87: return 0x9f; case 0x88: return 0xea; case 0x89: return 0xea; case 0x8a: return 0x00; // not contained in win1250 case 0x8b: return 0x00; // not contained in win1250 case 0x8c: return 0x8f; case 0x8d: return 0xaf; case 0x8e: return 0xa5; case 0x8f: return 0x8c; case 0x90: return 0xca; case 0x93: return 0xa3; case 0x94: return 0xf6; case 0x95: return 0xf3; case 0x96: return 0xaf; case 0x97: return 0x8f; case 0x99: return 0xd3; case 0x9a: return 0xd1; case 0x9c: return 0x00; // not contained in win1250 case 0xa0: return 0xb9; case 0xa1: return 0xaf; case 0xa2: return 0xf3; case 0xa3: return 0xbf; case 0xa4: return 0x00; // not contained in win1250 case 0xe1: return 0x8c; case 0xe3: return 0x00; // not contained in win1250 case 0xf5: return 0x00; // not contained in win1250 default: return c; } } // getUnicode includes various hacks for dealing with Morrowind's .fnt files that are *mostly* // in the expected win12XX encoding, but also have randomly swapped characters sometimes. // Looks like the Morrowind developers found standard encodings too boring and threw in some twists for fun. unsigned long getUnicode(unsigned char c, ToUTF8::Utf8Encoder& encoder, ToUTF8::FromType encoding) { if (encoding == ToUTF8::WINDOWS_1250) // Hack for polish font { const std::array str {static_cast(mapUtf8Char(c)), '\0'}; return utf8ToUnicode(encoder.getUtf8(std::string_view(str.data(), 1))); } else { const std::array str {static_cast(c), '\0'}; return utf8ToUnicode(encoder.getUtf8(std::string_view(str.data(), 1))); } } [[noreturn]] void fail(std::istream& stream, const std::string& fileName, const std::string& message) { std::stringstream error; error << "Font loading error: " << message; error << "\n File: " << fileName; error << "\n Offset: 0x" << std::hex << stream.tellg(); throw std::runtime_error(error.str()); } } namespace Gui { FontLoader::FontLoader(ToUTF8::FromType encoding, const VFS::Manager* vfs, float scalingFactor) : mVFS(vfs) , mFontHeight(std::clamp(Settings::Manager::getInt("font size", "GUI"), 12, 20)) , mScalingFactor(scalingFactor) { if (encoding == ToUTF8::WINDOWS_1252) mEncoding = ToUTF8::CP437; else mEncoding = encoding; MyGUI::ResourceManager::getInstance().unregisterLoadXmlDelegate("Resource"); MyGUI::ResourceManager::getInstance().registerLoadXmlDelegate("Resource") = MyGUI::newDelegate(this, &FontLoader::loadFontFromXml); } FontLoader::~FontLoader() { try { MyGUI::ResourceManager::getInstance().unregisterLoadXmlDelegate("Resource"); } catch(const MyGUI::Exception& e) { Log(Debug::Error) << "Error in the FontLoader destructor: " << e.what(); } for (std::vector::iterator it = mTextures.begin(); it != mTextures.end(); ++it) delete *it; mTextures.clear(); for (std::vector::iterator it = mFonts.begin(); it != mFonts.end(); ++it) { try { MyGUI::ResourceManager::getInstance().removeByName((*it)->getResourceName()); } catch(const MyGUI::Exception& e) { Log(Debug::Error) << "Error in the destructor: " << e.what(); } } mFonts.clear(); } void FontLoader::loadBitmapFonts(bool exportToFile) { for (const auto& path : mVFS->getRecursiveDirectoryIterator("Fonts/")) { if (Misc::getFileExtension(path) == "fnt") loadBitmapFont(path, exportToFile); } } void FontLoader::loadTrueTypeFonts() { osgMyGUI::DataManager* dataManager = dynamic_cast(&osgMyGUI::DataManager::getInstance()); if (!dataManager) { Log(Debug::Error) << "Can not load TrueType fonts: osgMyGUI::DataManager is not available."; return; } std::string oldDataPath = dataManager->getDataPath(""); dataManager->setResourcePath("fonts"); for (const auto& path : mVFS->getRecursiveDirectoryIterator("Fonts/")) { if (Misc::getFileExtension(path) == "omwfont") MyGUI::ResourceManager::getInstance().load(std::string(Misc::getFileName(path))); } dataManager->setResourcePath(oldDataPath); } typedef struct { float x; float y; } Point; typedef struct { float u1; // appears unused, always 0 Point top_left; Point top_right; Point bottom_left; Point bottom_right; float width; float height; float u2; // appears unused, always 0 float kerning; float ascent; } GlyphInfo; void FontLoader::loadBitmapFont(const std::string &fileName, bool exportToFile) { Files::IStreamPtr file = mVFS->get(fileName); float fontSize; file->read((char*)&fontSize, sizeof(fontSize)); if (!file->good()) fail(*file, fileName, "File too small to be a valid font"); int one; file->read((char*)&one, sizeof(one)); if (!file->good()) fail(*file, fileName, "File too small to be a valid font"); if (one != 1) fail(*file, fileName, "Unexpected value"); file->read((char*)&one, sizeof(one)); if (!file->good()) fail(*file, fileName, "File too small to be a valid font"); if (one != 1) fail(*file, fileName, "Unexpected value"); char name_[284]; file->read(name_, sizeof(name_)); if (!file->good()) fail(*file, fileName, "File too small to be a valid font"); std::string name(name_); GlyphInfo data[256]; file->read((char*)data, sizeof(data)); if (!file->good()) fail(*file, fileName, "File too small to be a valid font"); file.reset(); // Create the font texture std::string bitmapFilename = "Fonts/" + std::string(name) + ".tex"; Files::IStreamPtr bitmapFile = mVFS->get(bitmapFilename); int width, height; bitmapFile->read((char*)&width, sizeof(int)); bitmapFile->read((char*)&height, sizeof(int)); if (!bitmapFile->good()) fail(*bitmapFile, bitmapFilename, "File too small to be a valid bitmap"); if (width <= 0 || height <= 0) fail(*bitmapFile, bitmapFilename, "Width and height must be positive"); std::vector textureData; textureData.resize(width*height*4); bitmapFile->read(&textureData[0], width*height*4); if (!bitmapFile->good()) fail(*bitmapFile, bitmapFilename, "File too small to be a valid bitmap"); bitmapFile.reset(); std::string resourceName = name; if (exportToFile) { osg::ref_ptr image = new osg::Image; image->allocateImage(width, height, 1, GL_RGBA, GL_UNSIGNED_BYTE); assert (image->isDataContiguous()); memcpy(image->data(), &textureData[0], textureData.size()); Log(Debug::Info) << "Writing " << resourceName + ".png"; osgDB::writeImageFile(*image, resourceName + ".png"); } // Register the font with MyGUI MyGUI::ResourceManualFont* font = static_cast( MyGUI::FactoryManager::getInstance().createObject("Resource", "ResourceManualFont")); mFonts.push_back(font); MyGUI::ITexture* tex = MyGUI::RenderManager::getInstance().createTexture(bitmapFilename); tex->createManual(width, height, MyGUI::TextureUsage::Write, MyGUI::PixelFormat::R8G8B8A8); unsigned char* texData = reinterpret_cast(tex->lock(MyGUI::TextureUsage::Write)); memcpy(texData, &textureData[0], textureData.size()); tex->unlock(); // Using ResourceManualFont::setTexture, enable for MyGUI 3.2.3 /* osg::ref_ptr texture = new osg::Texture2D; texture->setImage(image); osgMyGUI::OSGTexture* myguiTex = new osgMyGUI::OSGTexture(texture); mTextures.push_back(myguiTex); font->setTexture(myguiTex); */ // We need to emulate loading from XML because the data members are private as of mygui 3.2.0 MyGUI::xml::Document xmlDocument; MyGUI::xml::ElementPtr root = xmlDocument.createRoot("ResourceManualFont"); std::string baseName(Misc::stemFile(fileName)); root->addAttribute("name", getInternalFontName(baseName)); MyGUI::xml::ElementPtr defaultHeight = root->createChild("Property"); defaultHeight->addAttribute("key", "DefaultHeight"); defaultHeight->addAttribute("value", fontSize); MyGUI::xml::ElementPtr source = root->createChild("Property"); source->addAttribute("key", "Source"); source->addAttribute("value", std::string(bitmapFilename)); MyGUI::xml::ElementPtr codes = root->createChild("Codes"); for(int i = 0; i < 256; i++) { float x1 = data[i].top_left.x*width; float y1 = data[i].top_left.y*height; float w = data[i].top_right.x*width - x1; float h = data[i].bottom_left.y*height - y1; ToUTF8::Utf8Encoder encoder(mEncoding); unsigned long unicodeVal = getUnicode(i, encoder, mEncoding); MyGUI::xml::ElementPtr code = codes->createChild("Code"); code->addAttribute("index", unicodeVal); code->addAttribute("coord", MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " " + MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h)); code->addAttribute("advance", data[i].width); code->addAttribute("bearing", MyGUI::utility::toString(data[i].kerning) + " " + MyGUI::utility::toString((fontSize-data[i].ascent))); code->addAttribute("size", MyGUI::IntSize(static_cast(data[i].width), static_cast(data[i].height))); // Fall back from unavailable Windows-1252 encoding symbols to similar characters available in the game fonts std::multimap additional; // fallback glyph index, unicode additional.insert(std::make_pair(156, 0x00A2)); // cent sign additional.insert(std::make_pair(89, 0x00A5)); // yen sign additional.insert(std::make_pair(221, 0x00A6)); // broken bar additional.insert(std::make_pair(99, 0x00A9)); // copyright sign additional.insert(std::make_pair(97, 0x00AA)); // prima ordinal indicator additional.insert(std::make_pair(60, 0x00AB)); // double left-pointing angle quotation mark additional.insert(std::make_pair(45, 0x00AD)); // soft hyphen additional.insert(std::make_pair(114, 0x00AE)); // registered trademark symbol additional.insert(std::make_pair(45, 0x00AF)); // macron additional.insert(std::make_pair(241, 0x00B1)); // plus-minus sign additional.insert(std::make_pair(50, 0x00B2)); // superscript two additional.insert(std::make_pair(51, 0x00B3)); // superscript three additional.insert(std::make_pair(44, 0x00B8)); // cedilla additional.insert(std::make_pair(49, 0x00B9)); // superscript one additional.insert(std::make_pair(111, 0x00BA)); // primo ordinal indicator additional.insert(std::make_pair(62, 0x00BB)); // double right-pointing angle quotation mark additional.insert(std::make_pair(63, 0x00BF)); // inverted question mark additional.insert(std::make_pair(65, 0x00C6)); // latin capital ae ligature additional.insert(std::make_pair(79, 0x00D8)); // latin capital o with stroke additional.insert(std::make_pair(97, 0x00E6)); // latin small ae ligature additional.insert(std::make_pair(111, 0x00F8)); // latin small o with stroke additional.insert(std::make_pair(79, 0x0152)); // latin capital oe ligature additional.insert(std::make_pair(111, 0x0153)); // latin small oe ligature additional.insert(std::make_pair(83, 0x015A)); // latin capital s with caron additional.insert(std::make_pair(115, 0x015B)); // latin small s with caron additional.insert(std::make_pair(89, 0x0178)); // latin capital y with diaresis additional.insert(std::make_pair(90, 0x017D)); // latin capital z with caron additional.insert(std::make_pair(122, 0x017E)); // latin small z with caron additional.insert(std::make_pair(102, 0x0192)); // latin small f with hook additional.insert(std::make_pair(94, 0x02C6)); // circumflex modifier additional.insert(std::make_pair(126, 0x02DC)); // small tilde additional.insert(std::make_pair(69, 0x0401)); // cyrillic capital io (no diaeresis latin e is available) additional.insert(std::make_pair(137, 0x0451)); // cyrillic small io additional.insert(std::make_pair(45, 0x2012)); // figure dash additional.insert(std::make_pair(45, 0x2013)); // en dash additional.insert(std::make_pair(45, 0x2014)); // em dash additional.insert(std::make_pair(39, 0x2018)); // left single quotation mark additional.insert(std::make_pair(39, 0x2019)); // right single quotation mark additional.insert(std::make_pair(44, 0x201A)); // single low quotation mark additional.insert(std::make_pair(39, 0x201B)); // single high quotation mark (reversed) additional.insert(std::make_pair(34, 0x201C)); // left double quotation mark additional.insert(std::make_pair(34, 0x201D)); // right double quotation mark additional.insert(std::make_pair(44, 0x201E)); // double low quotation mark additional.insert(std::make_pair(34, 0x201F)); // double high quotation mark (reversed) additional.insert(std::make_pair(43, 0x2020)); // dagger additional.insert(std::make_pair(216, 0x2021)); // double dagger (note: this glyph is not available) additional.insert(std::make_pair(46, 0x2026)); // ellipsis additional.insert(std::make_pair(37, 0x2030)); // per mille sign additional.insert(std::make_pair(60, 0x2039)); // single left-pointing angle quotation mark additional.insert(std::make_pair(62, 0x203A)); // single right-pointing angle quotation mark additional.insert(std::make_pair(101, 0x20AC)); // euro sign additional.insert(std::make_pair(84, 0x2122)); // trademark sign additional.insert(std::make_pair(45, 0x2212)); // minus sign for (std::multimap::iterator it = additional.begin(); it != additional.end(); ++it) { if (it->first != i) continue; code = codes->createChild("Code"); code->addAttribute("index", it->second); code->addAttribute("coord", MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " " + MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h)); code->addAttribute("advance", data[i].width); code->addAttribute("bearing", MyGUI::utility::toString(data[i].kerning) + " " + MyGUI::utility::toString((fontSize-data[i].ascent))); code->addAttribute("size", MyGUI::IntSize(static_cast(data[i].width), static_cast(data[i].height))); } // ASCII vertical bar, use this as text input cursor if (i == 124) { MyGUI::xml::ElementPtr cursorCode = codes->createChild("Code"); cursorCode->addAttribute("index", MyGUI::FontCodeType::Cursor); cursorCode->addAttribute("coord", MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " " + MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h)); cursorCode->addAttribute("advance", data[i].width); cursorCode->addAttribute("bearing", MyGUI::utility::toString(data[i].kerning) + " " + MyGUI::utility::toString((fontSize-data[i].ascent))); cursorCode->addAttribute("size", MyGUI::IntSize(static_cast(data[i].width), static_cast(data[i].height))); } // Question mark, use for NotDefined marker (used for glyphs not existing in the font) if (i == 63) { MyGUI::xml::ElementPtr cursorCode = codes->createChild("Code"); cursorCode->addAttribute("index", MyGUI::FontCodeType::NotDefined); cursorCode->addAttribute("coord", MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " " + MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h)); cursorCode->addAttribute("advance", data[i].width); cursorCode->addAttribute("bearing", MyGUI::utility::toString(data[i].kerning) + " " + MyGUI::utility::toString((fontSize-data[i].ascent))); cursorCode->addAttribute("size", MyGUI::IntSize(static_cast(data[i].width), static_cast(data[i].height))); } } // These are required as well, but the fonts don't provide them for (int i=0; i<2; ++i) { MyGUI::FontCodeType::Enum type; if(i == 0) type = MyGUI::FontCodeType::Selected; else // if (i == 1) type = MyGUI::FontCodeType::SelectedBack; MyGUI::xml::ElementPtr cursorCode = codes->createChild("Code"); cursorCode->addAttribute("index", type); cursorCode->addAttribute("coord", "0 0 0 0"); cursorCode->addAttribute("advance", "0"); cursorCode->addAttribute("bearing", "0 0"); cursorCode->addAttribute("size", "0 0"); } if (exportToFile) { Log(Debug::Info) << "Writing " << resourceName + ".xml"; xmlDocument.createDeclaration(); xmlDocument.save(resourceName + ".xml"); } font->deserialization(root, MyGUI::Version(3,2,0)); MyGUI::ResourceManualFont* bookFont = static_cast( MyGUI::FactoryManager::getInstance().createObject("Resource", "ResourceManualFont")); mFonts.push_back(bookFont); bookFont->deserialization(root, MyGUI::Version(3,2,0)); bookFont->setResourceName("Journalbook " + getInternalFontName(baseName)); // Remove automatically registered fonts for (std::vector::iterator it = mFonts.begin(); it != mFonts.end();) { if ((*it)->getResourceName() == font->getResourceName()) { MyGUI::ResourceManager::getInstance().removeByName(font->getResourceName()); it = mFonts.erase(it); } else if ((*it)->getResourceName() == bookFont->getResourceName()) { MyGUI::ResourceManager::getInstance().removeByName(bookFont->getResourceName()); it = mFonts.erase(it); } else ++it; } MyGUI::ResourceManager::getInstance().addResource(font); MyGUI::ResourceManager::getInstance().addResource(bookFont); } void FontLoader::loadFontFromXml(MyGUI::xml::ElementPtr _node, const std::string& _file, MyGUI::Version _version) { MyGUI::xml::ElementEnumerator resourceNode = _node->getElementEnumerator(); bool createCopy = false; while (resourceNode.next("Resource")) { std::string type, name; resourceNode->findAttribute("type", type); resourceNode->findAttribute("name", name); if (name.empty()) continue; if (Misc::StringUtils::ciEqual(type, "ResourceTrueTypeFont")) { createCopy = true; // For TrueType fonts we should override Size and Resolution properties // to allow to configure font size via config file, without need to edit XML files. // Also we should take UI scaling factor in account. int resolution = Settings::Manager::getInt("ttf resolution", "GUI"); resolution = std::clamp(resolution, 48, 960) * mScalingFactor; MyGUI::xml::ElementPtr resolutionNode = resourceNode->createChild("Property"); resolutionNode->addAttribute("key", "Resolution"); resolutionNode->addAttribute("value", std::to_string(resolution)); MyGUI::xml::ElementPtr sizeNode = resourceNode->createChild("Property"); sizeNode->addAttribute("key", "Size"); sizeNode->addAttribute("value", std::to_string(mFontHeight)); resourceNode->setAttribute("name", getInternalFontName(name)); } else if (Misc::StringUtils::ciEqual(type, "ResourceSkin") || Misc::StringUtils::ciEqual(type, "AutoSizedResourceSkin")) { // We should adjust line height for MyGUI widgets depending on font size MyGUI::xml::ElementPtr heightNode = resourceNode->createChild("Property"); heightNode->addAttribute("key", "HeightLine"); heightNode->addAttribute("value", std::to_string(mFontHeight+2)); } } MyGUI::ResourceManager::getInstance().loadFromXmlNode(_node, _file, _version); if (createCopy) { std::unique_ptr copy{_node->createCopy()}; MyGUI::xml::ElementEnumerator copyFont = copy->getElementEnumerator(); while (copyFont.next("Resource")) { std::string type, name; copyFont->findAttribute("type", type); copyFont->findAttribute("name", name); if (name.empty()) continue; if (Misc::StringUtils::ciEqual(type, "ResourceTrueTypeFont")) { // Since the journal and books use the custom scaling factor depending on resolution, // setup separate fonts with different Resolution to fit these windows. // These fonts have an internal prefix. int resolution = Settings::Manager::getInt("ttf resolution", "GUI"); resolution = std::clamp(resolution, 48, 960); float currentX = Settings::Manager::getInt("resolution x", "Video"); float currentY = Settings::Manager::getInt("resolution y", "Video"); // TODO: read size from openmw_layout.xml somehow float heightScale = (currentY / 520); float widthScale = (currentX / 600); float uiScale = std::min(widthScale, heightScale); resolution *= uiScale; MyGUI::xml::ElementPtr resolutionNode = copyFont->createChild("Property"); resolutionNode->addAttribute("key", "Resolution"); resolutionNode->addAttribute("value", std::to_string(resolution)); copyFont->setAttribute("name", "Journalbook " + getInternalFontName(name)); } } MyGUI::ResourceManager::getInstance().loadFromXmlNode(copy.get(), _file, _version); } } int FontLoader::getFontHeight() { return mFontHeight; } std::string FontLoader::getInternalFontName(const std::string& name) { const std::string lowerName = Misc::StringUtils::lowerCase(name); if (lowerName == Misc::StringUtils::lowerCase(Fallback::Map::getString("Fonts_Font_0"))) return "DefaultFont"; if (lowerName == "dejavusansmono") return "MonoFont"; // We need to use a TrueType monospace font to display debug texts properly. if (lowerName == Misc::StringUtils::lowerCase(Fallback::Map::getString("Fonts_Font_2"))) return "ScrollFont"; return name; } std::string FontLoader::getFontForFace(const std::string& face) { const std::string lowerFace = Misc::StringUtils::lowerCase(face); if (lowerFace == "magic cards") return "DefaultFont"; if (lowerFace == "daedric") return "ScrollFont"; return face; } }