WordTextView首行缩进左右对齐

项目中一般会遇到特殊的TextView,比如保证文字的左右对齐,首行缩进啦,这样的话系统定制的TextView就无法满足要求了,需要我们自定义WordTextView。

直接上效果图:
这里写图片描述

上面的是系统的TextView,下面的是WordTextView

package com.views;

import android.content.Context;
import android.graphics.Canvas;
import android.text.Layout;
import android.text.StaticLayout;
import android.text.TextPaint;
import android.util.AttributeSet;
import android.widget.TextView;

/**
 * By Jie 2017.7.9
 *
 *  Word排版文本View
 *
 * 首行缩进,左右对齐TextView
 */
public class WordTextView extends TextView {

    private int mLineY;//y轴绘制的坐标
    private int mViewWidth;//view总宽度
    private int mViewHeight;//view总高度

    public WordTextView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //如果设置的是确切值,那么就不再继续测量了
        if(MeasureSpec.getMode(widthMeasureSpec)==MeasureSpec.EXACTLY&&MeasureSpec.getMode(heightMeasureSpec)==MeasureSpec.EXACTLY){
            return;
        }
        //如果有的值测量的不准确,需要重新测量
        Layout layout = getLayout();//获取测量的文本布局
        mViewHeight =0;
        mViewWidth = layout.getWidth()+getPaddingLeft()+getPaddingRight();
        for (int i = 0; i < layout.getLineCount(); i++) {
            mViewHeight += getLineHeight();
        }
        mViewHeight = mViewHeight + getPaddingTop() + getPaddingBottom();
        setMeasuredDimension(mViewWidth,mViewHeight);//重新设置宽和高
    }


    /**
     *
     * 整体思路如下:
     *
     * 1  确定整个文本有多少行
     *
     * 2 重新绘制每一行的文本保证能左右对齐,或者首行缩进
     *
     * 3 最后一行正常绘制,因为他没有下一行,不一定要设置右对齐了
     * @param canvas
     */
    @Override
    protected void onDraw(Canvas canvas) {
        TextPaint paint = getPaint();
        mViewWidth = getMeasuredWidth();//获取测量的实际的宽度
        String text = (String) getText();//获取字符串string
        mLineY = 0;
        mLineY += getTextSize()*1.5;//开始绘制的y轴坐标
        Layout layout = getLayout();
        for (int i = 0; i < layout.getLineCount(); i++) {
            int lineStart = layout.getLineStart(i);
            int lineEnd = layout.getLineEnd(i);
            String line = text.substring(lineStart, lineEnd);

            float width = StaticLayout.getDesiredWidth(text, lineStart, lineEnd, getPaint());//每一行文本的宽度
            if (needScale(line) && i < layout.getLineCount()-1 ) {
                drawScaledText(canvas,line, width,i);//自己绘制,保证可以左右对齐或者首行缩进
            } else {
                canvas.drawText(line, 0, mLineY, paint);//不用自己绘制,系统绘制即可
            }
            mLineY += getLineHeight();//继续绘制下一行,找到下一行坐标
        }
    }

    /**
     * 思路如下:
     *
     * 举个例子,屏幕宽度100,文本90,要想让文本填充整个屏幕怎么办呢?
     * 很简单,让文本字符间隔填充剩下的10,还是不懂?
     * 假如文本有10个字符,那么我让每个字符绘制的时候都和上一个字符间隔多1,那么正好文本宽度就成了(90+10=100)可以左右对齐屏幕了
     * @param canvas
     * @param line
     * @param lineWidth
     * @param index
     */
    private void drawScaledText(Canvas canvas,String line, float lineWidth,int index) {
        float x = 0;
        if (isFirstLineOfParagraph(index)) {//如果是第一行就要加空格字符,保证可以首行缩进
            String blanks = "        ";
            canvas.drawText(blanks, x, mLineY, getPaint());
            float bw = StaticLayout.getDesiredWidth(blanks, getPaint());
            x += bw;
            lineWidth = lineWidth +bw;
        }

        float d = (mViewWidth-lineWidth) /line.length();//每个字符的额外间隔,专门是为了填充屏幕剩下的宽度
        for (int i = 0; i < line.length(); i++) {
            String c = String.valueOf(line.charAt(i));
            float cw = StaticLayout.getDesiredWidth(c, getPaint());
            canvas.drawText(c, x, mLineY, getPaint());
            x += cw+d;
        }
    }

    private boolean isFirstLineOfParagraph(int lineIndex) {
        return lineIndex==0;
    }

    private boolean needScale(String line) {
        if (line.length() == 0) {
            return false;
        } else {
            return line.charAt(line.length() - 1) != '\n';
        }
    }

}


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值