“全文”和“收起”实现

朋友圈列表的点击“全文”展开、点击“收起”折叠,实现起来很简单,主要是以下两步:

  • 获取item文本的行数
  • 记录item文本的状态

1.获取文本的行数

很容易想到获取文本的行数,超出规定行数便折叠文本,但没有方法可以直接根据字数计算出TextView的行数,所以只能用

content.setText();
content.getLineCount();

这时会发现这样获取到的行数为0,因为setText()后立即调用getLineCount()TextView还未完成measure,要想准确获取到TextView的行数有两种方法:

  • ViewTreeObserver监听View初始化的各种状态
    使用它的OnPreDrawListener在TextView完成测量和定位即将绘制时调用 getLineCount()即可得到TextView的真实行数
  • View.post(Runnable r)方法
    这个Runnable会被添加到一个顺序执行的UI事件队列,等执行到里面的代码时,View已经完成了measure和layout等一系列初始化工作,所以可以正确获取到View的高度等信息,很好用的方法,相比第一种方法的好处就是代码少且只执行一次,不用取消监听

    The UI event queue will process events in order. After setContentView() is invoked, the event queue will contain a message asking for a relayout, so anything you post to the queue will happen after the layout pass

这里还是用了第一种方法ViewTreeObserver,感觉语义性更好

holder.content.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
                    @Override
                    public boolean onPreDraw() {
                        //这个回调会调用多次,获取完行数记得注销监听
                        holder.content.getViewTreeObserver().removeOnPreDrawListener(this);
                        if(holder.content.getLineCount() > MAX_LINE_COUNT){
                            holder.content.setMaxLines(MAX_LINE_COUNT);
                            holder.expandOrCollapse.setVisibility(View.VISIBLE);
                            holder.expandOrCollapse.setText("全文");
                        }else{
                            holder.expandOrCollapse.setVisibility(View.GONE);
                        }
                        return true;
                    }
                });
holder.content.setMaxLines(Integer.MAX_VALUE);
holder.content.setText(Util.getContent(position));

2.记录item文本的状态

如果只是像上面写的那样每次初始化item时去获取文本的行数,然后根据行数选择是否折叠文本的话,会引发一个问题:

即已经获取过行数的position item滑出可视范围又滑回来时,根据RecyclerView的复用,TextView又会被重新测量高度行数然后是否折叠,有兴趣的同学可以试试,从列表顶部往下滑是没问题,但从底部往上滑,列表会不断跳动,在文字多的情况下甚至滑不回顶部,因为上面即将进入可视范围的item始终处于measure(展开)和超出行数折叠文本的死循环

所以当获取完每个position上的item文本行数后应把信息存起来,在这里我们定义三种状态并在每个item初始化时保存起来:
STATE_NOT_OVERFLOW //文本不超过规定行数
STATE_COLLAPSED //文本超过了规定行数,处于折叠状态
STATE_EXPANDED //文本超过了规定行数,被点击后处于展开状态

代码如下:

int state = mTextStateList.get(position, STATE_UNKNOW);
    //如果该item是第一次初始化,则去获取文本的行数
    if(state == STATE_UNKNOW){
        holder.content.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
            @Override
            public boolean onPreDraw() {
                //这个回调会调用多次,获取完行数记得注销监听
                holder.content.getViewTreeObserver().removeOnPreDrawListener(this);
                //记录文本的状态
                if(holder.content.getLineCount() > MAX_LINE_COUNT){
                    holder.content.setMaxLines(MAX_LINE_COUNT);
                    holder.expandOrCollapse.setVisibility(View.VISIBLE);
                    holder.expandOrCollapse.setText("全文");
                    mTextStateList.put(position, STATE_COLLAPSED);
                }else{
                    holder.expandOrCollapse.setVisibility(View.GONE);
                    mTextStateList.put(position, STATE_NOT_OVERFLOW);
                }
                return true;
            }
        });
        holder.content.setMaxLines(Integer.MAX_VALUE);
        holder.content.setText(Util.getContent(position));
    }else{
        //如果之前已经初始化过了,则使用保存的状态,无需再获取一次
        switch (state){
            case STATE_NOT_OVERFLOW:
                holder.expandOrCollapse.setVisibility(View.GONE);
                break;
            case STATE_COLLAPSED:
                holder.content.setMaxLines(MAX_LINE_COUNT);
                holder.expandOrCollapse.setVisibility(View.VISIBLE);
                holder.expandOrCollapse.setText("全文");
                break;
            case STATE_EXPANDED:
                holder.content.setMaxLines(Integer.MAX_VALUE);
                holder.expandOrCollapse.setVisibility(View.VISIBLE);
                holder.expandOrCollapse.setText("收起");
                break;
        }
        holder.content.setText(Util.getContent(position));
    }

最后设置点击事件:

holder.expandOrCollapse.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    int state = mTextStateList.get(position, STATE_UNKNOW);
                    if(state == STATE_COLLAPSED){
                        holder.content.setMaxLines(Integer.MAX_VALUE);
                        holder.expandOrCollapse.setText("收起");
                        mTextStateList.put(position, STATE_EXPANDED);
                    }else if(state == STATE_EXPANDED){
                        holder.content.setMaxLines(MAX_LINE_COUNT);
                        holder.expandOrCollapse.setText("全文");
                        mTextStateList.put(position, STATE_COLLAPSED);
                    }
                }
            });

最终效果图如下:


chip.gif

github地址:https://github.com/CrazyPumPkin/ExpandableText


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值