Android进阶——或许是处理“More&click”型多行的TextView换行的最优雅的一种方式

引言

相信很多Android APP 开发者在处理TextView 换行的时候都曾头痛不已过,尤其是在做复杂布局的时候,适配的时候都踩过不少坑。笔者也踩过,直到在一次查看源码的时候发现了ViewTreeObserver,总算是实现了优雅的格式化多行文本,在使用一个控件的时候抽点时间了解下提供的公共方法,有时候可以避免很多不必要的坑。

一、ViewTreeObserver概述

ViewTreeObserver顾名思义就是视图树的观察者角色,可以监听视图树的全局变化,比如整棵树的布局,开始的绘画传递,触摸模式的改变等等,都提供了对应的八个监听接口(A view tree observer is used to register listeners that can be notified of global changes in the view tree)。ViewTreeObserver用来注册监听器,在视图树全局发生变化时收到通知。它不能被应用实例化,因为它是由视图提供,只能通过调用android.view.View的getViewTreeObserver()来获取对应的实例。
这里写图片描述
其实整个ViewTreeObserver机制从源码上看,本质上就是个观察者模式,那么主要的角色就有两种:
- ViewTree视图树——在Android中所有视图由View和View的子类组成。ViewGroup也是view的子类,它是View的容器,它可以装载View和ViewGroup。这样ViewGroup和View以树形结构一层一层的嵌套组合,就形成了视图树。

  • Observer观察者。使用了观察者的设计模式,ViewTree是被观察者(或者说是主题、内容),ViewTreeObserver是观察者,通过ViewTreeObserver注册监听来观察ViewTree的变化,当ViewTree发生变化,就会调用ViewTreeObserver的相关方法来通知其这一改变。我们可以在ViewTreeObserver中add自己的监听器,从而得到ViewTree的某一变化的通知做出自己的逻辑处理。

二、SpannableString和ClickableSpan概述

SpannableString和ClickableSpan本质上就是高级的String,具体可参见Android进阶——借助强大Span家族增添丰富的特效及格式化字符串

三、实现思路

简单来说就是实现根据TextView要实现的字符串长度去动态适配。

  • 通过ViewTreeObserver机制监听,TextView绘制之后,在OnGlobalLayoutListener中计算原始的字符串长度,当多行显示的时候,以第一行所显示字符的个数为行长。

  • 判断最后一行的长度,并进行逻辑处理,添加“More”型字符串

  • 以Span系为辅助实现点击,并开放接口

四、实现源码

这里写图片描述

核心源码

package com.crazyview.loadertest;

import android.graphics.Color;
import android.text.Spannable;
import android.text.SpannableString;
import android.text.TextPaint;
import android.text.method.LinkMovementMethod;
import android.text.style.ClickableSpan;
import android.view.View;
import android.view.ViewTreeObserver;
import android.widget.TextView;

/**
 * Auther: Crazy.Mo
 * DateTime: 2017/11/14 10:55
 * Summary:
 */
public class FormatUtil {
    /**
     * @param textView 目标TextView
     * @param moreStr   more型字符串,当显示不完全的时候显示替代字符串
     * @param clickListener 点击的回调接口
     */
    public static void getTextMaxEms(final TextView textView, final String moreStr,  final LinkClickListener clickListener){
        final String contentStr=textView.getText().toString();
        ViewTreeObserver viewTreeObserver=textView.getViewTreeObserver();
        viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener(){

            @Override
            public void onGlobalLayout() {
                if(textView.getTag()==null){
                    textView.setTag(textView.getText().toString());
                }
                String currentStr=textView.getText().toString();
                ViewTreeObserver treeObserver=textView.getViewTreeObserver();
                treeObserver.removeOnGlobalLayoutListener(this);
                int lineCount=textView.getLineCount();
                if(lineCount>1) {
                    //获取第一行的文本长度当做每行文本的长度
                    int lineLength = textView.getText().subSequence(textView.getLayout().getLineStart(0), textView.getLayout().getLineEnd(0)).toString().length();

                    //获取最后一行文本的长度
                    int lastLineLength = textView.getText().subSequence(textView.getLayout().getLineStart(textView.getLayout().getLineCount() - 1), textView.getLayout().getLineEnd(textView.getLayout().getLineCount() - 1)).toString().length();

                    if (lastLineLength >= lineLength - moreStr.length() - 2) {
                        currentStr = currentStr.substring(0, contentStr.length() - (lastLineLength - (lineLength - moreStr.length() - 5))) + "...";
                    }
                    final String finalStr = currentStr + moreStr;
                    SpannableString spanString = new SpannableString(finalStr);
                    ClickableSpan clickSpan = new ClickableSpan() {

                        @Override
                        public void onClick(View widget) {
                            clickListener.onLinkClick(contentStr);
                        }

                        @Override
                        public void updateDrawState(TextPaint ds) {
                            ds.setColor(ds.linkColor);
                            ds.setUnderlineText(true);
                        }
                    };
                    spanString.setSpan(clickSpan, currentStr.length(), finalStr.length(), Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
                    textView.setText(spanString);
                    textView.setLinkTextColor(Color.RED);
                    //必须添加这一段
                    textView.setMovementMethod(LinkMovementMethod.getInstance());
                    textView.setFocusable(false);
                    textView.setClickable(false);
                    textView.setLongClickable(false);
                }
            }
        });
    }
 }

点击回调接口

package com.crazyview.loadertest;

/**
 * Auther: Crazy.Mo
 * DateTime: 2017/11/14 14:52
 * Summary:
 */
public interface LinkClickListener {
    void onLinkClick(Object object);
}

使用

package com.crazyview.loadertest;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;
import android.widget.Toast;

public class MultableTextActivity extends AppCompatActivity {
    private TextView textOneline,textView,textMult,textlastLine;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_multable_text);
        init();
    }

    private void init() {
        textView= (TextView) findViewById(R.id.tv_single);
        textView.setText("百世快递投递员于2017年10月14日下午四点左右");
        FormatUtil.getTextMaxEms(textView, "查看详情", /*textView.getText().toString(),*/ new LinkClickListener() {
            @Override
            public void onLinkClick(Object object) {
                Toast.makeText(MultableTextActivity.this,"查看详情Click"+(String)object,Toast.LENGTH_SHORT).show();
            }
        });

        textOneline= (TextView) findViewById(R.id.tv_oneline);
        textOneline.setText("百世快递投递员于2017年10月14日下午四点左右到我处成");
        FormatUtil.getTextMaxEms(textOneline, "查看详情", /*textOneline.getText().toString(),*/ new LinkClickListener() {
            @Override
            public void onLinkClick(Object object) {
                Toast.makeText(MultableTextActivity.this,"查看详情Click"+(String)object,Toast.LENGTH_SHORT).show();
            }
        });

        textMult= (TextView) findViewById(R.id.tv_mutlline);
        textMult.setText("百世快递投递员于2017年10月14日下午四点左右到我处成功揽件,直至今日2017年10月30日收件人还未收到包裹并且也无法在对方的网上查询到有关包裹的任何消息,曾经有一次去代办点领取包裹亲眼目睹包裹的胡乱抛放,于是在此期间多次联系对方客服,询问包裹情况,对方客服也数次明说24小时内会给一个反馈,由于时间久远只记得部分客服工号(LYWX035),但是24小时、48小时甚至72小时都无任何回复,由于此包裹所寄物品是从香港买回来的药,北京我家人急用已经严重延误了,恳请总局帮忙联系无耻百世快递,并请求赔偿原物品及1元精神损失。");
        FormatUtil.getTextMaxEms(textMult, "查看详情", /*textMult.getText().toString(),*/ new LinkClickListener() {
            @Override
            public void onLinkClick(Object object) {
                Toast.makeText(MultableTextActivity.this,"查看详情Click"+(String)object,Toast.LENGTH_SHORT).show();
            }
        });

        textlastLine= (TextView) findViewById(R.id.tv_lastline);
        textlastLine.setText("百世快递投递员于2017年10月14日下午四点左右到我处成功揽件,直至今日2017年10月30日收件人还未收到包裹并且也无法在对方的网上查询到有关包裹的任何消息,曾经有一次去代办点领取包裹亲眼目睹包裹的胡乱抛放,于是在此期间多次联系对方客服,询问包裹情况,对方客服也数次明说24小时内会给一个反馈,由于时间久远只记得部分客服工号(LYWX035),但是24小时、48小时甚至72小时都无任何回复,由于此包裹所寄物品是从香港买回来的药,北京我家人急用已经严重延误了,恳请总局帮忙联系无耻百世快递,并请");
        FormatUtil.getTextMaxEms(textlastLine, "查看详情", /*textlastLine.getText().toString(),*/ new LinkClickListener() {
            @Override
            public void onLinkClick(Object object) {
                Toast.makeText(MultableTextActivity.this,"查看详情Click"+(String)object,Toast.LENGTH_SHORT).show();
            }
        });
    }
}

布局

<?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"
    android:orientation="vertical">
    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="显示文字长度不够一行时:"
        android:textSize="16sp"
        android:textStyle="bold"/>
    <TextView
        android:id="@+id/tv_single"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        />

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="显示文字长度刚好一行时:"
        android:textSize="16sp"
        android:textStyle="bold"/>
    <TextView
        android:id="@+id/tv_oneline"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        />

    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="多行最后一行>=行长度减去more型再减去2"
        android:textSize="16sp"
        android:textStyle="bold"/>
    <TextView
        android:id="@+id/tv_mutlline"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        />
    <TextView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="多行最后一行小于行长度减去more型再减去2"
        android:textSize="16sp"
        android:textStyle="bold"/>
    <TextView
        android:id="@+id/tv_lastline"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        />
</LinearLayout>
### 回答1: 在Android中,TextView换行可以通过以下几种方式实现: 1. 在XML布局文件中使用“\n”或“\r\n”来表示换行,例如: ``` <TextView android:id="@+id/text_view" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="第一行\n第二行\n第三行" /> ``` 2. 在Java代码中使用“\n”或“\r\n”来设置TextView文本,例如: ``` TextView textView = findViewById(R.id.text_view); textView.setText("第一行\n第二行\n第三行"); ``` 3. 在XML布局文件中设置TextView的maxLines属性为一个较大的值,例如: ``` <TextView android:id="@+id/text_view" android:layout_width="wrap_content" android:layout_height="wrap_content" android:maxLines="100" android:text="第一行\n第二行\n第三行" /> ``` 以上三种方式都可以实现TextView换行的效果,具体选择哪种方式取决于实际需求。 ### 回答2: 在Android中,TextView是最常用的屏幕元素之一,主要用于显示文本内容。但是,如果文本内容太长,会导致文本显示超出屏幕范围或者只显示一行的情况。因此,我们需要换行来解决这个问题。 针对这种情况,我们可以使用多种方法来实现TextView换行。下面是几种比较常用的方法: 1. 通过在文本中手动添加换行符(\n) TextView支持文本中包含换行符,只需要在文本添加\n(或者\r\n)即可实现换行。这种方法非常简单,但是需要手动添加换行符,不太方便。 2. 设置TextView的maxLines属性 TextView的maxLines属性可以限制TextView显示的行数,当TextView文本内容超过指定行数时,会自动截断并在结尾添加省略号。可以通过设置maxLines的值来实现换行。 例如: ```xml <TextView android:layout_width="match_parent" android:layout_height="wrap_content" android:text="这里是一段很长很长的文本,如果不加处理,会导致超长超出屏幕显示。" android:maxLines="2" /> ``` 上面的代码将TextView的maxLines属性设置为2,当文本内容超过2行时,会自动换行。 3. 设置TextView的ellipsize属性 TextView的ellipsize属性可以控制TextView文本超出后的省略样式。可以将TextView的ellipsize属性设置为end(默认值),在文本超出后在结尾添加省略号,也可以设置为其他值,如start、middle、marquee等。 例如: ```xml <TextView android:layout_width="match_parent" android:layout_height="wrap_content" android:text="这里是一段很长很长的文本,如果不加处理,会导致超长超出屏幕显示。" android:maxLines="1" android:ellipsize="end" /> ``` 上面的代码将TextView的maxLines属性设置为1,当文本内容超过1行时,会自动在结尾添加省略号。 总之,在Android中实现TextView换行有多种方法,而具体的实现方式要根据需求来选择。常见的实现方式包括手动添加换行符、设置maxLines属性和设置ellipsize属性。 ### 回答3: 在Android中,TextView是最基本的控件之一,用于显示文本内容。TextView默认情况下是自动换行的,也就是说,当一行文本超过了TextView的边界时,它将会自动换行到新的一行。但是,有些时候,我们可能需要手动控制TextView换行,比如说在某些情况下我们需要强制一行只显示一个字符,或者强制在某个位置进行换行。 对于TextView换行,我们可以使用两种方式:在XML布局文件中或者在Java代码中设置相关属性。 1. 在XML布局文件中设置TextView换行 我们可以在XML布局文件中使用以下属性来设置TextView换行: - android:singleLine:表示是否只显示单行,设置为true时,TextView只显示一行,超出的内容将被截断,设置为false时,可以显示多行文本。 - android:ellipsize:表示文本截断的方式,可以设置成常用的4种方式:end、start、middle和marquee。 - android:maxLines:表示TextView最多可以显示的行数,超出的部分将被截断。 例如: ``` <TextView android:id="@+id/text_view" android:layout_width="match_parent" android:layout_height="wrap_content" android:singleLine="false" android:ellipsize="none" android:maxLines="3" android:text="这是一个多行TextView,当TextView中的文本内容超过TextView的宽度时,TextView会自动换行以适应屏幕。"/> ``` 2. 在Java代码中设置TextView换行 我们也可以在Java代码中使用以下方法来设置TextView换行: - setSingleLine(boolean singleLine):表示是否只显示单行,设置为true时,TextView只显示一行,超出的内容将被截断,设置为false时,可以显示多行文本。 - setEllipsize(TextUtils.TruncateAt where):表示文本截断的方式,可以设置成常用的4种方式:END、START、MIDDLE和MARQUEE。 - setMaxLines(int maxLines):表示TextView最多可以显示的行数,超出的部分将被截断。如果将maxLines设置为1,便可以实现在一个TextView中只显示一个字符的效果。 例如: ``` TextView textView = findViewById(R.id.text_view); textView.setSingleLine(false); textView.setEllipsize(TextUtils.TruncateAt.NONE); textView.setMaxLines(3); textView.setText("这是一个多行TextView,当TextView中的文本内容超过TextView的宽度时,TextView会自动换行以适应屏幕。"); ``` 总之,TextView自带自动换行功能,可以非常方便地实现自动换行。但同时,我们也可以通过设置属性或者调用相关方法来手动控制TextView换行。根据具体需求选择更为便捷的方式即可。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CrazyMo_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值