WPF 测试 GlyphTypeface 的 Baseline 行为

news/2025/1/8 7:04:53/文章来源:https://www.cnblogs.com/lindexi/p/18658843

本文内容里面只给出关键代码片段,如需要全部的项目文件,可到本文末尾找到本文所有代码的下载方法

前置博客: WPF 简单聊聊如何使用 DrawGlyphRun 绘制文本

大飞哥来问我关于文本行距和基线问题,在之前某只不远透露姓名的牛写了一段有趣的代码,设定了行距计算里面包含 1/5 的魔法数字。我开始猜测是基线计算的问题,结果一顿计算发现数据差异过大,没有解决开始的问题,只好将我测试的 GlyphTypeface 的 Baseline 行为记录

在 WPF 里面,可以通过 FontFamily 根据字体名字符串获取到 GlyphTypeface 对象,大概的代码如下

        var fontFamily = new FontFamily("微软雅黑");Typeface typeface = fontFamily.GetTypefaces().First();var success = typeface.TryGetGlyphTypeface(out GlyphTypeface glyphTypeface);if (!success){Debug.Fail("微软雅黑字体找不到");}

我尝试绘制一段文本,内容是“文本测试afgjqiWHXx”

这段文本的特征是中英文混排,且英文字符有穿越基线的字符

我尝试按照 WPF 简单聊聊如何使用 DrawGlyphRun 绘制文本 博客提供的方法构建 GlyphRun 进行绘制,代码如下

        var fontSize = 30;var text = "文本测试afgjqiWHXx";var glyphIndexList = new List<GlyphInfo>();for (var i = 0; i < text.Length; i++){var codePoint = (int) text[i]; // 这里的 Code Point 没有处理 Emoji 的高低代理字符if (glyphTypeface.CharacterToGlyphMap.TryGetValue(codePoint, out var glyphIndex)){var width = glyphTypeface.AdvanceWidths[glyphIndex] * fontSize;var height = glyphTypeface.AdvanceHeights[glyphIndex] * fontSize;glyphIndexList.Add(new GlyphInfo(glyphIndex, width, height));}else{// 进入字体回滚}}var pixelsPerDip = (float) VisualTreeHelper.GetDpi(this).PixelsPerDip;var baseline = glyphTypeface.Baseline * fontSize;var location = new Point(0, baseline);drawingContext.PushGuidelineSet(new GuidelineSet([0], [baseline]));var defaultXmlLanguage =XmlLanguage.GetLanguage(CultureInfo.CurrentUICulture.IetfLanguageTag);var glyphRun = new GlyphRun(glyphTypeface,bidiLevel: 0,isSideways: false,renderingEmSize: fontSize,pixelsPerDip: pixelsPerDip,glyphIndices: glyphIndexList.Select(t => t.GlyphIndex).ToList(),baselineOrigin: location, // 设置文本的偏移量advanceWidths: glyphIndexList.Select(t => t.AdvanceWidth).ToList(), // 设置每个字符的字宽,也就是字号glyphOffsets: null, // 设置每个字符的偏移量,可以为空characters: text.ToCharArray(),deviceFontName: null,clusterMap: null,caretStops: null,language: defaultXmlLanguage);drawingContext.DrawGlyphRun(Brushes.Black, glyphRun);

我尝试使用 DrawLine 将 baseline 的值绘制出来,代码如下

        drawingContext.DrawLine(new Pen(Brushes.Black,1), new Point(0, baseline), new Point(300, baseline));

运行代码,可见画出来的线条就刚好是文本的基线,非常正确

如此可证明将 GlyphTypeface 的 Baseline 属性乘以字号就是文本字符的基线

那 GlyphTypeface 的 Baseline 属性和 FontFamily 的有什么不同?绝大部分字体这两个属性都是相同的,但是由于字体可能存在加粗斜体等,为了更好的视觉呈现,确实存在不同的情况。有些 GlyphTypeface 和 FontFamily 存在不相同的 Baseline 属性

对于最终渲染来说,就应该获取对应的 GlyphTypeface 的基线。但由于 GlyphTypeface 和 FontFamily 的基线基本相差不大,也可以放心直接就用 FontFamily 的基线就好。毕竟在很多文本排版里面,是不期望只是加粗或带斜体一下,就让字体在行内上浮下沉

对于一些字体设计师来说,会特别修改加粗的基线,虽然从排版数值上让字体下沉,但视觉效果却刚好看起来是顺着的。从这个思路上说,拿 GlyphTypeface 的基线是更加正确的

通过 FormattedText 获取到的 Baseline 基本等于 FontFamily 的 Baseline 乘以字号,可能会和 GlyphTypeface 的不相同,如以下代码片段

                    var text = "1";var fontSize = 30;var formattedText = new FormattedText(text, CultureInfo.CurrentCulture,System.Windows.FlowDirection.LeftToRight, typeface, fontSize, Brushes.Black, pixelsPerDip);var sameGlyphTypefaceAndFormattedText = Math.Abs(formattedText.Baseline - glyphTypeface.Baseline * fontSize) < 0.01; var sameFontFamilyAndFormattedText = Math.Abs(formattedText.Baseline - fontFamily.Baseline * fontSize) < 0.01;

在我设备上的所有字体都是 sameFontFamilyAndFormattedText 为 true 的值。即如果只是想通过 FormattedText 获取基线,那完全和使用 FontFamily 的 Baseline 乘以字号是等价的

通过阅读 WPF 源代码,可以理解到 FormattedText 的 Baseline 为什么和 FontFamily 几乎等价,原因是 FormattedText 的 Baseline 是从首行 TextLine 的 Baseline 获取到的。在 SimpleTextLine 类型里面的 Baseline 属性定义如下

SimpleRun run = (SimpleRun)runs[count];
var realAscent = Math.Max(realAscent, run.Baseline);
_baselineOffset = formatter.IdealToReal(TextFormatterImp.RealToIdeal(realAscent), PixelsPerDip);/// <summary>/// Client to get the distance from top to baseline of this text line/// </summary>public override double Baseline{get { return _baselineOffset; }}

而 SimpleRun 的 Baseline 定义如下

        internal double Baseline{get{if (Ghost || EOT)return 0;return TextRun.Properties.Typeface.Baseline(TextRun.Properties.FontRenderingEmSize, 1, _pixelsPerDip, _textFormatterImp.TextFormattingMode);}}

可见是进入到 Typeface 的 Baseline 方法里面

    public class Typeface{internal double Baseline(double emSize, double toReal, double pixelsPerDip, TextFormattingMode textFormattingMode){return CachedTypeface.FirstFontFamily.Baseline(emSize, toReal, pixelsPerDip, textFormattingMode);            }}

如此可以看到,绕了一圈还是回到了 IFontFamily 的 Baseline 方法。来对比一下 FontFamily 类型的 Baseline 属性,以及 IFontFamily 接口的 PhysicalFontFamily 实现的 Baseline 方法

    public class FontFamily{internal IFontFamily FirstFontFamily { get; }public double Baseline{get{return FirstFontFamily.BaselineDesign;}set{VerifyMutable().SetBaseline(value);}}}internal sealed class PhysicalFontFamily : IFontFamily{double IFontFamily.BaselineDesign{get{return ((IFontFamily)this).Baseline(1, 1, 1, TextFormattingMode.Ideal);}}double IFontFamily.Baseline(double emSize, double toReal, double pixelsPerDip, TextFormattingMode textFormattingMode){if (textFormattingMode == TextFormattingMode.Ideal){return emSize * _family.Metrics.Baseline;}else{double realEmSize = emSize * toReal;return TextFormatterImp.RoundDipForDisplayMode(_family.DisplayMetrics((float)(realEmSize), checked((float)pixelsPerDip)).Baseline * realEmSize, pixelsPerDip) / toReal;}}}

如此可见 FormattedText 走的逻辑和 FontFamily 基本相同,只有一些数值上的差异而已

本文代码放在 github 和 gitee 上,可以使用如下命令行拉取代码。我整个代码仓库比较庞大,使用以下命令行可以进行部分拉取,拉取速度比较快

先创建一个空文件夹,接着使用命令行 cd 命令进入此空文件夹,在命令行里面输入以下代码,即可获取到本文的代码

git init
git remote add origin https://gitee.com/lindexi/lindexi_gd.git
git pull origin bbde4f7c3873aac83c466762ac12ae37a3dccfa4

以上使用的是国内的 gitee 的源,如果 gitee 不能访问,请替换为 github 的源。请在命令行继续输入以下代码,将 gitee 源换成 github 源进行拉取代码。如果依然拉取不到代码,可以发邮件向我要代码

git remote remove origin
git remote add origin https://github.com/lindexi/lindexi_gd.git
git pull origin bbde4f7c3873aac83c466762ac12ae37a3dccfa4

获取代码之后,进入 WPFDemo/LahallgucheHichawaki 文件夹,即可获取到源代码

更多技术博客,请参阅 博客导航

附录:

以下是我使用如下代码跑出来的基线集合

        var pixelsPerDip = VisualTreeHelper.GetDpi(this).PixelsPerDip;foreach (FontFamily? fontFamily in System.Windows.Media.Fonts.SystemFontFamilies){if (!fontFamily.FamilyNames.TryGetValue(XmlLanguage.GetLanguage("zh-CN"),out var name)){name = fontFamily.Source;}foreach (var typeface in fontFamily.GetTypefaces()){var typefaceName = typeface.FaceNames.First().Value;if (typeface.TryGetGlyphTypeface(out GlyphTypeface? glyphTypeface)){var text = "1";var fontSize = 30;var formattedText = new FormattedText(text, CultureInfo.CurrentCulture,System.Windows.FlowDirection.LeftToRight, typeface, fontSize, Brushes.Black, pixelsPerDip);var sameGlyphTypefaceAndFormattedText = Math.Abs(formattedText.Baseline - glyphTypeface.Baseline * fontSize) < 0.01; var sameFontFamilyAndFormattedText = Math.Abs(formattedText.Baseline - fontFamily.Baseline * fontSize) < 0.01;Debug.WriteLine($"""字体名: {name} - {typefaceName}斜体: {glyphTypeface.Style}加粗: {glyphTypeface.Weight}拉伸: {glyphTypeface.Stretch}基线 FontFamily: {fontFamily.Baseline}基线 GlyphTypeface: {glyphTypeface.Baseline}基线 FormattedText: {formattedText.Baseline / fontSize}基线相同 FontFamily == GlyphTypeface: {fontFamily.Baseline == glyphTypeface.Baseline}基线相近 GlyphTypeface ~ FormattedText: {sameGlyphTypefaceAndFormattedText}基线相近 FontFamily ~ FormattedText: {sameFontFamilyAndFormattedText}""");}}}

输出内容如下,也欢迎大家在自己的设备上运行以上代码

字体名: 更纱终端书呆黑体-简 - Regular
斜体: Normal
加粗: Normal
拉伸: Normal
基线 FontFamily: 0.965
基线 GlyphTypeface: 0.965
基线 FormattedText: 0.9650000000000001
基线相同 FontFamily == GlyphTypeface: True
基线相近 GlyphTypeface ~ FormattedText: True
基线相近 FontFamily ~ FormattedText: True字体名: 汉仪南宫体简 - Regular
斜体: Normal
加粗: Normal
拉伸: Normal
基线 FontFamily: 0.998046875
基线 GlyphTypeface: 0.859375
基线 FormattedText: 0.998
基线相同 FontFamily == GlyphTypeface: False
基线相近 GlyphTypeface ~ FormattedText: False
基线相近 FontFamily ~ FormattedText: True字体名: 汉仪南宫体简 - Oblique
斜体: Oblique
加粗: Normal
拉伸: Normal
基线 FontFamily: 0.998046875
基线 GlyphTypeface: 0.859375
基线 FormattedText: 0.998
基线相同 FontFamily == GlyphTypeface: False
基线相近 GlyphTypeface ~ FormattedText: False
基线相近 FontFamily ~ FormattedText: True字体名: 汉仪南宫体简 - Bold
斜体: Normal
加粗: Bold
拉伸: Normal
基线 FontFamily: 0.998046875
基线 GlyphTypeface: 0.859375
基线 FormattedText: 0.998
基线相同 FontFamily == GlyphTypeface: False
基线相近 GlyphTypeface ~ FormattedText: False
基线相近 FontFamily ~ FormattedText: True字体名: 方正硬笔行书简体_非压缩版 - Regular
斜体: Normal
加粗: Normal
拉伸: Normal
基线 FontFamily: 0.82421875
基线 GlyphTypeface: 0.76953125
基线 FormattedText: 0.8242222222222222
基线相同 FontFamily == GlyphTypeface: False
基线相近 GlyphTypeface ~ FormattedText: False
基线相近 FontFamily ~ FormattedText: True...

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/865699.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OpenVX功能参数分析

3. 节点参数节点创建函数的参数定义为原子类型,如vx_int32、vx_enum,或对象,如vx_scalar、vx_image。框架应将节点创建功能的原子变量转换为vx_scalar引用,供节点使用。vx_scalar类型的节点参数,可以在图形执行期间更改;而如果改变原子类型的节点参数(vx_int32等),则至…

OpenVX参数节点执行独立性

8.节点执行独立性 在如图2-5所示例中,客户端根据输入图像、梯度幅度和梯度相位。OpenVX并不要求并行运行,但可以由OpenVX供应商实现。 图2-5 具有一些独立节点的简单图形。 构造这种图形的代码,如下所示。 vx_context context = vxCreateContext(); vx_image images[] = {…

4书推荐《AI芯片开发核心技术详解》、《智能汽车传感器:原理设计应用》、《TVM编译器原理与实践》、《LLVM编译器原理与实践》

4本书推荐《AI芯片开发核心技术详解》、《智能汽车传感器:原理设计应用》、《TVM编译器原理与实践》、《LLVM编译器原理与实践》由清华大学出版社资深编辑赵佳霓老师策划编辑的新书《AI芯片开发核心技术详解》已经出版,京东、淘宝天猫、当当等网上,相应陆陆续续可以购买。该…

BOF编写-修改时间戳

模板配置 跟着网上的教程使用evilashz师傅的模板,下载模板解压至vs的模板目录: %UserProfile%\Documents\Visual Studio 2022\Templates\ProjectTemplates​​ 创建新项目选择刚刚新增的类型:Beacon Object File​。 ​​ ‍ 环境适配 生成时报错,我使用的是2022版本的,模…

【分布式系统】 分布式调度 设计,要考虑 哪些问题?

一、分布式调度框架的核心功能二、什么是 任务调度 ?三、什么是分布式 任务调度 ?四、分布式调度框架的主要功能五、分布式调度框架的核心业务场景六、分布式任务调度的核心组件七、分布式任务调度的架构模式八、常见的分布式调度框架包括九、XXL-Job的工作流程十、 Quartz 的…

一文搞懂L1-L3业务流程体系分析,价值流、端到端流程、职能流程

今天我们聊聊业务流程体系分析这个话题。 业务流程的概念 业务流程是企业为实现目标而制定的一套系统化的工作方法。它由一系列有序的业务活动组成,按照既定规则将资源(输入)转化为有价值的结果(输出)。 在业务架构分析阶段,业务流程发挥着关键作用: • 明确业务运作的方…

[SSL/TLS/PKI] 基于`Let’s Encrypt`,获取免费SSL证书

缘起:云厂商卖的SSL证书太贵 概述:Let’s Encrypt 支持申请免费SSL证书的CA机构支持提供免费SSL证书的CA机构有:Let’s Encrypt ZeroSSL SSL For Free Cloudflare BuyPass AWS本文我们来试试 Let’s Encrypt 怎么用。https://letsencrypt.org/zh-cn/getting-started/Let’s …

.NET 响应式编程 System.Reactive 系列文章(二):深入理解 IObservableT 和 IObserverT

在 Rx 中,数据流的生产和消费是通过 观察者模式(Observer Pattern) 实现的。这种模式定义了两种角色: - IObservable.NET 响应式编程 System.Reactive 系列文章(二):深入理解 IObservable<T> 和 IObserver<T>引言:为什么我们调整了学习顺序? 在上一篇文章…

WebPack站点实战(一)

以下文章来源于一位不愿透露姓名的热心网友 ,作者不愿透露姓名的热心网友 文章配套B站视频,很多话语简略了,建议配着视频看。 地址:https://www.bilibili.com/video/BV13F411P7XB/ 开始之前了,简单过一下下面几个方法加深印象,便于更好理解加载器。也可以直接从webpack标…

CUDA概念

1.1.0f加个f,成单精度计算,不加会默认成double 2.快速指令:__add()加两个下划线 3.CUDA计算能力1.3以上才支持双精度,4.0往后支持双精度浮点计算 单精度浮点型(float )专指占用32位存储空间的单精度(single-precision )值。单精度在一些处理器上比双精度更快而且只占用…

12.09百度机器翻译SDK实验

一、实验要求百度机器翻译SDK实验(2024.11.15日完成) 任务一:下载配置百度翻译Java相关库及环境(占10%)。 任务二:了解百度翻译相关功能并进行总结,包括文本翻译-通用版和文本翻译-词典版(占20%)。 任务三:完成百度翻译相关功能代码并测试调用,要求可以实现…

弹性波动力学笔记(十)罗格里德斯公式推导

在应力计算中大量需要轴旋转公式计算,因此本笔记给出了罗格里德斯轴旋转公式Note: Derivation of the Rodriguez Formula In this Note, we will derive a formula for \(\mathbf{R}(\widehat{\mathbf{n}},\theta)\) . Consider the three dimensional rotation of a vecto…