106void QWindowsDirectWriteFontDatabase::populateFamily(
const QString &familyName)
108 auto it = m_populatedFonts.find(familyName);
109 if (it == m_populatedFonts.end() && m_populatedBitmapFonts.contains(familyName)) {
110 qCDebug(lcQpaFonts) <<
"Populating bitmap font" << familyName;
111 QWindowsFontDatabase::populateFamily(familyName);
115 IDWriteFontFamily *fontFamily = it != m_populatedFonts.end() ? it.value() :
nullptr;
116 if (fontFamily ==
nullptr) {
117 qCWarning(lcQpaFonts) <<
"Cannot find" << familyName <<
"in list of fonts";
121 qCDebug(lcQpaFonts) <<
"Populate family:" << familyName;
123 wchar_t defaultLocale[LOCALE_NAME_MAX_LENGTH];
124 bool hasDefaultLocale = GetUserDefaultLocaleName(defaultLocale, LOCALE_NAME_MAX_LENGTH) != 0;
125 wchar_t englishLocale[] = L"en-us";
127 static const int SMOOTH_SCALABLE = 0xffff;
128 const QString foundryName;
129 const bool scalable =
true;
130 const bool antialias =
false;
131 const int size = SMOOTH_SCALABLE;
133 DirectWriteScope<IDWriteFontList> matchingFonts;
134 if (SUCCEEDED(fontFamily->GetMatchingFonts(DWRITE_FONT_WEIGHT_REGULAR,
135 DWRITE_FONT_STRETCH_NORMAL,
136 DWRITE_FONT_STYLE_NORMAL,
138 for (uint j = 0; j < matchingFonts->GetFontCount(); ++j) {
139 DirectWriteScope<IDWriteFont> font;
140 if (SUCCEEDED(matchingFonts->GetFont(j, &font))) {
141 DirectWriteScope<IDWriteFont2> font2;
142 if (!SUCCEEDED(font->QueryInterface(__uuidof(IDWriteFont2),
143 reinterpret_cast<
void **>(&font2)))) {
144 qCWarning(lcQpaFonts) <<
"COM object does not support IDWriteFont1";
148 QString defaultLocaleFamilyName;
149 QString englishLocaleFamilyName;
150 QString defaultLocaleGdiCompatibleFamilyName;
151 QString englishLocaleGdiCompatibleFamilyName;
153 DirectWriteScope<IDWriteFontFamily> fontFamily2;
154 if (SUCCEEDED(font2->GetFontFamily(&fontFamily2))) {
155 DirectWriteScope<IDWriteLocalizedStrings> names;
156 if (SUCCEEDED(fontFamily2->GetFamilyNames(&names))) {
157 defaultLocaleFamilyName = hasDefaultLocale ? localeString(*names, defaultLocale) : QString();
158 englishLocaleFamilyName = localeString(*names, englishLocale);
162 if (SUCCEEDED(font->GetInformationalStrings(DWRITE_INFORMATIONAL_STRING_WIN32_FAMILY_NAMES, &names, &ok)) && ok) {
163 defaultLocaleGdiCompatibleFamilyName = hasDefaultLocale ? localeString(*names, defaultLocale) : QString();
164 englishLocaleGdiCompatibleFamilyName = localeString(*names, englishLocale);
168 if (defaultLocaleFamilyName.isEmpty()
169 && englishLocaleFamilyName.isEmpty()
170 && englishLocaleGdiCompatibleFamilyName.isEmpty()
171 && defaultLocaleGdiCompatibleFamilyName.isEmpty()) {
172 englishLocaleFamilyName = familyName;
175 QFont::Stretch stretch = fromDirectWriteStretch(font2->GetStretch());
176 QFont::Style style = fromDirectWriteStyle(font2->GetStyle());
177 QFont::Weight weight = fromDirectWriteWeight(font2->GetWeight());
178 bool fixed = font2->IsMonospacedFont();
179 bool color = font2->IsColorFont();
181 DirectWriteScope<IDWriteFontFace> face;
182 if (SUCCEEDED(font->CreateFontFace(&face))) {
183 QSupportedWritingSystems writingSystems = supportedWritingSystems(*face);
185 auto registerFontName = [&](
const QString &rFamilyName,
186 const QString &rStyleName) {
187 qCDebug(lcQpaFonts) <<
"Family" << familyName <<
"has variant"
188 << rFamilyName <<
", "
189 << rStyleName <<
", "
195 QPlatformFontDatabase::registerFont(rFamilyName,
207 new FontHandle(*face, rFamilyName));
211 DirectWriteScope<IDWriteLocalizedStrings> names;
212 if (SUCCEEDED(font2->GetFaceNames(&names))) {
213 QString defaultLocaleStyleName = hasDefaultLocale ? localeString(*names, defaultLocale) : QString();
214 QString englishLocaleStyleName = localeString(*names, englishLocale);
215 qCDebug(lcQpaFonts) <<
">>> Storing English locale names";
216 registerFontName(englishLocaleFamilyName, englishLocaleStyleName);
220 if (!defaultLocaleFamilyName.isEmpty()
221 && defaultLocaleFamilyName != englishLocaleFamilyName) {
222 qCDebug(lcQpaFonts) <<
">>> Storing default locale names";
223 registerFontName(defaultLocaleFamilyName, defaultLocaleStyleName);
228 if (!defaultLocaleGdiCompatibleFamilyName.isEmpty()
229 || !englishLocaleGdiCompatibleFamilyName.isEmpty()) {
230 QString defaultLocaleGdiCompatibleStyleName;
231 QString englishLocaleGdiCompatibleStyleName;
233 DirectWriteScope<IDWriteLocalizedStrings> names;
235 if (SUCCEEDED(font->GetInformationalStrings(DWRITE_INFORMATIONAL_STRING_WIN32_SUBFAMILY_NAMES, &names, &ok)) && ok) {
236 defaultLocaleGdiCompatibleStyleName = hasDefaultLocale ? localeString(*names, defaultLocale) : QString();
237 englishLocaleGdiCompatibleStyleName = localeString(*names, englishLocale);
239 if (!englishLocaleGdiCompatibleFamilyName.isEmpty()
240 && englishLocaleGdiCompatibleFamilyName != englishLocaleFamilyName) {
241 qCDebug(lcQpaFonts) <<
">>> Storing English GDI compatible names";
242 registerFontName(englishLocaleGdiCompatibleFamilyName,
243 englishLocaleGdiCompatibleStyleName);
246 if (!defaultLocaleGdiCompatibleFamilyName.isEmpty()
247 && defaultLocaleGdiCompatibleFamilyName != englishLocaleGdiCompatibleFamilyName
248 && defaultLocaleGdiCompatibleFamilyName != defaultLocaleFamilyName) {
249 qCDebug(lcQpaFonts) <<
">>> Storing Default locale GDI compatible names";
250 registerFontName(defaultLocaleGdiCompatibleFamilyName,
251 defaultLocaleGdiCompatibleStyleName);
322QFontEngine *QWindowsDirectWriteFontDatabase::fontEngine(
const QFontDef &fontDef,
void *handle)
324 const FontHandle *fontHandle =
static_cast<
const FontHandle *>(handle);
325 IDWriteFontFace *face = fontHandle->fontFace;
326 if (face ==
nullptr) {
327 qCDebug(lcQpaFonts) <<
"Falling back to GDI";
328 return QWindowsFontDatabase::fontEngine(fontDef, handle);
331 DWRITE_FONT_SIMULATIONS simulations = DWRITE_FONT_SIMULATIONS_NONE;
332 if (fontDef.weight >= QFont::DemiBold || fontDef.style != QFont::StyleNormal) {
333 DirectWriteScope<IDWriteFontFace3> face3;
334 if (SUCCEEDED(face->QueryInterface(__uuidof(IDWriteFontFace3),
335 reinterpret_cast<
void **>(&face3)))) {
336 if (fontDef.weight >= QFont::DemiBold && face3->GetWeight() < DWRITE_FONT_WEIGHT_DEMI_BOLD)
337 simulations |= DWRITE_FONT_SIMULATIONS_BOLD;
339 if (fontDef.style != QFont::StyleNormal && face3->GetStyle() == DWRITE_FONT_STYLE_NORMAL)
340 simulations |= DWRITE_FONT_SIMULATIONS_OBLIQUE;
344 DirectWriteScope<IDWriteFontFace5> newFace;
345 if (!fontDef.variableAxisValues.isEmpty() || simulations != DWRITE_FONT_SIMULATIONS_NONE) {
346 DirectWriteScope<IDWriteFontFace5> face5;
347 if (SUCCEEDED(face->QueryInterface(__uuidof(IDWriteFontFace5),
348 reinterpret_cast<
void **>(&face5)))) {
349 DirectWriteScope<IDWriteFontResource> font;
350 if (SUCCEEDED(face5->GetFontResource(&font))) {
351 UINT32 fontAxisCount = font->GetFontAxisCount();
352 QVarLengthArray<DWRITE_FONT_AXIS_VALUE, 8> fontAxisValues(fontAxisCount);
354 if (!fontDef.variableAxisValues.isEmpty()) {
355 if (SUCCEEDED(face5->GetFontAxisValues(fontAxisValues.data(), fontAxisCount))) {
356 for (UINT32 i = 0; i < fontAxisCount; ++i) {
357 if (
auto maybeTag = QFont::Tag::fromValue(qToBigEndian<UINT32>(fontAxisValues[i].axisTag))) {
358 if (fontDef.variableAxisValues.contains(*maybeTag))
359 fontAxisValues[i].value = fontDef.variableAxisValues.value(*maybeTag);
365 if (SUCCEEDED(font->CreateFontFace(simulations,
366 !fontDef.variableAxisValues.isEmpty() ? fontAxisValues.data() :
nullptr,
367 !fontDef.variableAxisValues.isEmpty() ? fontAxisCount : 0,
371 qCWarning(lcQpaFonts) <<
"DirectWrite: Can't create font face for variable axis values";
377 QWindowsFontEngineDirectWrite *fontEngine =
new QWindowsFontEngineDirectWrite(face, fontDef.pixelSize, data());
378 fontEngine->initFontInfo(fontDef, defaultVerticalDPI());
472QStringList QWindowsDirectWriteFontDatabase::addApplicationFont(
const QByteArray &fontData,
const QString &fileName, QFontDatabasePrivate::ApplicationFont *applicationFont)
474 qCDebug(lcQpaFonts) <<
"Adding application font" << fileName;
476 QByteArray loadedData = fontData;
477 if (loadedData.isEmpty()) {
478 QFile file(fileName);
479 if (!file.open(QIODevice::ReadOnly)) {
480 qCWarning(lcQpaFonts) <<
"Cannot open" << fileName <<
"for reading.";
481 return QStringList();
483 loadedData = file.readAll();
486 QList<IDWriteFontFace *> faces = createDirectWriteFaces(loadedData, fileName);
487 if (faces.isEmpty()) {
488 qCWarning(lcQpaFonts) <<
"Failed to create DirectWrite face from font data. Font may be unsupported.";
489 return QStringList();
492 QSet<std::pair<QString, QString> > registeredFonts;
494 for (
int i = 0; i < faces.size(); ++i) {
495 IDWriteFontFace *face = faces.at(i);
496 wchar_t defaultLocale[LOCALE_NAME_MAX_LENGTH];
497 bool hasDefaultLocale = GetUserDefaultLocaleName(defaultLocale, LOCALE_NAME_MAX_LENGTH) != 0;
498 wchar_t englishLocale[] = L"en-us";
500 static const int SMOOTH_SCALABLE = 0xffff;
501 const bool scalable =
true;
502 const bool antialias =
false;
503 const int size = SMOOTH_SCALABLE;
505 QSupportedWritingSystems writingSystems = supportedWritingSystems(face);
506 DirectWriteScope<IDWriteFontFace3> face3;
507 if (SUCCEEDED(face->QueryInterface(__uuidof(IDWriteFontFace3),
508 reinterpret_cast<
void **>(&face3)))) {
510 QFont::Stretch stretch = fromDirectWriteStretch(face3->GetStretch());
511 QFont::Style style = fromDirectWriteStyle(face3->GetStyle());
512 QFont::Weight weight = fromDirectWriteWeight(face3->GetWeight());
513 bool fixed = face3->IsMonospacedFont();
514 bool color = face3->IsColorFont();
516 auto registerFamilyAndStyle = [&](
const std::pair<QString, QString> &familyAndStyle)
518 if (registeredFonts.contains(familyAndStyle))
520 registeredFonts.insert(familyAndStyle);
521 ret.insert(familyAndStyle.first);
523 qCDebug(lcQpaFonts) <<
"\tRegistering alternative:" << familyAndStyle.first
524 <<
":" << familyAndStyle.second;
525 if (applicationFont !=
nullptr) {
526 QFontDatabasePrivate::ApplicationFont::Properties properties;
527 properties.style = style;
528 properties.weight = weight;
529 properties.familyName = familyAndStyle.first;
530 properties.styleName = familyAndStyle.second;
531 applicationFont->properties.append(properties);
534 QPlatformFontDatabase::registerFont(familyAndStyle.first,
535 familyAndStyle.second,
546 new FontHandle(face, familyAndStyle.first));
549 QString defaultLocaleFamilyName;
550 QString englishLocaleFamilyName;
552 IDWriteLocalizedStrings *names =
nullptr;
553 if (SUCCEEDED(face3->GetFamilyNames(&names))) {
554 defaultLocaleFamilyName = hasDefaultLocale ? localeString(names, defaultLocale) : QString();
555 englishLocaleFamilyName = localeString(names, englishLocale);
560 QString defaultLocaleStyleName;
561 QString englishLocaleStyleName;
562 if (SUCCEEDED(face3->GetFaceNames(&names))) {
563 defaultLocaleStyleName = hasDefaultLocale ? localeString(names, defaultLocale) : QString();
564 englishLocaleStyleName = localeString(names, englishLocale);
569 qCDebug(lcQpaFonts) <<
"\tFont names:" << englishLocaleFamilyName <<
", " << defaultLocaleFamilyName
570 <<
", style names:" << englishLocaleStyleName <<
", " << defaultLocaleStyleName
571 <<
", stretch:" << stretch
572 <<
", style:" << style
573 <<
", weight:" << weight
574 <<
", fixed:" << fixed;
577 const auto key = std::make_pair(englishLocaleFamilyName, englishLocaleStyleName);
578 if (!englishLocaleFamilyName.isEmpty())
579 registerFamilyAndStyle(key);
583 const auto key = std::make_pair(defaultLocaleFamilyName, defaultLocaleStyleName);
584 if (!defaultLocaleFamilyName.isEmpty())
585 registerFamilyAndStyle(key);
588 collectAdditionalNames(*face3,
589 hasDefaultLocale ? defaultLocale :
nullptr,
591 registerFamilyAndStyle);
594 qCWarning(lcQpaFonts) <<
"Unable to query IDWriteFontFace3 interface from font face.";
632void QWindowsDirectWriteFontDatabase::populateFontDatabase()
634 wchar_t defaultLocale[LOCALE_NAME_MAX_LENGTH];
635 bool hasDefaultLocale = GetUserDefaultLocaleName(defaultLocale, LOCALE_NAME_MAX_LENGTH) != 0;
636 wchar_t englishLocale[] = L"en-us";
638 const QString defaultFontName = defaultFont().family();
639 const QString systemDefaultFontName = systemDefaultFont().family();
641 DirectWriteScope<IDWriteFontCollection2> fontCollection;
642 DirectWriteScope<IDWriteFactory6> factory6;
643 if (FAILED(data()->directWriteFactory->QueryInterface(__uuidof(IDWriteFactory6),
644 reinterpret_cast<
void **>(&factory6)))) {
645 qCWarning(lcQpaFonts) <<
"Can't initialize IDWriteFactory6. Use GDI font engine instead.";
649 if (SUCCEEDED(factory6->GetSystemFontCollection(
false,
650 DWRITE_FONT_FAMILY_MODEL_TYPOGRAPHIC,
652 QSet<QString> registeredFamilies;
653 for (uint i = 0; i < fontCollection->GetFontFamilyCount(); ++i) {
654 DirectWriteScope<IDWriteFontFamily2> fontFamily;
655 if (SUCCEEDED(fontCollection->GetFontFamily(i, &fontFamily))) {
656 auto registerFamily = [&](
const std::pair<QString, QString> &familyAndStyle) {
657 const QString registeredFamily = familyAndStyle.first;
658 if (registeredFamilies.contains(registeredFamily))
660 registeredFamilies.insert(registeredFamily);
662 qCDebug(lcQpaFonts) <<
"Registering font family" << registeredFamily;
663 registerFontFamily(registeredFamily);
664 m_populatedFonts.insert(registeredFamily, *fontFamily);
665 fontFamily->AddRef();
667 if (registeredFamily == defaultFontName
668 && defaultFontName != systemDefaultFontName) {
669 qCDebug(lcQpaFonts) <<
"Adding default font" << systemDefaultFontName
670 <<
"as alternative to" << registeredFamily;
672 m_populatedFonts.insert(systemDefaultFontName, *fontFamily);
673 fontFamily->AddRef();
677 QString defaultLocaleName;
678 QString englishLocaleName;
679 DirectWriteScope<IDWriteLocalizedStrings> names;
680 if (SUCCEEDED(fontFamily->GetFamilyNames(&names))) {
681 if (hasDefaultLocale)
682 defaultLocaleName = localeString(*names, defaultLocale);
683 englishLocaleName = localeString(*names, englishLocale);
687 const auto key = std::make_pair(defaultLocaleName, QString{});
688 if (!defaultLocaleName.isEmpty())
693 const auto key = std::make_pair(englishLocaleName, QString{});
694 if (!englishLocaleName.isEmpty())
698 for (uint j = 0; j < fontFamily->GetFontCount(); ++j) {
699 DirectWriteScope<IDWriteFont3> font;
700 if (SUCCEEDED(fontFamily->GetFont(j, &font))) {
701 collectAdditionalNames(*font,
702 hasDefaultLocale ? defaultLocale :
nullptr,
713 HDC dummy = GetDC(0);
715 lf.lfCharSet = DEFAULT_CHARSET;
716 lf.lfFaceName[0] = 0;
717 lf.lfPitchAndFamily = 0;
718 EnumFontFamiliesEx(dummy, &lf, populateBitmapFonts,
reinterpret_cast<intptr_t>(
this), 0);