Qt源码那些事儿-Linux下Qt的QFontDataBase字体引擎解析

2024-05-12 19:48

本文主要是介绍Qt源码那些事儿-Linux下Qt的QFontDataBase字体引擎解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

公众号:Qt那些事儿

公众号

简介

Qt中对于字体处理最常用的三个类

  • QFontDataBase
  • QFont
  • QFontInfo

今天主要讲的是Qt在Linux下字体的基础-QFontDataBase类。

QFontDataBase

QFontDataBase为Qt提供了当前系统所有可用的字体及其信息

Linux平台

Linux平台下,Qt解析字体相关的库使用的是FontConfig。
Fontconfig是用于配置和自定义字体访问的库。

Fontconfig可以:
自动安装时发现新字体,从而消除了常见的配置问题。
执行字体名称替换,以便在缺少字体时可以选择适当的替代字体。
确定完全覆盖一组语言所需的一组字体。
因为使用基于XML的配置文件而构建了GUI配置工具(尽管具有自动发现功能,但我们认为这种需求已最小化)。
即使已安装了数千种字体,也可以在减少内存使用的同时,高效,快速地在已安装的字体集中找到所需的字体。
与X Render Extension和FreeType配合使用,可在显示器上实现高质量,抗锯齿和亚像素渲染的文本。

Fontconfig不会:
自己渲染字体(留给FreeType或其他渲染机制)
以任何方式依赖于X Window系统,因此仅打印机应用程序不具有此类依赖关系

Qt的流程

所以Qt解析字体的一个流程即

使用FontConfig提取字体相关的信息(比如字体名称,风格)。组织成Qt自己的数据结构。然后再提供给QFont,QFontInfo等类来使用。

先上Qt中使用FontConfig解析字体相关信息的代码

//qt-everywhere-src-5.15.0/qtbase/src/platformsupport/fontdatabases/fontconfig/qfontconfigdatabase.cpp
void QFontconfigDatabase::populateFontDatabase()
{FcInit();FcFontSet  *fonts;{FcObjectSet *os = FcObjectSetCreate();FcPattern *pattern = FcPatternCreate();const char *properties [] = {FC_FAMILY, FC_STYLE, FC_WEIGHT, FC_SLANT,FC_SPACING, FC_FILE, FC_INDEX,FC_LANG, FC_CHARSET, FC_FOUNDRY, FC_SCALABLE, FC_PIXEL_SIZE,FC_WIDTH, FC_FAMILYLANG,
#if FC_VERSION >= 20297FC_CAPABILITY,
#endif(const char *)nullptr};const char **p = properties;while (*p) {FcObjectSetAdd(os, *p);++p;}fonts = FcFontList(nullptr, pattern, os);FcObjectSetDestroy(os);FcPatternDestroy(pattern);}

上面代码首先将 字体中需要解析的信息(比如字体名称FC_FAMILY,字体风格FC_STYLEden 等等) 塞到FcObjectSet中,等待解析。然后通过函数FcFontList获取系统中所有的字体信息(上面我们添加的)FcFontSet *fonts

然后开始通过循环解析我们系统中的字体

    for (int i = 0; i < fonts->nfont; i++)populateFromPattern(fonts->fonts[i]);

下面开始讲解 populateFromPattern

    if (FcPatternGetString(pattern, FC_FAMILY, 0, &value) != FcResultMatch)return;familyName = QString::fromUtf8((const char *)value);if (FcPatternGetString(pattern, FC_FAMILYLANG, 0, &value) == FcResultMatch)familyNameLang = QString::fromUtf8((const char *)value);

通过上面的函数我们得到了字体的名称,跟名称对应的语言。即 FC_FAMILY FC_FAMILYLANG。这里只上了部分代码,其它的函数也是如此获取的字体相关的信息

//xxxxxxxxxxxxxxxxxxxxxxxxxxxif (FcPatternGetInteger(pattern, FC_SLANT, 0, &slant_value) != FcResultMatch)slant_value = FC_SLANT_ROMAN;if (FcPatternGetInteger(pattern, FC_WEIGHT, 0, &weight_value) != FcResultMatch)weight_value = FC_WEIGHT_REGULAR;if (FcPatternGetInteger(pattern, FC_WIDTH, 0, &width_value) != FcResultMatch)width_value = FC_WIDTH_NORMAL;if (FcPatternGetInteger(pattern, FC_SPACING, 0, &spacing_value) != FcResultMatch)spacing_value = FC_PROPORTIONAL;if (FcPatternGetString(pattern, FC_FILE, 0, &file_value) != FcResultMatch)file_value = nullptr;if (FcPatternGetInteger(pattern, FC_INDEX, 0, &indexValue) != FcResultMatch)indexValue = 0;if (FcPatternGetBool(pattern, FC_SCALABLE, 0, &scalable) != FcResultMatch)scalable = FcTrue;if (FcPatternGetString(pattern, FC_FOUNDRY, 0, &foundry_value) != FcResultMatch)foundry_value = nullptr;if (FcPatternGetString(pattern, FC_STYLE, 0, &style_value) != FcResultMatch)style_value = nullptr;if (FcPatternGetBool(pattern,FC_ANTIALIAS,0,&antialias) != FcResultMatch)antialias = true;//xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

这样,我们就得到了所有的字体信息,然后最后通过函数放入到Qt中

void qt_registerFont(const QString &familyName, const QString &stylename,const QString &foundryname, int weight,QFont::Style style, int stretch, bool antialiased,bool scalable, int pixelSize, bool fixedPitch,const QSupportedWritingSystems &writingSystems, void *handle)
{QFontDatabasePrivate *d = privateDb();qCDebug(lcFontDb) << "Adding font: familyName" << familyName << "stylename" << stylename << "weight" << weight<< "style" << style << "pixelSize" << pixelSize << "antialiased" << antialiased << "fixed" << fixedPitch;QtFontStyle::Key styleKey;styleKey.style = style;styleKey.weight = weight;styleKey.stretch = stretch;QtFontFamily *f = d->family(familyName, QFontDatabasePrivate::EnsureCreated);f->fixedPitch = fixedPitch;for (int i = 0; i < QFontDatabase::WritingSystemsCount; ++i) {if (writingSystems.supported(QFontDatabase::WritingSystem(i)))f->writingSystems[i] = QtFontFamily::Supported;}QtFontFoundry *foundry = f->foundry(foundryname, true);QtFontStyle *fontStyle = foundry->style(styleKey, stylename, true);fontStyle->smoothScalable = scalable;fontStyle->antialiased = antialiased;QtFontSize *size = fontStyle->pixelSize(pixelSize ? pixelSize : SMOOTH_SCALABLE, true);if (size->handle) {QPlatformIntegration *integration = QGuiApplicationPrivate::platformIntegration();if (integration)integration->fontDatabase()->releaseHandle(size->handle);}size->handle = handle;f->populated = true;
}

最后总结下:实际上,Qt在不同的平台,分别调用的当前系统的库函数,来封装成统一的接口提供给开发者,我们不需要自己再引用对应的库了。这才是Qt的理念。
Code Less

公众号:Qt那些事儿

公众号

这篇关于Qt源码那些事儿-Linux下Qt的QFontDataBase字体引擎解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/983574

相关文章

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer

Qt中实现多线程导出数据功能的四种方式小结

《Qt中实现多线程导出数据功能的四种方式小结》在以往的项目开发中,在很多地方用到了多线程,本文将记录下在Qt开发中用到的多线程技术实现方法,以导出指定范围的数字到txt文件为例,展示多线程不同的实现方... 目录前言导出文件的示例工具类QThreadQObject的moveToThread方法实现多线程QC

Linux中的自定义协议+序列反序列化用法

《Linux中的自定义协议+序列反序列化用法》文章探讨网络程序在应用层的实现,涉及TCP协议的数据传输机制、结构化数据的序列化与反序列化方法,以及通过JSON和自定义协议构建网络计算器的思路,强调分层... 目录一,再次理解协议二,序列化和反序列化三,实现网络计算器3.1 日志文件3.2Socket.hpp

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

Linux之UDP和TCP报头管理方式

《Linux之UDP和TCP报头管理方式》文章系统讲解了传输层协议UDP与TCP的核心区别:UDP无连接、不可靠,适合实时传输(如视频),通过端口号标识应用;TCP有连接、可靠,通过确认应答、序号、窗... 目录一、关于端口号1.1 端口号的理解1.2 端口号范围的划分1.3 认识知名端口号1.4 一个进程

Linux权限管理与ACL访问控制详解

《Linux权限管理与ACL访问控制详解》Linux权限管理涵盖基本rwx权限(通过chmod设置)、特殊权限(SUID/SGID/StickyBit)及ACL精细授权,由umask决定默认权限,需合... 目录一、基本权限概述1. 基本权限与数字对应关系二、权限管理命令(chmod)1. 字符模式语法2.

在Linux系统上连接GitHub的方法步骤(适用2025年)

《在Linux系统上连接GitHub的方法步骤(适用2025年)》在2025年,使用Linux系统连接GitHub的推荐方式是通过SSH(SecureShell)协议进行身份验证,这种方式不仅安全,还... 目录步骤一:检查并安装 Git步骤二:生成 SSH 密钥步骤三:将 SSH 公钥添加到 github

Linux线程同步/互斥过程详解

《Linux线程同步/互斥过程详解》文章讲解多线程并发访问导致竞态条件,需通过互斥锁、原子操作和条件变量实现线程安全与同步,分析死锁条件及避免方法,并介绍RAII封装技术提升资源管理效率... 目录01. 资源共享问题1.1 多线程并发访问1.2 临界区与临界资源1.3 锁的引入02. 多线程案例2.1 为

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

Oracle数据库定时备份脚本方式(Linux)

《Oracle数据库定时备份脚本方式(Linux)》文章介绍Oracle数据库自动备份方案,包含主机备份传输与备机解压导入流程,强调需提前全量删除原库数据避免报错,并需配置无密传输、定时任务及验证脚本... 目录说明主机脚本备机上自动导库脚本整个自动备份oracle数据库的过程(建议全程用root用户)总结