514 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			514 lines
		
	
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is part of Desktop App Toolkit,
 | |
| // a set of libraries for developing nice desktop applications.
 | |
| //
 | |
| // For license and copyright information please follow this link:
 | |
| // https://github.com/desktop-app/legal/blob/master/LEGAL
 | |
| //
 | |
| #include "ui/style/style_core_font.h"
 | |
| 
 | |
| #include "ui/ui_log.h"
 | |
| #include "base/algorithm.h"
 | |
| #include "ui/integration.h"
 | |
| 
 | |
| #include <QtCore/QMap>
 | |
| #include <QtCore/QVector>
 | |
| #include <QtGui/QFontInfo>
 | |
| #include <QtGui/QFontDatabase>
 | |
| #include <QtWidgets/QApplication>
 | |
| 
 | |
| void style_InitFontsResource() {
 | |
| #ifndef DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| 	Q_INIT_RESOURCE(fonts);
 | |
| #endif // !DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| #ifdef Q_OS_WIN
 | |
| 	Q_INIT_RESOURCE(win);
 | |
| #elif defined Q_OS_MAC // Q_OS_WIN
 | |
| 	Q_INIT_RESOURCE(mac);
 | |
| #elif defined Q_OS_LINUX && !defined DESKTOP_APP_USE_PACKAGED // Q_OS_WIN || Q_OS_MAC
 | |
| 	Q_INIT_RESOURCE(linux);
 | |
| #endif // Q_OS_WIN || Q_OS_MAC || (Q_OS_LINUX && !DESKTOP_APP_USE_PACKAGED)
 | |
| }
 | |
| 
 | |
| namespace style {
 | |
| namespace internal {
 | |
| namespace {
 | |
| 
 | |
| QMap<QString, int> fontFamilyMap;
 | |
| QVector<QString> fontFamilies;
 | |
| QMap<uint32, FontData*> fontsMap;
 | |
| 
 | |
| uint32 fontKey(int size, uint32 flags, int family) {
 | |
| 	return (((uint32(family) << 12) | uint32(size)) << 6) | flags;
 | |
| }
 | |
| 
 | |
| QString RemoveSemiboldFromName(const QString &familyName) {
 | |
| 	auto removedSemibold = familyName;
 | |
| 	removedSemibold.remove("Semibold", Qt::CaseInsensitive);
 | |
| 	return removedSemibold.trimmed();
 | |
| }
 | |
| 
 | |
| bool IsRealSemibold(const QString &familyName) {
 | |
| 	const auto removedSemibold = RemoveSemiboldFromName(familyName);
 | |
| 
 | |
| 	QFont originalFont(familyName);
 | |
| 	QFont withoutSemiboldFont(removedSemibold);
 | |
| 	withoutSemiboldFont.setStyleName("Semibold");
 | |
| 
 | |
| 	QFontInfo originalFontInfo(originalFont);
 | |
| 	QFontInfo withoutSemiboldInfo(withoutSemiboldFont);
 | |
| 
 | |
| 	if (originalFontInfo.family().trimmed().compare(familyName, Qt::CaseInsensitive) &&
 | |
| 		!withoutSemiboldInfo.family().trimmed().compare(removedSemibold, Qt::CaseInsensitive) &&
 | |
| 		!withoutSemiboldInfo.styleName().trimmed().compare("Semibold", Qt::CaseInsensitive)) {
 | |
| 		return true;
 | |
| 	} else {
 | |
| 		return false;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QString ParseFamilyName(const QString &familyName) {
 | |
| 	if (IsRealSemibold(familyName)) {
 | |
| 		return RemoveSemiboldFromName(familyName);
 | |
| 	} else {
 | |
| 		return familyName;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| bool ValidateFont(const QString &familyName, int flags = 0) {
 | |
| 	const auto parsedFamily = ParseFamilyName(familyName);
 | |
| 
 | |
| 	QFont checkFont(parsedFamily);
 | |
| 	checkFont.setBold(flags & style::internal::FontBold);
 | |
| 	checkFont.setItalic(flags & style::internal::FontItalic);
 | |
| 	checkFont.setUnderline(flags & style::internal::FontUnderline);
 | |
| 
 | |
| 	if (IsRealSemibold(familyName)) {
 | |
| 		checkFont.setStyleName("Semibold");
 | |
| 	}
 | |
| 
 | |
| 	auto realFamily = QFontInfo(checkFont).family();
 | |
| 	if (realFamily.trimmed().compare(parsedFamily, Qt::CaseInsensitive)) {
 | |
| 		UI_LOG(("Font Error: could not resolve '%1' font, got '%2'.").arg(familyName).arg(realFamily));
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	auto metrics = QFontMetrics(checkFont);
 | |
| 	if (!metrics.height()) {
 | |
| 		UI_LOG(("Font Error: got a zero height in '%1'.").arg(familyName));
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| bool LoadCustomFont(const QString &filePath, const QString &familyName, int flags = 0) {
 | |
| 	auto regularId = QFontDatabase::addApplicationFont(filePath);
 | |
| 	if (regularId < 0) {
 | |
| 		UI_LOG(("Font Error: could not add '%1'.").arg(filePath));
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	const auto found = [&] {
 | |
| 		for (auto &family : QFontDatabase::applicationFontFamilies(regularId)) {
 | |
| 			UI_LOG(("Font: from '%1' loaded '%2'").arg(filePath).arg(family));
 | |
| 			if (!family.trimmed().compare(familyName, Qt::CaseInsensitive)) {
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 		return false;
 | |
| 	}();
 | |
| 	if (!found) {
 | |
| 		UI_LOG(("Font Error: could not locate '%1' font in '%2'.").arg(familyName).arg(filePath));
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	return ValidateFont(familyName, flags);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString SystemMonospaceFont() {
 | |
| 	const auto type = QFontDatabase::FixedFont;
 | |
| 	return QFontDatabase::systemFont(type).family();
 | |
| }
 | |
| 
 | |
| bool TryFont(const QString &attempt) {
 | |
| 	const auto resolved = QFontInfo(QFont(attempt)).family();
 | |
| 	return !resolved.trimmed().compare(attempt, Qt::CaseInsensitive);
 | |
| }
 | |
| 
 | |
| [[nodiscard]] QString ManualMonospaceFont() {
 | |
| 	const auto kTryFirst = std::initializer_list<QString>{
 | |
| 		"Consolas",
 | |
| 		"Liberation Mono",
 | |
| 		"Menlo",
 | |
| 		"Courier"
 | |
| 	};
 | |
| 	for (const auto &family : kTryFirst) {
 | |
| 		if (TryFont(family)) {
 | |
| 			return family;
 | |
| 		}
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| QString MonospaceFont() {
 | |
| 	static const auto family = [&]() -> QString {
 | |
| 		if (TryFont(CustomMonospaceFont)) {
 | |
| 			return CustomMonospaceFont;
 | |
| 		}
 | |
| 
 | |
| 		const auto manual = ManualMonospaceFont();
 | |
| 		const auto system = SystemMonospaceFont();
 | |
| 
 | |
| #if defined Q_OS_WIN || defined Q_OS_MAC
 | |
| 		// Prefer our monospace font.
 | |
| 		const auto useSystem = manual.isEmpty();
 | |
| #else // Q_OS_WIN || Q_OS_MAC
 | |
| 		// Prefer system monospace font.
 | |
| 		const auto metrics = QFontMetrics(QFont(system));
 | |
| 		const auto useSystem = manual.isEmpty()
 | |
| 			|| (metrics.charWidth("i", 0) == metrics.charWidth("W", 0));
 | |
| #endif // Q_OS_WIN || Q_OS_MAC
 | |
| 		return (useSystem || UseSystemFont) ? system : manual;
 | |
| 	}();
 | |
| 
 | |
| 	return family;
 | |
| }
 | |
| 
 | |
| QFontMetrics GetFontMetrics(int size) {
 | |
| #ifdef DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| 	QFont originalFont("Open Sans");
 | |
| #else // !DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| 	QFont originalFont("DAOpenSansRegular");
 | |
| #endif // !DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| 	originalFont.setPixelSize(size);
 | |
| 	return QFontMetrics(originalFont);
 | |
| }
 | |
| 
 | |
| enum {
 | |
| 	FontTypeRegular = 0,
 | |
| 	FontTypeRegularItalic,
 | |
| 	FontTypeBold,
 | |
| 	FontTypeBoldItalic,
 | |
| 	FontTypeSemibold,
 | |
| 	FontTypeSemiboldItalic,
 | |
| 
 | |
| 	FontTypesCount,
 | |
| };
 | |
| #ifndef DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| QString FontTypeFiles[FontTypesCount] = {
 | |
| 	"DAOpenSansRegular",
 | |
| 	"DAOpenSansRegularItalic",
 | |
| 	"DAOpenSansSemiboldAsBold",
 | |
| 	"DAOpenSansSemiboldItalicAsBold",
 | |
| 	"DAOpenSansSemiboldAsBold",
 | |
| 	"DAOpenSansSemiboldItalicAsBold",
 | |
| };
 | |
| QString FontTypeNames[FontTypesCount] = {
 | |
| 	"DAOpenSansRegular",
 | |
| 	"DAOpenSansRegularItalic",
 | |
| 	"DAOpenSansSemibold",
 | |
| 	"DAOpenSansSemiboldItalic",
 | |
| 	"DAOpenSansSemibold",
 | |
| 	"DAOpenSansSemiboldItalic",
 | |
| };
 | |
| QString FontTypePersianFallback[FontTypesCount] = {
 | |
| 	"DAVazirRegular",
 | |
| 	"DAVazirRegular",
 | |
| 	"DAVazirMedium",
 | |
| 	"DAVazirMedium",
 | |
| 	"DAVazirMedium",
 | |
| 	"DAVazirMedium",
 | |
| };
 | |
| #endif // !DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| int32 FontTypeFlags[FontTypesCount] = {
 | |
| 	0,
 | |
| 	FontItalic,
 | |
| 	FontBold,
 | |
| 	FontBold | FontItalic,
 | |
| 	FontSemibold,
 | |
| 	FontSemibold | FontItalic,
 | |
| };
 | |
| #ifdef Q_OS_WIN
 | |
| QString FontTypeWindowsFallback[FontTypesCount] = {
 | |
| 	"Segoe UI",
 | |
| 	"Segoe UI",
 | |
| 	"Segoe UI",
 | |
| 	"Segoe UI",
 | |
| 	"Segoe UI",
 | |
| 	"Segoe UI",
 | |
| };
 | |
| #endif // Q_OS_WIN
 | |
| 
 | |
| bool Started = false;
 | |
| QString Overrides[FontTypesCount];
 | |
| 
 | |
| } // namespace
 | |
| 
 | |
| QString CustomMainFont;
 | |
| QString CustomSemiboldFont;
 | |
| QString CustomMonospaceFont;
 | |
| bool CustomSemiboldIsBold = false;
 | |
| bool UseSystemFont = false;
 | |
| bool UseOriginalMetrics = false;
 | |
| 
 | |
| void StartFonts() {
 | |
| 	if (Started) {
 | |
| 		return;
 | |
| 	}
 | |
| 	Started = true;
 | |
| 
 | |
| 	style_InitFontsResource();
 | |
| 
 | |
| 	const auto integrationExists = Ui::Integration::Exists();
 | |
| 	if (integrationExists) {
 | |
| 		Ui::Integration::Instance().startFontsBegin();
 | |
| 	}
 | |
| 
 | |
| #ifndef DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| 	if (!UseSystemFont) {
 | |
| 		LoadCustomFont(":/gui/fonts/DAVazirRegular.ttf", "DAVazirRegular");
 | |
| 		LoadCustomFont(":/gui/fonts/DAVazirBold.ttf", "DAVazirBold", style::internal::FontBold);
 | |
| 		LoadCustomFont(":/gui/fonts/DAVazirMedium.ttf", "DAVazirMedium", style::internal::FontSemibold);
 | |
| 
 | |
| 		bool areGood[FontTypesCount] = { false };
 | |
| 		for (auto i = 0; i != FontTypesCount; ++i) {
 | |
| 			const auto file = FontTypeFiles[i];
 | |
| 			const auto name = FontTypeNames[i];
 | |
| 			const auto flags = FontTypeFlags[i];
 | |
| 			areGood[i] = LoadCustomFont(":/gui/fonts/" + file + ".ttf", name, flags);
 | |
| 			Overrides[i] = name;
 | |
| 
 | |
| #ifdef Q_OS_WIN
 | |
| 			// Attempt to workaround a strange font bug with Open Sans Semibold not loading.
 | |
| 			// See https://github.com/telegramdesktop/tdesktop/issues/3276 for details.
 | |
| 			// Crash happens on "options.maxh / _t->_st->font->height" with "division by zero".
 | |
| 			// In that place "_t->_st->font" is "semiboldFont" is "font(13 "Open Sans Semibold").
 | |
| 			const auto fallback = FontTypeWindowsFallback[i];
 | |
| 			if (!areGood[i]) {
 | |
| 				if (ValidateFont(fallback, flags)) {
 | |
| 					Overrides[i] = fallback;
 | |
| 					UI_LOG(("Fonts Info: Using '%1' instead of '%2'.").arg(fallback).arg(name));
 | |
| 				}
 | |
| 			}
 | |
| 			// Disable default fallbacks to Segoe UI, see:
 | |
| 			// https://github.com/telegramdesktop/tdesktop/issues/5368
 | |
| 			//
 | |
| 			//QFont::insertSubstitution(name, fallback);
 | |
| #endif // Q_OS_WIN
 | |
| 
 | |
| #if defined Q_OS_WIN || defined Q_OS_LINUX
 | |
| 		const auto persianFallback = FontTypePersianFallback[i];
 | |
| 		QFont::insertSubstitution(name, persianFallback);
 | |
| #endif // Q_OS_WIN || Q_OS_LINUX
 | |
| 	}
 | |
| 
 | |
| #ifdef Q_OS_MAC
 | |
| 		auto list = QStringList();
 | |
| 		list.append("STIXGeneral");
 | |
| 		list.append(".SF NS Text");
 | |
| 		list.append("Helvetica Neue");
 | |
| 		list.append("Lucida Grande");
 | |
| 		for (const auto &name : FontTypeNames) {
 | |
| 			QFont::insertSubstitutions(name, list);
 | |
| 		}
 | |
| #endif // Q_OS_MAC
 | |
| 	}
 | |
| #endif // !DESKTOP_APP_USE_PACKAGED_FONTS
 | |
| 
 | |
| 	if (!CustomMainFont.isEmpty() && ValidateFont(CustomMainFont)) {
 | |
| 		Overrides[FontTypeRegular] = CustomMainFont;
 | |
| 		Overrides[FontTypeRegularItalic] = CustomMainFont;
 | |
| 		Overrides[FontTypeBold] = CustomMainFont;
 | |
| 		Overrides[FontTypeBoldItalic] = CustomMainFont;
 | |
| 	}
 | |
| 	if (!CustomSemiboldFont.isEmpty() && ValidateFont(CustomSemiboldFont)) {
 | |
| 		Overrides[FontTypeSemibold] = CustomSemiboldFont;
 | |
| 		Overrides[FontTypeSemiboldItalic] = CustomSemiboldFont;
 | |
| 	}
 | |
| 
 | |
| 	auto appFont = QApplication::font();
 | |
| 	appFont.setStyleStrategy(QFont::PreferQuality);
 | |
| 	QApplication::setFont(appFont);
 | |
| 
 | |
| 	if (integrationExists) {
 | |
| 		Ui::Integration::Instance().startFontsEnd();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QString GetPossibleEmptyOverride(int32 flags) {
 | |
| 	flags = flags & (FontBold | FontSemibold | FontItalic);
 | |
| 	int32 flagsBold = flags & (FontBold | FontItalic);
 | |
| 	int32 flagsSemibold = flags & (FontSemibold | FontItalic);
 | |
| 	if (flagsSemibold == (FontSemibold | FontItalic)) {
 | |
| 		return Overrides[FontTypeSemiboldItalic];
 | |
| 	} else if (flagsSemibold == FontSemibold) {
 | |
| 		return Overrides[FontTypeSemibold];
 | |
| 	} else if (flagsBold == (FontBold | FontItalic)) {
 | |
| 		return Overrides[FontTypeBoldItalic];
 | |
| 	} else if (flagsBold == FontBold) {
 | |
| 		return Overrides[FontTypeBold];
 | |
| 	} else if (flags == FontItalic) {
 | |
| 		return Overrides[FontTypeRegularItalic];
 | |
| 	} else if (flags == 0) {
 | |
| 		return Overrides[FontTypeRegular];
 | |
| 	}
 | |
| 	return QString();
 | |
| }
 | |
| 
 | |
| QString GetFontOverride(int32 flags) {
 | |
| 	const auto result = GetPossibleEmptyOverride(flags);
 | |
| 	return result.isEmpty() ? "Open Sans" : result;
 | |
| }
 | |
| 
 | |
| void destroyFonts() {
 | |
| 	for (auto fontData : fontsMap) {
 | |
| 		delete fontData;
 | |
| 	}
 | |
| 	fontsMap.clear();
 | |
| }
 | |
| 
 | |
| int registerFontFamily(const QString &family) {
 | |
| 	auto result = fontFamilyMap.value(family, -1);
 | |
| 	if (result < 0) {
 | |
| 		result = fontFamilies.size();
 | |
| 		fontFamilyMap.insert(family, result);
 | |
| 		fontFamilies.push_back(family);
 | |
| 	}
 | |
| 	return result;
 | |
| }
 | |
| 
 | |
| FontData::FontData(int size, uint32 flags, int family, Font *other)
 | |
| : m(f)
 | |
| , _size(size)
 | |
| , _flags(flags)
 | |
| , _family(family) {
 | |
| 	const auto fontOverride = ParseFamilyName(GetFontOverride(flags));
 | |
| 	const auto overrideIsEmpty = GetPossibleEmptyOverride(flags).isEmpty();
 | |
| 
 | |
| 	if (other) {
 | |
| 		memcpy(modified, other, sizeof(modified));
 | |
| 	} else {
 | |
| 		memset(modified, 0, sizeof(modified));
 | |
| 	}
 | |
| 	modified[_flags] = Font(this);
 | |
| 
 | |
| 	if (_flags & FontMonospace) {
 | |
| 		f.setFamily(MonospaceFont());
 | |
| 	} else if (!UseSystemFont || !overrideIsEmpty) {
 | |
| 		f.setFamily(fontOverride);
 | |
| 	}
 | |
| 
 | |
| 	f.setPixelSize(size);
 | |
| 	f.setBold((_flags & FontBold) || (_flags & FontSemibold));
 | |
| 	f.setItalic(_flags & FontItalic);
 | |
| 	f.setUnderline(_flags & FontUnderline);
 | |
| 	f.setStrikeOut(_flags & FontStrikeOut);
 | |
| 
 | |
| 	if ((_flags & FontBold) || (_flags & FontSemibold)) {
 | |
| 		if (CustomSemiboldIsBold) {
 | |
| 			f.setBold(true);
 | |
| 		} else {
 | |
| 			f.setStyleName("Semibold");
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (IsRealSemibold(fontOverride)) {
 | |
| 		f.setStyleName("Semibold");
 | |
| 	}
 | |
| 
 | |
| 	m = QFontMetrics(f);
 | |
| 
 | |
| 	if (UseOriginalMetrics && !(_flags & FontMonospace)) {
 | |
| 		const auto mOrig = GetFontMetrics(size);
 | |
| 
 | |
| 		height = mOrig.height();
 | |
| 		ascent = mOrig.ascent();
 | |
| 		descent = mOrig.descent();
 | |
| 	} else {
 | |
| 		height = m.height();
 | |
| 		ascent = m.ascent();
 | |
| 		descent = m.descent();
 | |
| 	}
 | |
| 
 | |
| 	spacew = width(QLatin1Char(' '));
 | |
| 	elidew = width("...");
 | |
| }
 | |
| 
 | |
| Font FontData::bold(bool set) const {
 | |
| 	return otherFlagsFont(FontBold, set);
 | |
| }
 | |
| 
 | |
| Font FontData::italic(bool set) const {
 | |
| 	return otherFlagsFont(FontItalic, set);
 | |
| }
 | |
| 
 | |
| Font FontData::underline(bool set) const {
 | |
| 	return otherFlagsFont(FontUnderline, set);
 | |
| }
 | |
| 
 | |
| Font FontData::strikeout(bool set) const {
 | |
| 	return otherFlagsFont(FontStrikeOut, set);
 | |
| }
 | |
| 
 | |
| Font FontData::semibold(bool set) const {
 | |
| 	return otherFlagsFont(FontSemibold, set);
 | |
| }
 | |
| 
 | |
| Font FontData::monospace(bool set) const {
 | |
| 	return otherFlagsFont(FontMonospace, set);
 | |
| }
 | |
| 
 | |
| int FontData::size() const {
 | |
| 	return _size;
 | |
| }
 | |
| 
 | |
| uint32 FontData::flags() const {
 | |
| 	return _flags;
 | |
| }
 | |
| 
 | |
| int FontData::family() const {
 | |
| 	return _family;
 | |
| }
 | |
| 
 | |
| Font FontData::otherFlagsFont(uint32 flag, bool set) const {
 | |
| 	int32 newFlags = set ? (_flags | flag) : (_flags & ~flag);
 | |
| 	if (!modified[newFlags].v()) {
 | |
| 		modified[newFlags] = Font(_size, newFlags, _family, modified);
 | |
| 	}
 | |
| 	return modified[newFlags];
 | |
| }
 | |
| 
 | |
| Font::Font(int size, uint32 flags, const QString &family) {
 | |
| 	if (fontFamilyMap.isEmpty()) {
 | |
| 		for (uint32 i = 0, s = fontFamilies.size(); i != s; ++i) {
 | |
| 			fontFamilyMap.insert(fontFamilies.at(i), i);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	auto i = fontFamilyMap.constFind(family);
 | |
| 	if (i == fontFamilyMap.cend()) {
 | |
| 		fontFamilies.push_back(family);
 | |
| 		i = fontFamilyMap.insert(family, fontFamilies.size() - 1);
 | |
| 	}
 | |
| 	init(size, flags, i.value(), 0);
 | |
| }
 | |
| 
 | |
| Font::Font(int size, uint32 flags, int family) {
 | |
| 	init(size, flags, family, 0);
 | |
| }
 | |
| 
 | |
| Font::Font(int size, uint32 flags, int family, Font *modified) {
 | |
| 	init(size, flags, family, modified);
 | |
| }
 | |
| 
 | |
| void Font::init(int size, uint32 flags, int family, Font *modified) {
 | |
| 	uint32 key = fontKey(size, flags, family);
 | |
| 	auto i = fontsMap.constFind(key);
 | |
| 	if (i == fontsMap.cend()) {
 | |
| 		i = fontsMap.insert(key, new FontData(size, flags, family, modified));
 | |
| 	}
 | |
| 	ptr = i.value();
 | |
| }
 | |
| 
 | |
| } // namespace internal
 | |
| } // namespace style
 |