EllipsizeEndTextview末尾省略自定义View

需求:在某些场景下,直接在xml或者代码里面设置android:ellipsize="end"不生效,可以参考这种解决方案,但是会存在截断单词的场景

原理:默认开启自动截取功能autoTruncate = true,可以在xml中设置关闭,需要设置两次super.setText(text, type)文本,第一次设置,是为了拿到的TextView的最大宽度,当文本宽度,大于view的最大宽度,才开始计算截取点,当大于时通过paint.breakText计算截取点,但是计算出的截取点和'...'拼接之后会出现最后一个点显示半个情形,大概原因是因为计算误差导致的,所以在计算的时候多减了一个.的位置距离,后续补充空格处理

使用:同TextView的使用方式一样

自定义View具体实现:

kotlin 复制代码
import android.content.Context
import android.util.AttributeSet
import android.view.ViewGroup
import androidx.appcompat.widget.AppCompatTextView


/**
 * 需要设置两次text才能拿到具体的宽度值
 * 相较之前版本算法上有提升
 * 考虑到代码入侵性,添加了属性做改造
 * autoTruncate = false关闭截取功能,当作一个普通的TextView使用
 */
class EllipsizeEndTextview(context: Context, attributeSet: AttributeSet?) :
    AppCompatTextView(context, attributeSet) {

    //是否自动截取
    private var autoTruncate = true

    //内部缓存一个原始值,方便外部获取
    private var originalText: CharSequence? = ""

    init {
        val a = context.obtainStyledAttributes(attributeSet, R.styleable.EllipsizeEndTextView)
        autoTruncate = a.getBoolean(R.styleable.EllipsizeEndTextView_autoTruncate, true)
        a.recycle()
    }

    override fun setText(text: CharSequence?, type: BufferType?) {
        if (autoTruncate) {
            adaptText(text ?: "", type)
        } else {
            super.setText(text, type)
        }
    }

    private fun adaptText(content: CharSequence, type: BufferType?) {
        if (originalText?.equals(content) == true) return
        originalText = content
        super.setText(text, type)
        post {
            try {
                var availableWidth = width - compoundPaddingStart - compoundPaddingEnd
                if (layoutParams is ViewGroup.MarginLayoutParams) {
                    //某些场景下使用的是外边距,也会影响内部...的显示
                    availableWidth =
                        availableWidth - (layoutParams as ViewGroup.MarginLayoutParams).leftMargin - (layoutParams as ViewGroup.MarginLayoutParams).rightMargin
                }
                val textString = (originalText ?: "").toString()
                if (paint.measureText(textString) <= availableWidth) {
                    return@post
                }
                val ellipsis = "..."
                val ellipsisWidth = paint.measureText(ellipsis)
                //存在计算误差,会导致出现半个点的情况,就多留一个.号的位置,后续用空格补充(和之前count - 1的原理差不多,都是保障.显示完整)
                val maxTextWidth = (availableWidth - ellipsisWidth / 3 * 4).coerceAtLeast(0f)
                val count = paint.breakText(
                    textString.toCharArray(),
                    0,
                    textString.length,
                    maxTextWidth,
                    null
                )
                val display = if (count > 0) {
                    //存在计算误差,添加空格补充,空格数量不定,保障撑满即可
                    textString.substring(0, count) + ellipsis + "      "
                } else {
                    if (ellipsisWidth <= availableWidth) ellipsis else ""
                }
                Log.d("EllipsizeEndTextview", "adaptText: $display")
                super.setText(display, type)
            } catch (e: Exception) {
                Log.e("EllipsizeEndTextview", "adaptText: ${e.message}")
            }
        }
    }

    override fun getText(): CharSequence {
        return originalText ?: ""
    }
}

涉及到的属性:

kotlin 复制代码
<!-- EllipsizeEndTextView 自定义属性 -->
<declare-styleable name="EllipsizeEndTextView">
    <!-- 是否自动截取 -->
    <attr name="autoTruncate" format="boolean" />
</declare-styleable>
相关推荐
那我掉的头发算什么2 分钟前
【javaEE】多线程进阶--CAS与原子类
android·java·jvm·java-ee·intellij-idea
BBB努力学习程序设计3 分钟前
超好用的轮播图神器:Swiper插件入门指南
前端·html
Yue丶越9 分钟前
【Python】基础语法入门(二)
android·开发语言·python
q***087410 分钟前
MySQL压缩版安装详细图解
android·mysql·adb
SteveCode21 分钟前
血赚不亏!Java 17 9 个炸裂特性,程序员看完直呼:太香了!
java
九鼎创展科技23 分钟前
九鼎创展发布X3588SCV4核心板,集成LPDDR5内存,提升RK3588S平台性能边界
android·人工智能·嵌入式硬件·硬件工程
帧栈26 分钟前
开发避坑指南(70):Vue3 Http请求头携带token下载pdf文件解决方案
前端·vue.js
h***066534 分钟前
项目升级Sass版本或升级Element Plus版本遇到的问题
前端·rust·sass
BLOB_10100136 分钟前
关于懒人复制idea项目的坑
java·ide·intellij-idea
Moe48839 分钟前
Spring Boot 自动配置核心:AutoConfigurationImportSelector 深度解析
java·后端·设计模式