2013-06-06 20:13:30 +00:00
|
|
|
#include "fontloader.hpp"
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
#include <array>
|
2014-12-21 00:28:06 +00:00
|
|
|
#include <stdexcept>
|
2022-02-12 11:00:35 +00:00
|
|
|
#include <string_view>
|
2014-12-21 00:28:06 +00:00
|
|
|
|
2015-04-30 23:41:33 +00:00
|
|
|
#include <osg/Image>
|
|
|
|
|
|
|
|
#include <osgDB/WriteFile>
|
2013-06-06 20:13:30 +00:00
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
#include <MyGUI_FactoryManager.h>
|
|
|
|
#include <MyGUI_RenderManager.h>
|
2013-06-06 20:13:30 +00:00
|
|
|
#include <MyGUI_ResourceManager.h>
|
|
|
|
#include <MyGUI_ResourceManualFont.h>
|
2022-08-20 11:07:23 +00:00
|
|
|
#include <MyGUI_ResourceTrueTypeFont.h>
|
2013-06-06 20:13:30 +00:00
|
|
|
#include <MyGUI_XmlDocument.h>
|
|
|
|
|
2018-08-14 15:42:41 +00:00
|
|
|
#include <components/debug/debuglog.hpp>
|
|
|
|
|
2022-07-12 13:27:59 +00:00
|
|
|
#include <components/fallback/fallback.hpp>
|
|
|
|
|
2015-04-30 23:41:33 +00:00
|
|
|
#include <components/vfs/manager.hpp>
|
2013-09-10 20:11:57 +00:00
|
|
|
|
2022-08-02 22:00:54 +00:00
|
|
|
#include <components/misc/strings/algorithm.hpp>
|
2013-06-06 20:13:30 +00:00
|
|
|
|
2022-08-23 05:35:13 +00:00
|
|
|
#include <components/myguiplatform/scalinglayer.hpp>
|
2015-04-30 23:41:33 +00:00
|
|
|
|
2023-07-16 18:46:54 +00:00
|
|
|
#include <components/settings/values.hpp>
|
2020-05-28 19:09:10 +00:00
|
|
|
|
2013-06-06 20:13:30 +00:00
|
|
|
namespace
|
|
|
|
{
|
2023-05-21 14:39:32 +00:00
|
|
|
MyGUI::xml::ElementPtr getProperty(MyGUI::xml::ElementPtr resourceNode, std::string_view propertyName)
|
2022-08-23 05:35:13 +00:00
|
|
|
{
|
|
|
|
MyGUI::xml::ElementPtr propertyNode = nullptr;
|
|
|
|
MyGUI::xml::ElementEnumerator propertyIterator = resourceNode->getElementEnumerator();
|
|
|
|
while (propertyIterator.next("Property"))
|
|
|
|
{
|
2023-05-21 14:39:32 +00:00
|
|
|
if (propertyIterator->findAttribute("key") == propertyName)
|
2022-08-23 05:35:13 +00:00
|
|
|
{
|
|
|
|
propertyNode = propertyIterator.current();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return propertyNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
MyGUI::IntSize getBookSize(MyGUI::IDataStream* layersStream)
|
|
|
|
{
|
|
|
|
MyGUI::xml::Document xmlDocument;
|
|
|
|
xmlDocument.open(layersStream);
|
|
|
|
MyGUI::xml::ElementPtr root = xmlDocument.getRoot();
|
|
|
|
MyGUI::xml::ElementEnumerator layersIterator = root->getElementEnumerator();
|
|
|
|
while (layersIterator.next("Layer"))
|
|
|
|
{
|
2023-05-21 14:39:32 +00:00
|
|
|
if (layersIterator->findAttribute("name") == "JournalBooks")
|
2022-08-23 05:35:13 +00:00
|
|
|
{
|
|
|
|
MyGUI::xml::ElementPtr sizeProperty = getProperty(layersIterator.current(), "Size");
|
2023-05-21 14:39:32 +00:00
|
|
|
if (sizeProperty != nullptr)
|
|
|
|
{
|
2023-08-07 10:43:30 +00:00
|
|
|
auto sizeValue = sizeProperty->findAttribute("value");
|
2023-05-21 14:39:32 +00:00
|
|
|
if (!sizeValue.empty())
|
|
|
|
return MyGUI::IntSize::parse(sizeValue);
|
|
|
|
}
|
2022-08-23 05:35:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return MyGUI::RenderManager::getInstance().getViewSize();
|
|
|
|
}
|
|
|
|
|
2022-02-12 11:00:35 +00:00
|
|
|
unsigned long utf8ToUnicode(std::string_view utf8)
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
2022-02-14 18:56:50 +00:00
|
|
|
if (utf8.empty())
|
|
|
|
return 0;
|
2013-06-06 20:13:30 +00:00
|
|
|
size_t i = 0;
|
|
|
|
unsigned long unicode;
|
2015-03-11 23:21:46 +00:00
|
|
|
size_t numbytes;
|
2013-06-06 20:13:30 +00:00
|
|
|
unsigned char ch = utf8[i++];
|
|
|
|
if (ch <= 0x7F)
|
|
|
|
{
|
|
|
|
unicode = ch;
|
2015-03-11 23:21:46 +00:00
|
|
|
numbytes = 0;
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
else if (ch <= 0xBF)
|
|
|
|
{
|
|
|
|
throw std::logic_error("not a UTF-8 string");
|
|
|
|
}
|
|
|
|
else if (ch <= 0xDF)
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
unicode = ch & 0x1F;
|
2015-03-11 23:21:46 +00:00
|
|
|
numbytes = 1;
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
else if (ch <= 0xEF)
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
unicode = ch & 0x0F;
|
2015-03-11 23:21:46 +00:00
|
|
|
numbytes = 2;
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
else if (ch <= 0xF7)
|
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
unicode = ch & 0x07;
|
2015-03-11 23:21:46 +00:00
|
|
|
numbytes = 3;
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
throw std::logic_error("not a UTF-8 string");
|
|
|
|
}
|
2015-03-11 23:21:46 +00:00
|
|
|
for (size_t j = 0; j < numbytes; ++j)
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
2016-08-29 10:38:24 +00:00
|
|
|
ch = utf8[i++];
|
2013-06-06 20:13:30 +00:00
|
|
|
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;
|
|
|
|
}
|
2013-09-10 20:11:57 +00:00
|
|
|
|
2021-05-21 10:56:46 +00:00
|
|
|
/// This is a hack for Polish font
|
2022-09-22 18:26:05 +00:00
|
|
|
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;
|
2021-05-21 10:56:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-12 11:00:35 +00:00
|
|
|
// getUnicode includes various hacks for dealing with Morrowind's .fnt files that are *mostly*
|
2014-06-05 14:16:16 +00:00
|
|
|
// 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.
|
2022-02-12 11:00:35 +00:00
|
|
|
unsigned long getUnicode(unsigned char c, ToUTF8::Utf8Encoder& encoder, ToUTF8::FromType encoding)
|
2013-09-10 20:11:57 +00:00
|
|
|
{
|
2021-05-21 10:56:46 +00:00
|
|
|
if (encoding == ToUTF8::WINDOWS_1250) // Hack for polish font
|
2022-02-12 11:00:35 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::array<char, 2> str{ static_cast<char>(mapUtf8Char(c)), '\0' };
|
2022-02-12 11:00:35 +00:00
|
|
|
return utf8ToUnicode(encoder.getUtf8(std::string_view(str.data(), 1)));
|
|
|
|
}
|
2013-09-10 20:11:57 +00:00
|
|
|
else
|
2022-02-12 11:00:35 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
const std::array<char, 2> str{ static_cast<char>(c), '\0' };
|
2022-02-12 11:00:35 +00:00
|
|
|
return utf8ToUnicode(encoder.getUtf8(std::string_view(str.data(), 1)));
|
|
|
|
}
|
2013-09-10 20:11:57 +00:00
|
|
|
}
|
|
|
|
|
2023-05-21 14:39:32 +00:00
|
|
|
[[noreturn]] void fail(std::istream& stream, std::string_view fileName, std::string_view message)
|
2014-12-21 00:28:06 +00:00
|
|
|
{
|
|
|
|
std::stringstream error;
|
|
|
|
error << "Font loading error: " << message;
|
|
|
|
error << "\n File: " << fileName;
|
2022-04-15 00:15:39 +00:00
|
|
|
error << "\n Offset: 0x" << std::hex << stream.tellg();
|
2014-12-21 00:28:06 +00:00
|
|
|
throw std::runtime_error(error.str());
|
|
|
|
}
|
|
|
|
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
|
2014-09-12 02:27:23 +00:00
|
|
|
namespace Gui
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
|
|
|
|
2022-07-09 15:58:40 +00:00
|
|
|
FontLoader::FontLoader(ToUTF8::FromType encoding, const VFS::Manager* vfs, float scalingFactor)
|
2015-04-30 23:41:33 +00:00
|
|
|
: mVFS(vfs)
|
2021-04-15 11:14:41 +00:00
|
|
|
, mScalingFactor(scalingFactor)
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
|
|
|
if (encoding == ToUTF8::WINDOWS_1252)
|
|
|
|
mEncoding = ToUTF8::CP437;
|
|
|
|
else
|
|
|
|
mEncoding = encoding;
|
2020-05-28 19:09:10 +00:00
|
|
|
|
|
|
|
MyGUI::ResourceManager::getInstance().unregisterLoadXmlDelegate("Resource");
|
2022-09-22 18:26:05 +00:00
|
|
|
MyGUI::ResourceManager::getInstance().registerLoadXmlDelegate("Resource")
|
|
|
|
= MyGUI::newDelegate(this, &FontLoader::overrideLineHeight);
|
2022-08-23 05:35:13 +00:00
|
|
|
|
|
|
|
loadFonts();
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
|
2022-08-20 11:07:23 +00:00
|
|
|
void FontLoader::loadFonts()
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
std::string defaultFont{ Fallback::Map::getString("Fonts_Font_0") };
|
|
|
|
std::string scrollFont{ Fallback::Map::getString("Fonts_Font_2") };
|
2022-08-20 11:07:23 +00:00
|
|
|
loadFont(defaultFont, "DefaultFont");
|
|
|
|
loadFont(scrollFont, "ScrollFont");
|
2022-09-22 18:26:05 +00:00
|
|
|
loadFont("DejaVuLGCSansMono",
|
|
|
|
"MonoFont"); // We need to use a TrueType monospace font to display debug texts properly.
|
2022-08-20 11:07:23 +00:00
|
|
|
|
|
|
|
// Use our TrueType fonts as a fallback.
|
2022-09-22 18:26:05 +00:00
|
|
|
if (!MyGUI::ResourceManager::getInstance().isExist("DefaultFont")
|
|
|
|
&& !Misc::StringUtils::ciEqual(defaultFont, "MysticCards"))
|
2022-08-30 07:54:20 +00:00
|
|
|
loadFont("MysticCards", "DefaultFont");
|
2022-09-22 18:26:05 +00:00
|
|
|
if (!MyGUI::ResourceManager::getInstance().isExist("ScrollFont")
|
|
|
|
&& !Misc::StringUtils::ciEqual(scrollFont, "DemonicLetters"))
|
2022-08-30 07:54:20 +00:00
|
|
|
loadFont("DemonicLetters", "ScrollFont");
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
|
2022-08-20 11:07:23 +00:00
|
|
|
void FontLoader::loadFont(const std::string& fileName, const std::string& fontId)
|
2018-09-04 05:15:07 +00:00
|
|
|
{
|
2022-08-20 11:07:23 +00:00
|
|
|
if (mVFS->exists("fonts/" + fileName + ".fnt"))
|
|
|
|
loadBitmapFont(fileName + ".fnt", fontId);
|
|
|
|
else if (mVFS->exists("fonts/" + fileName + ".omwfont"))
|
|
|
|
loadTrueTypeFont(fileName + ".omwfont", fontId);
|
|
|
|
else
|
|
|
|
Log(Debug::Error) << "Font '" << fileName << "' is not found.";
|
|
|
|
}
|
|
|
|
|
|
|
|
void FontLoader::loadTrueTypeFont(const std::string& fileName, const std::string& fontId)
|
|
|
|
{
|
|
|
|
Log(Debug::Info) << "Loading font file " << fileName;
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
osgMyGUI::DataManager* dataManager
|
|
|
|
= dynamic_cast<osgMyGUI::DataManager*>(&osgMyGUI::DataManager::getInstance());
|
2018-09-04 05:15:07 +00:00
|
|
|
if (!dataManager)
|
|
|
|
{
|
2022-08-20 11:07:23 +00:00
|
|
|
Log(Debug::Error) << "Can not load TrueType font " << fontId << ": osgMyGUI::DataManager is not available.";
|
2018-09-04 05:15:07 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-08-23 05:35:13 +00:00
|
|
|
// TODO: it may be worth to take in account resolution change, but it is not safe to replace used assets
|
|
|
|
std::unique_ptr<MyGUI::IDataStream> layersStream(dataManager->getData("openmw_layers.xml"));
|
|
|
|
MyGUI::IntSize bookSize = getBookSize(layersStream.get());
|
|
|
|
float bookScale = osgMyGUI::ScalingLayer::getScaleFactor(bookSize);
|
|
|
|
|
2023-05-21 14:39:32 +00:00
|
|
|
const auto oldDataPath = dataManager->getDataPath({});
|
2022-07-09 16:42:18 +00:00
|
|
|
dataManager->setResourcePath("fonts");
|
2022-08-20 11:07:23 +00:00
|
|
|
std::unique_ptr<MyGUI::IDataStream> dataStream(dataManager->getData(fileName));
|
|
|
|
|
|
|
|
MyGUI::xml::Document xmlDocument;
|
|
|
|
xmlDocument.open(dataStream.get());
|
|
|
|
MyGUI::xml::ElementPtr root = xmlDocument.getRoot();
|
|
|
|
|
|
|
|
MyGUI::xml::ElementEnumerator resourceNode = root->getElementEnumerator();
|
|
|
|
bool valid = false;
|
|
|
|
if (resourceNode.next("Resource"))
|
|
|
|
{
|
2023-05-21 14:39:32 +00:00
|
|
|
valid = resourceNode->findAttribute("type") == "ResourceTrueTypeFont";
|
2022-08-20 11:07:23 +00:00
|
|
|
}
|
2022-07-09 16:42:18 +00:00
|
|
|
|
2022-08-20 11:07:23 +00:00
|
|
|
if (valid == false)
|
2022-07-07 15:34:18 +00:00
|
|
|
{
|
2022-08-20 11:07:23 +00:00
|
|
|
dataManager->setResourcePath(oldDataPath);
|
|
|
|
Log(Debug::Error) << "Can not load TrueType font " << fontId << ": " << fileName << " is invalid.";
|
|
|
|
return;
|
2022-07-07 15:34:18 +00:00
|
|
|
}
|
2022-07-09 16:42:18 +00:00
|
|
|
|
2022-08-23 05:35:13 +00:00
|
|
|
int resolution = 70;
|
|
|
|
MyGUI::xml::ElementPtr resolutionNode = getProperty(resourceNode.current(), "Resolution");
|
|
|
|
if (resolutionNode == nullptr)
|
|
|
|
{
|
|
|
|
resolutionNode = resourceNode->createChild("Property");
|
|
|
|
resolutionNode->addAttribute("key", "Resolution");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
resolution = MyGUI::utility::parseInt(resolutionNode->findAttribute("value"));
|
2022-08-20 11:07:23 +00:00
|
|
|
|
2022-08-23 05:35:13 +00:00
|
|
|
resolutionNode->setAttribute("value", MyGUI::utility::toString(resolution * std::ceil(mScalingFactor)));
|
2022-08-20 11:07:23 +00:00
|
|
|
|
|
|
|
MyGUI::xml::ElementPtr sizeNode = resourceNode->createChild("Property");
|
|
|
|
sizeNode->addAttribute("key", "Size");
|
2023-07-16 18:46:54 +00:00
|
|
|
sizeNode->addAttribute("value", std::to_string(Settings::gui().mFontSize));
|
2022-08-20 11:07:23 +00:00
|
|
|
|
|
|
|
MyGUI::ResourceTrueTypeFont* font = static_cast<MyGUI::ResourceTrueTypeFont*>(
|
2022-09-22 18:26:05 +00:00
|
|
|
MyGUI::FactoryManager::getInstance().createObject("Resource", "ResourceTrueTypeFont"));
|
|
|
|
font->deserialization(resourceNode.current(), MyGUI::Version(3, 2, 0));
|
2022-08-20 11:07:23 +00:00
|
|
|
font->setResourceName(fontId);
|
|
|
|
MyGUI::ResourceManager::getInstance().addResource(font);
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
resolutionNode->setAttribute(
|
|
|
|
"value", MyGUI::utility::toString(static_cast<int>(resolution * bookScale * mScalingFactor)));
|
2022-08-20 11:07:23 +00:00
|
|
|
|
|
|
|
MyGUI::ResourceTrueTypeFont* bookFont = static_cast<MyGUI::ResourceTrueTypeFont*>(
|
2022-09-22 18:26:05 +00:00
|
|
|
MyGUI::FactoryManager::getInstance().createObject("Resource", "ResourceTrueTypeFont"));
|
|
|
|
bookFont->deserialization(resourceNode.current(), MyGUI::Version(3, 2, 0));
|
2022-08-20 11:07:23 +00:00
|
|
|
bookFont->setResourceName("Journalbook " + fontId);
|
|
|
|
MyGUI::ResourceManager::getInstance().addResource(bookFont);
|
|
|
|
|
2022-07-09 16:42:18 +00:00
|
|
|
dataManager->setResourcePath(oldDataPath);
|
2022-08-20 11:07:23 +00:00
|
|
|
|
|
|
|
if (resourceNode.next("Resource"))
|
2022-09-22 18:26:05 +00:00
|
|
|
Log(Debug::Warning) << "Font file " << fileName
|
|
|
|
<< " contains multiple Resource entries, only first one will be used.";
|
2022-07-07 15:34:18 +00:00
|
|
|
}
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
void FontLoader::loadBitmapFont(const std::string& fileName, const std::string& fontId)
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
2022-08-20 11:07:23 +00:00
|
|
|
Log(Debug::Info) << "Loading font file " << fileName;
|
|
|
|
|
|
|
|
Files::IStreamPtr file = mVFS->get("fonts/" + fileName);
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
float fontSize;
|
2015-04-30 23:41:33 +00:00
|
|
|
file->read((char*)&fontSize, sizeof(fontSize));
|
|
|
|
if (!file->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "File too small to be a valid font");
|
2014-12-21 00:28:06 +00:00
|
|
|
|
2013-06-06 20:13:30 +00:00
|
|
|
int one;
|
2015-04-30 23:41:33 +00:00
|
|
|
file->read((char*)&one, sizeof(one));
|
|
|
|
if (!file->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "File too small to be a valid font");
|
2013-06-06 20:13:30 +00:00
|
|
|
|
2014-12-21 00:28:06 +00:00
|
|
|
if (one != 1)
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "Unexpected value");
|
2014-12-21 00:28:06 +00:00
|
|
|
|
2015-04-30 23:41:33 +00:00
|
|
|
file->read((char*)&one, sizeof(one));
|
|
|
|
if (!file->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "File too small to be a valid font");
|
2014-12-21 00:28:06 +00:00
|
|
|
|
|
|
|
if (one != 1)
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "Unexpected value");
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
char name_[284];
|
2015-04-30 23:41:33 +00:00
|
|
|
file->read(name_, sizeof(name_));
|
|
|
|
if (!file->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "File too small to be a valid font");
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
GlyphInfo data[256];
|
2015-04-30 23:41:33 +00:00
|
|
|
file->read((char*)data, sizeof(data));
|
|
|
|
if (!file->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*file, fileName, "File too small to be a valid font");
|
2015-04-30 23:41:33 +00:00
|
|
|
|
|
|
|
file.reset();
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
// Create the font texture
|
2023-08-02 13:59:40 +00:00
|
|
|
std::string bitmapFilename = "fonts/" + std::string(name_) + ".tex";
|
2015-04-30 23:41:33 +00:00
|
|
|
|
|
|
|
Files::IStreamPtr bitmapFile = mVFS->get(bitmapFilename);
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
int width, height;
|
2015-04-30 23:41:33 +00:00
|
|
|
bitmapFile->read((char*)&width, sizeof(int));
|
|
|
|
bitmapFile->read((char*)&height, sizeof(int));
|
2014-12-21 00:28:06 +00:00
|
|
|
|
2015-04-30 23:41:33 +00:00
|
|
|
if (!bitmapFile->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*bitmapFile, bitmapFilename, "File too small to be a valid bitmap");
|
2014-12-21 00:28:06 +00:00
|
|
|
|
|
|
|
if (width <= 0 || height <= 0)
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*bitmapFile, bitmapFilename, "Width and height must be positive");
|
2013-06-06 20:13:30 +00:00
|
|
|
|
2015-04-30 23:41:33 +00:00
|
|
|
std::vector<char> textureData;
|
2022-09-22 18:26:05 +00:00
|
|
|
textureData.resize(width * height * 4);
|
|
|
|
bitmapFile->read(textureData.data(), width * height * 4);
|
2015-04-30 23:41:33 +00:00
|
|
|
if (!bitmapFile->good())
|
2022-04-15 00:15:39 +00:00
|
|
|
fail(*bitmapFile, bitmapFilename, "File too small to be a valid bitmap");
|
2015-04-30 23:41:33 +00:00
|
|
|
bitmapFile.reset();
|
2013-06-06 20:13:30 +00:00
|
|
|
|
2015-06-03 16:00:45 +00:00
|
|
|
MyGUI::ITexture* tex = MyGUI::RenderManager::getInstance().createTexture(bitmapFilename);
|
|
|
|
tex->createManual(width, height, MyGUI::TextureUsage::Write, MyGUI::PixelFormat::R8G8B8A8);
|
|
|
|
unsigned char* texData = reinterpret_cast<unsigned char*>(tex->lock(MyGUI::TextureUsage::Write));
|
2022-09-03 15:41:35 +00:00
|
|
|
memcpy(texData, textureData.data(), textureData.size());
|
2015-06-03 16:00:45 +00:00
|
|
|
tex->unlock();
|
|
|
|
|
2013-06-06 20:13:30 +00:00
|
|
|
// 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");
|
2022-07-07 19:20:22 +00:00
|
|
|
|
2022-08-20 11:07:23 +00:00
|
|
|
root->addAttribute("name", fontId);
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
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");
|
2023-05-21 14:39:32 +00:00
|
|
|
source->addAttribute("value", bitmapFilename);
|
2013-06-06 20:13:30 +00:00
|
|
|
MyGUI::xml::ElementPtr codes = root->createChild("Codes");
|
|
|
|
|
2024-03-27 08:44:35 +00:00
|
|
|
// Fall back from unavailable Windows-1252 encoding symbols to similar characters available in the game
|
|
|
|
// fonts
|
|
|
|
std::multimap<int, int> additional; // fallback glyph index, unicode
|
|
|
|
additional.emplace(156, 0x00A2); // cent sign
|
|
|
|
additional.emplace(89, 0x00A5); // yen sign
|
|
|
|
additional.emplace(221, 0x00A6); // broken bar
|
|
|
|
additional.emplace(99, 0x00A9); // copyright sign
|
|
|
|
additional.emplace(97, 0x00AA); // prima ordinal indicator
|
|
|
|
additional.emplace(60, 0x00AB); // double left-pointing angle quotation mark
|
|
|
|
additional.emplace(45, 0x00AD); // soft hyphen
|
|
|
|
additional.emplace(114, 0x00AE); // registered trademark symbol
|
|
|
|
additional.emplace(45, 0x00AF); // macron
|
|
|
|
additional.emplace(241, 0x00B1); // plus-minus sign
|
|
|
|
additional.emplace(50, 0x00B2); // superscript two
|
|
|
|
additional.emplace(51, 0x00B3); // superscript three
|
|
|
|
additional.emplace(44, 0x00B8); // cedilla
|
|
|
|
additional.emplace(49, 0x00B9); // superscript one
|
|
|
|
additional.emplace(111, 0x00BA); // primo ordinal indicator
|
|
|
|
additional.emplace(62, 0x00BB); // double right-pointing angle quotation mark
|
|
|
|
additional.emplace(63, 0x00BF); // inverted question mark
|
|
|
|
additional.emplace(65, 0x00C6); // latin capital ae ligature
|
|
|
|
additional.emplace(79, 0x00D8); // latin capital o with stroke
|
|
|
|
additional.emplace(97, 0x00E6); // latin small ae ligature
|
|
|
|
additional.emplace(111, 0x00F8); // latin small o with stroke
|
|
|
|
additional.emplace(79, 0x0152); // latin capital oe ligature
|
|
|
|
additional.emplace(111, 0x0153); // latin small oe ligature
|
|
|
|
additional.emplace(83, 0x015A); // latin capital s with caron
|
|
|
|
additional.emplace(115, 0x015B); // latin small s with caron
|
|
|
|
additional.emplace(89, 0x0178); // latin capital y with diaresis
|
|
|
|
additional.emplace(90, 0x017D); // latin capital z with caron
|
|
|
|
additional.emplace(122, 0x017E); // latin small z with caron
|
|
|
|
additional.emplace(102, 0x0192); // latin small f with hook
|
|
|
|
additional.emplace(94, 0x02C6); // circumflex modifier
|
|
|
|
additional.emplace(126, 0x02DC); // small tilde
|
|
|
|
additional.emplace(69, 0x0401); // cyrillic capital io (no diaeresis latin e is available)
|
|
|
|
additional.emplace(137, 0x0451); // cyrillic small io
|
|
|
|
additional.emplace(45, 0x2012); // figure dash
|
|
|
|
additional.emplace(45, 0x2013); // en dash
|
|
|
|
additional.emplace(45, 0x2014); // em dash
|
|
|
|
additional.emplace(39, 0x2018); // left single quotation mark
|
|
|
|
additional.emplace(39, 0x2019); // right single quotation mark
|
|
|
|
additional.emplace(44, 0x201A); // single low quotation mark
|
|
|
|
additional.emplace(39, 0x201B); // single high quotation mark (reversed)
|
|
|
|
additional.emplace(34, 0x201C); // left double quotation mark
|
|
|
|
additional.emplace(34, 0x201D); // right double quotation mark
|
|
|
|
additional.emplace(44, 0x201E); // double low quotation mark
|
|
|
|
additional.emplace(34, 0x201F); // double high quotation mark (reversed)
|
|
|
|
additional.emplace(43, 0x2020); // dagger
|
|
|
|
additional.emplace(216, 0x2021); // double dagger (note: this glyph is not available)
|
|
|
|
additional.emplace(46, 0x2026); // ellipsis
|
|
|
|
additional.emplace(37, 0x2030); // per mille sign
|
|
|
|
additional.emplace(60, 0x2039); // single left-pointing angle quotation mark
|
|
|
|
additional.emplace(62, 0x203A); // single right-pointing angle quotation mark
|
|
|
|
additional.emplace(101, 0x20AC); // euro sign
|
|
|
|
additional.emplace(84, 0x2122); // trademark sign
|
|
|
|
additional.emplace(45, 0x2212); // minus sign
|
|
|
|
|
2022-09-22 18:26:05 +00:00
|
|
|
for (int i = 0; i < 256; i++)
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
2022-09-22 18:26:05 +00:00
|
|
|
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;
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
ToUTF8::Utf8Encoder encoder(mEncoding);
|
2022-02-12 11:00:35 +00:00
|
|
|
unsigned long unicodeVal = getUnicode(i, encoder, mEncoding);
|
2013-06-06 20:13:30 +00:00
|
|
|
|
|
|
|
MyGUI::xml::ElementPtr code = codes->createChild("Code");
|
|
|
|
code->addAttribute("index", unicodeVal);
|
2022-09-22 18:26:05 +00:00
|
|
|
code->addAttribute("coord",
|
|
|
|
MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " " + MyGUI::utility::toString(w)
|
|
|
|
+ " " + MyGUI::utility::toString(h));
|
2013-06-06 20:13:30 +00:00
|
|
|
code->addAttribute("advance", data[i].width);
|
2022-09-22 18:26:05 +00:00
|
|
|
code->addAttribute("bearing",
|
|
|
|
MyGUI::utility::toString(data[i].kerning) + " "
|
|
|
|
+ MyGUI::utility::toString((fontSize - data[i].ascent)));
|
|
|
|
code->addAttribute(
|
|
|
|
"size", MyGUI::IntSize(static_cast<int>(data[i].width), static_cast<int>(data[i].height)));
|
|
|
|
|
2024-03-27 08:44:35 +00:00
|
|
|
for (auto [it, end] = additional.equal_range(i); it != end; ++it)
|
2014-06-05 14:16:16 +00:00
|
|
|
{
|
2018-03-30 21:51:43 +00:00
|
|
|
code = codes->createChild("Code");
|
2024-03-27 08:44:35 +00:00
|
|
|
code->addAttribute("index", it->second);
|
2022-09-22 18:26:05 +00:00
|
|
|
code->addAttribute("coord",
|
|
|
|
MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " "
|
|
|
|
+ MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h));
|
2018-03-30 21:51:43 +00:00
|
|
|
code->addAttribute("advance", data[i].width);
|
2022-09-22 18:26:05 +00:00
|
|
|
code->addAttribute("bearing",
|
|
|
|
MyGUI::utility::toString(data[i].kerning) + " "
|
|
|
|
+ MyGUI::utility::toString((fontSize - data[i].ascent)));
|
|
|
|
code->addAttribute(
|
|
|
|
"size", MyGUI::IntSize(static_cast<int>(data[i].width), static_cast<int>(data[i].height)));
|
2014-06-05 14:16:16 +00:00
|
|
|
}
|
|
|
|
|
2013-06-06 20:13:30 +00:00
|
|
|
// 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);
|
2022-09-22 18:26:05 +00:00
|
|
|
cursorCode->addAttribute("coord",
|
|
|
|
MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " "
|
|
|
|
+ MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h));
|
2013-06-06 20:13:30 +00:00
|
|
|
cursorCode->addAttribute("advance", data[i].width);
|
2022-09-22 18:26:05 +00:00
|
|
|
cursorCode->addAttribute("bearing",
|
|
|
|
MyGUI::utility::toString(data[i].kerning) + " "
|
|
|
|
+ MyGUI::utility::toString((fontSize - data[i].ascent)));
|
|
|
|
cursorCode->addAttribute(
|
|
|
|
"size", MyGUI::IntSize(static_cast<int>(data[i].width), static_cast<int>(data[i].height)));
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
2014-06-05 14:15:47 +00:00
|
|
|
|
|
|
|
// 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);
|
2022-09-22 18:26:05 +00:00
|
|
|
cursorCode->addAttribute("coord",
|
|
|
|
MyGUI::utility::toString(x1) + " " + MyGUI::utility::toString(y1) + " "
|
|
|
|
+ MyGUI::utility::toString(w) + " " + MyGUI::utility::toString(h));
|
2014-06-05 14:15:47 +00:00
|
|
|
cursorCode->addAttribute("advance", data[i].width);
|
2022-09-22 18:26:05 +00:00
|
|
|
cursorCode->addAttribute("bearing",
|
|
|
|
MyGUI::utility::toString(data[i].kerning) + " "
|
|
|
|
+ MyGUI::utility::toString((fontSize - data[i].ascent)));
|
|
|
|
cursorCode->addAttribute(
|
|
|
|
"size", MyGUI::IntSize(static_cast<int>(data[i].width), static_cast<int>(data[i].height)));
|
2014-06-05 14:15:47 +00:00
|
|
|
}
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// These are required as well, but the fonts don't provide them
|
2022-09-22 18:26:05 +00:00
|
|
|
for (int i = 0; i < 2; ++i)
|
2013-06-06 20:13:30 +00:00
|
|
|
{
|
|
|
|
MyGUI::FontCodeType::Enum type;
|
2022-09-22 18:26:05 +00:00
|
|
|
if (i == 0)
|
2013-06-06 20:13:30 +00:00
|
|
|
type = MyGUI::FontCodeType::Selected;
|
2018-04-17 09:11:05 +00:00
|
|
|
else // if (i == 1)
|
2013-06-06 20:13:30 +00:00
|
|
|
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");
|
2014-08-07 14:27:39 +00:00
|
|
|
cursorCode->addAttribute("size", "0 0");
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
|
2022-07-27 08:15:07 +00:00
|
|
|
// Register the font with MyGUI
|
|
|
|
MyGUI::ResourceManualFont* font = static_cast<MyGUI::ResourceManualFont*>(
|
2022-09-22 18:26:05 +00:00
|
|
|
MyGUI::FactoryManager::getInstance().createObject("Resource", "ResourceManualFont"));
|
|
|
|
font->deserialization(root, MyGUI::Version(3, 2, 0));
|
2013-06-06 20:13:30 +00:00
|
|
|
|
2018-09-03 16:45:04 +00:00
|
|
|
MyGUI::ResourceManualFont* bookFont = static_cast<MyGUI::ResourceManualFont*>(
|
2022-09-22 18:26:05 +00:00
|
|
|
MyGUI::FactoryManager::getInstance().createObject("Resource", "ResourceManualFont"));
|
|
|
|
bookFont->deserialization(root, MyGUI::Version(3, 2, 0));
|
2022-08-20 11:07:23 +00:00
|
|
|
bookFont->setResourceName("Journalbook " + fontId);
|
2018-09-03 16:45:04 +00:00
|
|
|
|
2013-06-06 20:13:30 +00:00
|
|
|
MyGUI::ResourceManager::getInstance().addResource(font);
|
2018-09-03 16:45:04 +00:00
|
|
|
MyGUI::ResourceManager::getInstance().addResource(bookFont);
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|
|
|
|
|
2023-08-07 15:40:38 +00:00
|
|
|
void FontLoader::overrideLineHeight(MyGUI::xml::ElementPtr _node, std::string_view _file, MyGUI::Version _version)
|
2020-05-28 19:09:10 +00:00
|
|
|
{
|
2023-08-08 05:06:05 +00:00
|
|
|
// We should adjust line height for MyGUI widgets depending on font size
|
2020-05-28 19:09:10 +00:00
|
|
|
MyGUI::xml::ElementEnumerator resourceNode = _node->getElementEnumerator();
|
|
|
|
while (resourceNode.next("Resource"))
|
|
|
|
{
|
2023-08-07 10:43:30 +00:00
|
|
|
auto type = resourceNode->findAttribute("type");
|
2022-07-07 19:20:22 +00:00
|
|
|
|
2023-08-08 05:06:05 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(type, "ResourceLayout"))
|
|
|
|
{
|
|
|
|
MyGUI::xml::ElementEnumerator resourceRootNode = resourceNode->getElementEnumerator();
|
|
|
|
while (resourceRootNode.next("Widget"))
|
|
|
|
{
|
|
|
|
if (resourceRootNode->findAttribute("name") != "Root")
|
|
|
|
continue;
|
|
|
|
|
|
|
|
MyGUI::xml::ElementPtr heightNode = resourceRootNode->createChild("UserString");
|
|
|
|
heightNode->addAttribute("key", "HeightLine");
|
|
|
|
heightNode->addAttribute("value", std::to_string(Settings::gui().mFontSize + 2));
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-05-28 19:09:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MyGUI::ResourceManager::getInstance().loadFromXmlNode(_node, _file, _version);
|
|
|
|
}
|
|
|
|
|
2023-05-21 14:39:32 +00:00
|
|
|
std::string_view FontLoader::getFontForFace(std::string_view face)
|
2022-07-07 19:20:22 +00:00
|
|
|
{
|
2023-05-21 14:39:32 +00:00
|
|
|
if (Misc::StringUtils::ciEqual(face, "daedric"))
|
2022-07-07 19:20:22 +00:00
|
|
|
return "ScrollFont";
|
|
|
|
|
2022-08-30 07:54:20 +00:00
|
|
|
return "DefaultFont";
|
2022-07-07 19:20:22 +00:00
|
|
|
}
|
2013-06-06 20:13:30 +00:00
|
|
|
}
|