Android TextView文字大小自动适应不指定宽高

一、项目中出现文字很长的textview,如果需要在一行全部显示,那么当文字很长时,就要自动缩小文字,尽量显示。一般都是从给了最大textsize值,和最小textsize值,仍然显示不下,则在后面显示”...“。网上查了很多资料,都有这样那样的问题。比如:android 8.0 textView自动缩小字体特性,设置: 

<android.support.v7.widget.AppCompatTextView
      android:layout_width="match_parent"
      android:layout_height="200dp"
      app:autoSizeTextType="uniform"
      app:autoSizeMinTextSize="12sp"
      app:autoSizeMaxTextSize="100sp"
      app:autoSizeStepGranularity="2sp" />

但是这样的话,需要 width 或者 height 至少有一个是固定值。而实际项目中,width、height不是固定的,所以不满足要求。

再比如:在onDraw()中for循环,通过从最大textsize一直downto到最小textsize,并利用paint.measureText("test")测量文字宽度,从而设置合适的textsize,但是经实际测试,在某些特殊字符串时,出现测量不准确,textview文字一直闪烁现象。

还有的时候,一个LinearLayout中,会有两个textview,但是文字缩放后,baseline不对齐。还有的文字从大显示成小的可以,再从小的显示大的文字,就失效了,等等各种问题。

总之:实际组入项目过程中,测试发现各种各样的问题,最后决定自己重新写一个自定义TextView实现,代码如下:

package com.example.view

import android.content.Context
import android.graphics.Paint.FontMetricsInt
import android.text.TextPaint
import android.util.AttributeSet
import android.util.TypedValue
import com.example.view.R
import kotlin.math.max
import kotlin.math.min


class AutoFitTextView @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet,
    defStyleAttr: Int = 0
) : TextView(context, attrs, defStyleAttr) {
    private var mViewWidth = 0
    private var mMinTextSize = 8f
    private var mMaxTextSize = 14f
    private val mTextPaint = TextPaint()
    private var mFontMetricsInt: FontMetricsInt? = null
    private var mSourceTextContentSize = 0
    private var mLoadContent: String? = null
    private var mViewHeight = 0
    private var mViewHeightSpec = 0

    init {
        initAttrs(context, attrs)
    }

    private fun initAttrs(
        context: Context,
        attrs: AttributeSet?
    ) {
        val mTypedArray =
            context.obtainStyledAttributes(attrs, R.styleable.AutoFitTextView)
        val minTextSize = mTypedArray.getDimension(
            R.styleable.AutoFitTextView_minTextSize,
            context.resources.getDimension(R.dimen.8sp)
        )
        val maxTextSize = mTypedArray.getDimension(
            R.styleable.AutoFitTextView_maxTextSize,
            context.resources.getDimension(R.dimen.14sp)
        )
        mMinTextSize = min(minTextSize, maxTextSize)
        mMaxTextSize = max(minTextSize, maxTextSize)
        mTextPaint.textSize = mMaxTextSize
        setTextSize(TypedValue.COMPLEX_UNIT_SP, mMaxTextSize)
        mTypedArray.recycle()
    }

    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        mViewWidth = MeasureSpec.getSize(widthMeasureSpec) - paddingLeft - paddingRight
        if (mFontMetricsInt == null) {
            mFontMetricsInt = mTextPaint.fontMetricsInt
        }
        mViewHeight = (mFontMetricsInt?.bottom ?: 0) - (mFontMetricsInt?.top ?: 0)
        mViewHeightSpec = MeasureSpec.makeMeasureSpec(
            mViewHeight + paddingTop + paddingBottom,
            MeasureSpec.EXACTLY
        )
        mLoadContent = text.toString()
        for (item in mMaxTextSize.toInt() downTo mMinTextSize.toInt()) {
            paint.textSize = item.toFloat()
            mSourceTextContentSize = paint.measureText(mLoadContent).toInt()
            if ((mViewWidth - mSourceTextContentSize) >= 0 || item == mMinTextSize.toInt()) {
                setTextSize(TypedValue.COMPLEX_UNIT_PX, item.toFloat())
                break
            }
        }
        super.onMeasure(widthMeasureSpec, mViewHeightSpec)
    }

    override fun onTextChanged(
        text: CharSequence?,
        start: Int,
        lengthBefore: Int,
        lengthAfter: Int
    ) {
        setTextSize(TypedValue.COMPLEX_UNIT_PX, mMaxTextSize)
        super.onTextChanged(text, start, lengthBefore, lengthAfter)
    }

    fun setMaxTextSize(size: Float) {
        mTextPaint.textSize = mMaxTextSize
        setTextSize(TypedValue.COMPLEX_UNIT_PX, mMaxTextSize)
        mMaxTextSize = size
    }

    fun setMinTextSize(size: Float) {
        mMinTextSize = size
    }
}

 attrs.xml

<declare-styleable name="AutoFitTextView">
        <attr name="minTextSize" format="dimension" />
        <attr name="maxTextSize" format="dimension" />
</declare-styleable>

 使用:

<com.example.view.AutoFitTextView
                android:id="@+id/evaluationAmountTv"
                android:layout_width="match_parent"
                android:layout_height="wrap_content"
                android:ellipsize="end"
                android:gravity="center_vertical|end"
                android:maxLines="1"
                android:singleLine="true"
                android:text="test"
                android:textColor="#8A8A99"
                app:maxTextSize="14sp"
                app:minTextSize="8sp"
                tools:text="0,003,12" />

其中, 自定义控件中,需要注意如下代码:

paint.measureText(mLoadContent).toInt()

计算文字宽度时,paint要用当前TextView的TextPaint,而不是自己new一个TextPaint实例,因为TextPaint是和各种资源有关的,TextView的源代码里,会对TextPaint有各种隐藏属性的设置,所以,如果自己new一个实例进行测量文字宽度,得出来的宽度与实际绘制出来的宽度不一致,导致仍然会有文字大小不合适出现"..."的情况。具体参考:正确调用Paint的measureText()方法取得字符串显示的宽度值_xuhuan_wh的专栏-CSDN博客

经测试,没啥大问题。如有问题,请继续沟通交流。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值