首页 > 编程语言 >Android文字基线Baseline算法的使用讲解

Android文字基线Baseline算法的使用讲解

时间:2022-08-22 12:13:47浏览次数:74  
标签:文字 Baseline 距离 基线 Android 绘制 Descent

引言

Baseline是文字绘制时所参照的基准线,只有先确定了Baseline的位置,我们才能准确的将文字绘制在我们想要的位置上。Baseline的概念在我们使用TextView等系统控件直接设置文字内容时是用不到的,但是如果我们想要在Canvas画布上面绘制文字时,Baseline的概念就必不可少了。

我们先了解一下Android中Canvas画布绘制文字的方法,如下图:

参数示意:

text,文字内容 x,文字从画布上开始绘制的x坐标(Canvas是一个原点在左上角的平面坐标系) y,Baseline所在的y坐标,不少人已开始以为y是绘制文字区域的底部坐标,其实是不正确的,这是两个概念 paint,画笔,设置的文字的大小颜色等属性 了解了文字绘制的方法,我们现在就了解一下这个参数y(Baseline)的计算方法。

Baseline的概念

我们先看一行文字各区域的分布示意图

从上图来看,Baseline不难理解,它就是E和h的下边界线。我们还可以得出一个结论,文字的高度=Descent+Ascent

然而,上面这个公式并不完全准确,我们再看一个图:

我们看到,如果文字的上方有一些特殊的符号,比如上图中的~或者是我们汉语拼音中的声调时,文字区域又会多出一部分Leading。

因此,完整的公式应该是

文字的高度=Descent+Ascent+Leading。

那么,为什么第一幅图中没有说明Leading的存在呢,原因是我们通常在绘制一行英文或者中文时,Leading的高度为0。我们看一个证据图,下图是在绘制英文文字时调试取得的数据。

其中leading=0,所以我们在文字绘制时不需要考虑Leading,图中的数值都是距离Baseline的距离,在Baseline上方为负值,下方为正值。

Baseline位置(y轴坐标)的计算

为了方便我们对计算过程进行理解,我画了一幅帮助图,如下:

假设我们是在画布Canvas的顶部绘制一行文字,规定一行文字的高度是y,文字区域的高度是Height(TOP和BOTTOM之间,TOP到0和BOTTOM到y的距离相等,这样文字才看起来是居中)。因此,0到y和TOP到BOTTOM的中线是重合的,y轴坐标都是y/2。

我们要绘制一行文字时,设计必然会告诉我们0到y的距离,所以中线的位置也是固定的y/2,那么我们设置了Paint的文字大小后,Ascent和Descent又能直接得到,就可以算出中线到基线的距离,公式如下:

基线到中线的距离=(Descent+Ascent)/2-Descent

注意,实际获取到的Ascent是负数。公式推导过程如下:

中线到BOTTOM的距离是(Descent+Ascent)/2,这个距离又等于Descent+中线到基线的距离,即(Descent+Ascent)/2=基线到中线的距离+Descent。

有了基线到中线的距离,我们只要知道任何一行文字中线的位置,就可以马上得到基线的位置,从而得到Canvas的drawText方法中参数y的值。

Android获取中线到基线距离的代码,Paint需要设置文字大小textsize。

1 2 3 4 5 6 7 8 9 10 11 /**  * 计算绘制文字时的基线到中轴线的距离  *  * @param p  * @param centerY  * @return 基线和centerY的距离  */ public static float getBaseline(Paint p) {   FontMetrics fontMetrics = p.getFontMetrics();   return (fontMetrics.descent - fontMetrics.ascent) / 2 -fontMetrics.descent; }

标签:文字,Baseline,距离,基线,Android,绘制,Descent
From: https://www.cnblogs.com/Free-Thinker/p/16612394.html

相关文章

  • Java 断点下载(下载续传)服务端及客户端(Android)代码
    原文:Java断点下载(下载续传)服务端及客户端(Android)代码-Stars-One的杂货小窝最近在研究断点下载(下载续传)的功能,此功能需要服务端和客户端进行对接编写,本篇也是......
  • 关于Android中接口Reponse反序列化的思考
    总结kotlindataclass返回值最佳实践,使用ks!字段全写默认值,不可空,取消全部问号!ks速度更快并且支持N多数据类型protobuf+jsonkt默认值只有两种方式可以获取使用k......
  • Compose和AndroidView的交互
    1、在ComposeUI中加载AndroidView控件Compose中可以加载AndroidView还是比较简单的,直接引入AndroidView来加载AndroidView布局文件。@ComposablefunGreetin......
  • Android消息机制解析
    Android消息机制解析为什么主线程中可以直接使用Handler?Handler的运行需要底层的MessageQueue和Looper支撑,MessageQueue是以单链表为数据结构的消息列表,Looper以......
  • Android四大组件——Service——和Activity的通信
    任务描述:在MyService里提供一个下载功能,然后再Activity中可以决定何时开始下载,以及随时查看下载进度。按照之前的思路:在MyService中创建一个方法,用以对下载进行管理。然后......
  • Android自定义Dialog
    开发时我们可能需要一些非全屏view的一些提示信息,或者不想去创建需要在清单文件注册的Activity来显示view,那么我们就需要借助一些其他的窗体子类来完成需求如:Dialog,PopuW......
  • 直播商城源码,Android RecycleView 加载网络图片
    直播商城源码,AndroidRecycleView加载网络图片1.布局 <android.support.v7.widget.RecyclerView  android:id="@+id/recyclerView"  android:layout_width="m......
  • 解决 Android真机调试 卸载后无法再次安装成功
    1、配置环境变量androidsdk中的tools目录和platform-tools目录都配置到系统环境变量path中; 2、运行cmd输入命令adbdevices查看当前连接设备的机器编号;......
  • Android生成密钥散列
    接入facebook登录和分享时需要在facebook后台添加密钥散列,下面是生成方式第一种(简单,准确)记住要用相应的签名文件进行签名哦try{PackageInfoinf......
  • Android接入firebase
    android的出海应用借用firebase来进行数据统计分析和应用的崩溃报告,下面是firebase的接入步骤:google提供了两种接入方式,我们讲第二种通过androidstudio的菜单AndroidSt......