一起来学Kotlin~我爱编程程序猿阵线联盟-汇总各类技术干货

Android 浅谈 maxEms 属性

2018-05-12  本文已影响149人  阿策神奇

      小菜最近调整一个小需求,为了整体显示效果,需要限制一部分文字的长度,超过部分以...代替。
      小菜本想偷个懒,用 android:maxLength="6" 属性配合 android:maxLines="1" 以及 android:ellipsize="end" 来实现,但是只可限制字符床度为6,没有省略号。然后想起有一个 android:maxEms="6" 属性来实现,默认超过长度以省略号结束。结果发现并非按字符长度计算,小菜还是太天真了。

android:maxEms="6"

Tips1: android:singleLine="true" 属性已经在 API 中不建议使用,小菜在现有的设备中测试与 android:maxLines="1" 属性效果完全一致。
Tips2: 在使用 android:maxEms="6" 属性时, TextView 的宽度需为 wrap_content 方式。


以下是小菜测试时遇到的问题:
左侧是从 maxEms = “1” 开始递增到 “16”,右侧是测量文字所占的宽度:


纯汉字 纯字母
纯数字

测试发现:

  1. 无论是文字还是字母或是数字,设置完 maxEms 之后,文字所占的宽度是一致的,随着 maxEms 的递增,文字的宽度也是相同幅度递增的;
  2. 不管是文字还是字母或数字,都不是单纯的按照字符个数来展示的,而是所占屏幕的宽度,所以并不是网上一些朋友说的显示内容为 maxEms - 1 +”...“。
  3. 若限制字符串长度请尝试 maxLength,若字号不变,限制文字所在屏幕宽度,可尝试 maxEms。

TextView 源码中 maxEms

/**
* Makes the TextView at most this many ems wide
*
* @attr ref android.R.styleable#TextView_maxEms
*/
@android.view.RemotableViewMethod
public void setMaxEms(int maxems) {
    mMaxWidth = maxems;
    mMaxWidthMode = EMS;
    requestLayout();
    invalidate();
}
/**
* @return the maximum width of the TextView, expressed in ems or -1 if the maximum width
* was set in pixels instead (using {@link #setMaxWidth(int)} or {@link #setWidth(int)}).
* 文本视图的最大宽度,以EMS表示,或如果宽度为1,则表示最大宽度
* 设置为像素(使用{@ Link LyStMax宽度(int)}或{@ Link LyStSuffelt(int)})
* @see #setMaxEms(int)
* @see #setEms(int)
*
* @attr ref android.R.styleable#TextView_maxEms
*/
public int getMaxEms() {
    return mMaxWidthMode == EMS ? mMaxWidth : -1;
}

小菜查阅相关资料以及自己的理解是:

  1. em 是字体宽度的排版单位,16 点字体中的一个是 16 分;
  2. em 和 ex 单元取决于字体,并且对于文档中的每个元素可能不同。em 只是字体大小。在具有 2in 字体的元素中,1em 因此意味着 2in。在 em 中表示大小,例如边距和填充,意味着它们与字体大小有关,并且如果用户有大字体(例如,在大屏幕上)或小字体(例如,在手持设备上),大小将成比例。
  3. 它是字母 M 在给定的英语字体大小中的宽度。所以 2em 是这个字体中字母 M 的两倍。字体不同于英语,它是这个字体中最宽的字母宽度,这个宽度是不同的像素大小,然后是英语字体中的 M 的宽度大小,但是它仍然是 1EM。所以如果我用 12sp 的英文字体使用文本,1M 相对于这个 12sp 的英语字体,用意大利字体加上1。

测试主要代码:

// xml 中 TextView
<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:ellipsize="end"
    android:maxEms="6"
    android:maxLines="1"
    android:text="@string/test_str1" />
// Kotlin 获取文字宽度
fun getTextViewWidth(tv: TextView): String {
    val spec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED)
    tv.measure(spec, spec)
    val measuredWidthTicketNum = tv.getMeasuredWidth()
    return measuredWidthTicketNum.toString()
}

      下面是小菜的公众号,欢迎闲来吐槽~


小菜公众号
上一篇下一篇

猜你喜欢

热点阅读