【Android】TextView的文字长度测量及各种padding解析

本篇文章已授权微信公众号 安卓巴士Android开发者门户 独家发布

老规矩,先上张图
o,这篇好像是分析篇,没有效果图。不管了,位置占着,老规矩不能坏,下面开始正文。


这篇博客会讲得比较杂:

  1. TextView里各种padding的含义?
  2. 如何计算每行文字的长度?
  3. 设置android:maxLines="1"和android:singleLine="true"有什么区别?
  4. 为什么设置android:maxLines="1"时TextView的跑马灯效果就不能正常工作?

TextView里各种padding的含义?

虽然我们平时只用TextView显示纯文本数据,但其实TextView支持设置Background,四周的drawable小图标,以及Span数据比如文本或图片。在Android里不管是什么控件都是占据一个矩形的空间,各个矩形之间基本都可以设置padding属性,这样一来TextView就被分成多个矩形区域了,而TextView也提供了获取各种padding和长度的接口,如下:

getWidth(), getHeight()
getPaddingLeft/Right/Top/Bottom()
getCompoundPaddingLeft/Right/Top/Bottom()
getExtendedPaddingBottom/Top()
getTotalPaddingLeft/Right/Top/Bottom()

先看一下布局代码:

6.png

@drawable/icon_person_n
7.png

下面是效果图,做了一些标注:

8.png

下面来看一下上述获取长度的接口的数据,你可以使用打日志方式,我是比较习惯用调试方式:

9.png

所以,总结一下:
getWidth(), getHeight():对应你代码里的layout_width和layout_height。

getPaddiingLeft/Right/Top/Bottom():对应代码里的Padding。
以上两个比较容易理解,毕竟经常打交道。

getCompoundPaddingLeft/Top/Right/Bottom(): 翻译成中文就是获取混合的Padding, 既然是混合的,那么它的值也就是padding + 图片的大小 + drawablePadding的值。说得通俗点就是,它是获取文字区域到TextView边界之间的间隔。附上源码:

10.png

getExtendedPaddingTop():这个是当有部分文字没有显示出来时,也就是设置了maxLine时,它的值就等于首行文字到TextView顶端的距离。同理,getExtendedPaddingBottom()就是最后一行文字到TextVeiw底部距离。其他情况下,他的值等于getCompoundPaddingTop/Bottom()的值。这个源码不多,但也不怎么好讲解,就贴两张图对比下,就明白了。

11.png

12.png

getTotalPaddingLeft/Right/Top/Bottom():翻译下就是获取总的Padding值,看了下源码,左右的值直接就是等于compoundPadding的值,上下的值等于ExtendedPadding的值再加上offset的值(跟Gravity的垂直方向的布局有关。说得通俗点就是,不管有没有maxLines,上下的值都分别等于首行到TextView顶端和末行到TextView底部的值。

这些接口除了前面两个比较常用外,其他基本很少用吧,我也是因为在看TextView的跑马灯部分的源码才接触到,然后为了弄明白才记录下来的。至于后面那些接口的应用场景,getCompoundPadding()这个的应用场景倒是很明确,可以用来判断相应的drawable是否发生点击事件之类的需求。至于extendedPadding和totalPadding这两个的应用场景,我想了想,觉得应该是涉及需要计算显示出来后的文字高度的相关需求时会用到吧。有对这些接口很熟悉的童鞋可以分享出来哈,一起学习学习。

最后用一张图总结一下,我把TextView分成内容区域,内容区域和TextView边界之间的间隔就是padding的值,内容区域包括drawable区域和文字区域,drawable区域和文字区域之间的间隔就是drawablePadding的值,文字区域和TextView之间的间隔就是CompoundPadding的值。

13.png

如何计算每行文字的长度?

Q:每行文字的长度不就等于TextView的宽度吗?直接getWidth()不就好了?
A:再看一下上面那部分内容你就清楚了,只有当TextView宽度设置为wrap_content,且没有背景图或drawable时,文字的长度才等于getWidth();当文字很少,没有填充满时,或是溢出时,文字的长度都得另外计算。

Q:每行文字的长度不一样长吗?
A:因为TextView有自己的换行策略,如下图所示,显然每行的文字长度不一样长。

14.png

Q:文字的长度是指哪段长度?
A:看需求吧,我觉得通常情况下都是只需要计算显示在屏幕上的可见区域的每行文字的长度即可。还有那么一种需求,当设置了溢出内容用...表示时,那么其实每行文字的实际长度就不止可见区域那么长了。

那么该如何计算文字的长度呢?单单根据上一部分里的各种Padding值肯定不够,根据各种Padding顶多计算出文字区域的宽度,但实际上每一行文字并不会那么刚刚好占满文字区域的宽度,那么就还得借助其他来进行计算。

方法1:TextView.getPaint().measureText(String text)

15.png

但这种方法只是测试传入的text在该TextView的配置下的总长度,并不是计算每一行的长度。

方法2:TextView.getLayout().getLineWidth(int line)

16.png

TextView对应的是图14,正好,利用方法1验证一下,这个方法计算得到的是不是每行文字的长度。
17.png

完全正确,所以说这个方法确实计算得到的是每一行文字的实际长度,注意这里是实际长度,也就是说当设置singleLine属性时,用这个方法测量得到的是一整行文字的长度,包括溢出部分。

设置android:maxLines="1"和android:singleLine="true"有什么区别?

官方是推荐说不要再使用singleLine,用maxLines="1"代替。但其实这两个的效果是不一样的,官方api接口里有说明,都是英文我就不贴图了,大概翻译下:
maxLines:限制TextView的最高高度,大概就是指通过限制行数来限制最高高度。
singleLine: 强制设置TextView的文字不换行。

区别就是:maxLines还是会默认自动进行换行策略,假如一段文字自动换行后有5行,maxLines设置为1,那么就只显示第一行的内容,其他行不显示。
但是,如果是设置了singleLine, 那么这段可以有5行的文字将会被强制放在1行里,然后看最多能显示多少字符,剩下的不显示。

这样的区别就是导致了很多人在使用TextVeiw的跑马灯效果时不能正常工作的状态,所以下面单独列出个问题来讲。

为什么设置android:maxLines="1"时TextView的跑马灯效果就不能正常工作?

明白了maxLines="1"和singleLine的区别后,只要再明白跑马灯的原理,就很容易理解为什么设置成maxLines="1"时跑马灯不工作了。我在上一篇博客里写过跑马灯启动的条件,具体的分析可以去上一篇看,这里大概说下。

跑马灯要启动要同时满足四个条件,其中有一个条件就是这一行的文字长度要大于文字区域的宽度,文字区域的宽度就是TextView的getWidth()扣去ComPoundpaddingLeft再扣去CompoundPaddingRight剩下的长度。
如果是maxLines="1"的话,那么就像上一问中分析的那样,所有的文字其实已经被自动换行了,只显示第一行,而换行是什么,就是为了让每行文字的长度超过文字区域的宽度才进行的换行,也就是说,如果一段文字经过TextView的换行后,那么每行的文字长度都不会超过文字区域的长度。这样一来,自然就不满足跑马灯的启动条件之一了,跑马灯也就不能正常工作了。
singleLine的话,则是不会对一段文字进行换行处理,这样一来,自然就超过了文字区域的长度,所以如果要设置跑马灯效果的话,只能用singleLine不能用maxLines="1"。


QQ图片20180316094923.jpg

最近刚开通了公众号,想激励自己坚持写作下去,初期主要分享原创的Android或Android-Tv方面的小知识,感兴趣的可以点一波关注,谢谢支持~~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349

推荐阅读更多精彩内容