143 lines
3.9 KiB
C++
143 lines
3.9 KiB
C++
/*
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
*/
|
|
|
|
#include "Host/SystemFontResolver.hpp"
|
|
#include "Base/Utils.hpp"
|
|
#include "Base/Logger.hpp"
|
|
#include <Windows.h>
|
|
#include <dwrite_3.h>
|
|
#include <filesystem>
|
|
|
|
#define QFR_DESCRIPTION 1
|
|
|
|
#pragma comment(lib, "Dwrite.lib")
|
|
|
|
namespace
|
|
{
|
|
template<typename T>
|
|
struct DirectWriteAutoReleasable final
|
|
{
|
|
DirectWriteAutoReleasable() : ptr(nullptr) {}
|
|
~DirectWriteAutoReleasable()
|
|
{
|
|
if (ptr)
|
|
{
|
|
ptr->Release();
|
|
}
|
|
}
|
|
|
|
T* operator->() { return ptr; }
|
|
|
|
T* ptr;
|
|
};
|
|
}
|
|
|
|
namespace OpenVulkano
|
|
{
|
|
const std::string& SystemFontResolver::GetSystemFontPath(const std::string& fontName)
|
|
{
|
|
// font name -> filename
|
|
static std::map<std::string, std::string> fontFileMapping = ReadSystemFonts();
|
|
static std::string fallbackString;
|
|
auto it = fontFileMapping.find(Utils::ToLower(fontName));
|
|
return it == fontFileMapping.end() ? fallbackString : it->second;
|
|
}
|
|
|
|
std::map<std::string, std::string> SystemFontResolver::ReadSystemFonts()
|
|
{
|
|
DirectWriteAutoReleasable<IDWriteFactory7> dwrite;
|
|
HRESULT hr = DWriteCreateFactory(DWRITE_FACTORY_TYPE_SHARED, __uuidof(IDWriteFactory7), (IUnknown**) &dwrite);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
Logger::DATA->error("Could not read system fonts. DWriteCreateFactory has failed. Error code {}", hr);
|
|
return {};
|
|
}
|
|
|
|
DirectWriteAutoReleasable<IDWriteFontSet> matchingFonts;
|
|
hr = dwrite->GetSystemFontSet(&matchingFonts.ptr);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
Logger::DATA->error("Could not read system fonts. GetSystemFontSet has failed. Error code {}", hr);
|
|
return {};
|
|
}
|
|
|
|
std::map<std::string, std::string> fontFileMapping;
|
|
const UINT32 familyCount = matchingFonts->GetFontCount();
|
|
// thank you Microsoft for function that is not even documented, but exists and it's the only function that
|
|
// can return real font name from font filename (e.g. font filename is times.ttf that corresponds to Times New Roman)
|
|
int(WINAPI* GetFontResourceInfoW)(wchar_t*, unsigned long*, void*, unsigned long);
|
|
*(FARPROC*) &GetFontResourceInfoW = GetProcAddress(GetModuleHandleA("gdi32"), "GetFontResourceInfoW");
|
|
|
|
for (UINT32 i = 0; i < familyCount; ++i)
|
|
{
|
|
DirectWriteAutoReleasable<IDWriteFontFaceReference> faceRef;
|
|
hr = matchingFonts->GetFontFaceReference(i, &faceRef.ptr);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
DirectWriteAutoReleasable<IDWriteFontFile> file;
|
|
hr = faceRef->GetFontFile(&file.ptr);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
DirectWriteAutoReleasable<IDWriteFontFileLoader> loader;
|
|
hr = file->GetLoader(&loader.ptr);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
DirectWriteAutoReleasable<IDWriteLocalFontFileLoader> localLoader;
|
|
hr = loader->QueryInterface(&localLoader.ptr);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
const void* fileKey;
|
|
UINT32 fileKeySize;
|
|
hr = file->GetReferenceKey(&fileKey, &fileKeySize);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Get font path
|
|
WCHAR filePath[MAX_PATH];
|
|
hr = localLoader->GetFilePathFromKey(fileKey, fileKeySize, filePath, MAX_PATH);
|
|
if (!SUCCEEDED(hr))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
// Get font name
|
|
unsigned long size = 0;
|
|
if (!GetFontResourceInfoW(filePath, &size, NULL, QFR_DESCRIPTION))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
std::wstring fontName;
|
|
fontName.resize(size);
|
|
if (GetFontResourceInfoW(filePath, &size, fontName.data(), QFR_DESCRIPTION))
|
|
{
|
|
// remove null-terminated characters since size is always bigger than needed
|
|
std::string fontNameCropped(fontName.begin(), fontName.end());
|
|
const size_t realSize = strlen(fontNameCropped.data());
|
|
fontNameCropped.resize(realSize);
|
|
Utils::ToLower(fontNameCropped);
|
|
fontFileMapping[std::move(fontNameCropped)] = std::string(filePath, filePath + wcslen(filePath));
|
|
}
|
|
}
|
|
return fontFileMapping;
|
|
}
|
|
|
|
}
|