TextView 悬挂缩进实现及解析

XML:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <TextView
        android:id="@+id/tv_indent"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />
</LinearLayout>
代码:
  • autoSplitText(tv,indent):该方法用来切割字符串,在需要换行的文本处添加换行符和缩进空格。
    private String autoSplitText(final TextView tv, final String indent) {
        //此处拿到得还是一串长文本
        final String rawText = tv.getText().toString(); //原始文本
        final Paint tvPaint = tv.getPaint(); //paint,包含字体等信息
        final float tvWidth = tv.getWidth() - tv.getPaddingLeft() - tv.getPaddingRight(); //控件本身宽度

        //将缩进转换成空格
        String indentSpace = "";
        float indentWidth = 0;
        if (!TextUtils.isEmpty(indent)) {
            float rawIndentWidth = tvPaint.measureText(indent); //获取缩进的宽度
            if (rawIndentWidth < tvWidth) { //如果缩进宽度比控件本身宽度小的话
                while ((indentWidth = tvPaint.measureText(indentSpace)) < rawIndentWidth) {//将缩进的宽度转变为空格的形式表示出来
                    indentSpace += " ";
                }
                //减去一个空格宽度
                indentSpace = indentSpace.substring(0, indentSpace.length() - 1);
            }
        }

        //将原始文本按行拆分
        // \r 光标重新回到本行开头
        // \n 光标往下一行
        String[] rawTextLines = rawText.replaceAll("\r", "").split("\n");
        StringBuilder sbNewText = new StringBuilder();
        for (String rawTextLine : rawTextLines) {
            if (tvPaint.measureText(rawTextLine) <= tvWidth) {
                //如果整行宽度在控件可用宽度之内,就不处理了
                sbNewText.append(rawTextLine);
            } else {
                //如果整行宽度超过控件可用宽度,则按字符测量,在超过可用宽度的前一个字符处手动换行
                float lineWidth = 0;//记录文字行宽
                for (int cnt = 0; cnt != rawTextLine.length(); ++cnt) {
                    char ch = rawTextLine.charAt(cnt);
                    if (lineWidth < 0.1f && cnt != 0) {
                        //当lineWidth宽度等与0并且cnt不等于0时候,可以确认是在第二行,此时在行头加上缩进。
                        sbNewText.append(indentSpace);
                        lineWidth += indentWidth;
                    }
                    lineWidth += tvPaint.measureText(String.valueOf(ch));
                    if (lineWidth <= tvWidth) {
                        //文字行宽小于控件行宽,将文字添加进StringBuilder中。
                        sbNewText.append(ch);
                    } else {
                        //如果文字行宽大于控件行宽,加入换行符号。清空文字宽度,cnt 减 1后再重新计算。
                        //走到这一步之后会从for循环开头走,在下一行加缩进。
                        sbNewText.append("\n");
                        lineWidth = 0;
                        --cnt;
                    }
                }
            }
            sbNewText.append("\n");
        }
        //把结尾多余的\n去掉
        if (!rawText.endsWith("\n")) {
            sbNewText.deleteCharAt(sbNewText.length() - 1);
        }
        return sbNewText.toString();
    }
}
  • 使用
    • 注册一个观察者来监听视图,当视图树的布局发生改变或者View在视图树的可见状态发生改变时会调用的接口ViewTreeObserver.OnGlobalLayoutListener()。
    • 并且在ViewTreeObserver.OnGlobalLayoutListener()回调函数中设置SpannableStringBuilder点击也不会有冲突。
final TextView tvIndent = (TextView) findViewById(R.id.tv_indent);
String upgradeContent = "1,Google LLC 是一家美国跨国科技公司,专门从事互联网相关服务和产品,包括在线广告技术,搜索引擎,云计算,软件和硬件。\n2,Google由Larry Page和Sergey Brin于1998年创立,当时他们是博士。学生在斯坦福大学在加州。\n3,他们共同拥有约14%的股份,并通过超级投票控制56%的股东投票权股票。\n4,他们于1998年9月4日将Google合并为一家私营公司。\n5,2004年8月19日首次公开募股(IPO),谷歌搬到位于加利福尼亚州山景城的总部,绰号为Googleplex。\n6,2015年8月,谷歌宣布计划重组其作为一家名为Alphabet Inc.的企业集团的各种利益。谷歌是Alphabet的主要子公司,并将继续成为Alphabet互联网利益的保护伞公司。\n7,Sundar Pichai被任命为Google首席执行官,取代Larry Page成为Alphabet的首席执行官。";
tvIndent.setText(upgradeContent);
tvIndent.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
    @RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN)
    @Override
    public void onGlobalLayout() {
        //回调之后取消监听,防止回调被来回调取。
      tvIndent.getViewTreeObserver().removeOnGlobalLayoutListener(this);
        String indentTextDesc = autoSplitText(tvIndent, "1,");
        String targetClick = "搜索引擎";
        SpannableStringBuilder contentSpannableBuilder = new SpannableStringBuilder(indentTextDesc);
        contentSpannableBuilder.setSpan(new ClickableSpan() {
            @Override
            public void onClick(View widget) {
                Toast.makeText(MainActivity.this, "Toast 被点击", Toast.LENGTH_LONG).show();
            }
            @Override
            public void updateDrawState(TextPaint ds) {
                ds.setUnderlineText(true);
                ds.setColor(Color.parseColor("#067ce4"));
                ds.setFakeBoldText(true);
            }
        }, indentTextDesc.indexOf(targetClick), indentTextDesc.indexOf(targetClick) + targetClick.length(), Spanned.SPAN_INCLUSIVE_INCLUSIVE);
        tvIndent.setMovementMethod(LinkMovementMethod.getInstance());
        tvIndent.setText(contentSpannableBuilder);
    }
});
效果:

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值