Android开发经验谈程序员AndroidUI

[Android]使用VLayout,你可能遇到的问题

2017-12-13  本文已影响1752人  Vane_Subin

概述

VirtualLayout是一个针对RecyclerView的LayoutManager扩展, 主要提供一整套布局方案和布局间的组件复用的问题。

设计思路
主要功能

以上是VLayout 在 GitHub 上的介绍

RecyclerView 焦点

如果发现你的RecyclerView 在滑动的过程中经常跳屏(暂且称它为跳屏,现象是总是跳到某个item),那么你可能遇到RecyclerView与子item某控件的焦点冲突问题。处理这类问题的方法有很多,例如你可以在你的xml中使用android:descendantFocusability,这里就不赘述了。

横向滑动组件

在使用VLayout的过程中,你会发现根本找不到一个横向线性布局的Helper(LinearLayoutHelper不支持横向)


GitHub issues #9

那么我们按照VLayout设计者的建议,自行嵌套一个横向的RecyclerView。创建类HorizontalAdapter 并继承DelegateAdapter.Adapter<HorizontalViewHolder>

public class HorizontalAdapter extends DelegateAdapter.Adapter<HorizontalViewHolder> {

    private Context mContext;

    private LayoutHelper mLayoutHelper;

    public HorizontalAdapter(Context context, LayoutHelper layoutHelper) {
        this.mContext = context;
        this.mLayoutHelper = layoutHelper;
    }

    @Override
    public LayoutHelper onCreateLayoutHelper() {
        return mLayoutHelper;
    }

    @Override
    public HorizontalViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        if (viewType == 4) {
            return new HorizontalViewHolder(
                    LayoutInflater.from(mContext).inflate(R.layout.layout_recyclerview, parent, false));
        }
        return null;
    }

    @Override
    public void onBindViewHolder(NormalViewHolder holder, int position) {
        if (holder.itemView instanceof RecyclerView) {
            RecyclerView recyclerView = (RecyclerView) holder.itemView;
            LinearLayoutManager linearLayoutManager = new LinearLayoutManager(mContext);
            linearLayoutManager.setOrientation(LinearLayoutManager.HORIZONTAL);
            recyclerView.setLayoutManager(linearLayoutManager);
            recyclerView.setAdapter(new HotItemAdapter(mContext, list,listener));
        }
    }

    @Override
    public int getItemViewType(int position) {
        return 4;
    }

    @Override
    public int getItemCount() {
        return 1;
    }
}

至此,初步的Adapter已经码好了。那么问题来了

横向RecyclerView 状态重置
这句话什么意思,就是当你的横向RecyclerView滑出屏幕外被复用时,之前对横向RecyclerView或其子item做的操作都将重置。例如横向RecyclerView滑动过的距离,当你滑出屏幕外后再滑回来,横向RecyclerView又回到第一个item。
解决思路:
当横向RecyclerView 将被复用时,记录RecyclerView的状态。当横向RecyclerView 即将出现在屏幕内时,恢复其状态。
我们在HorizontalAdapter中重写两个方法onViewDetachedFromWindowonViewAttachedToWindow,以上面滑动距离为例

    @Override
    public void onViewDetachedFromWindow(NormalViewHolder holder) {
        if (holder.itemView instanceof RecyclerView) {
            RecyclerView recyclerView = ((RecyclerView) holder.itemView);
            LinearLayoutManager manager = (LinearLayoutManager) recyclerView.getLayoutManager();
            position = manager.findFirstVisibleItemPosition();
            View view = manager.findViewByPosition(position);
            ViewGroup.MarginLayoutParams lp = 
                              (ViewGroup.MarginLayoutParams) view.getLayoutParams();
            if (view != null) {
                xOffset = view.getLeft() - lp.leftMargin; //如果你设置了margin则减去
            }
        }
        super.onViewDetachedFromWindow(holder);
    }

    @Override
    public void onViewAttachedToWindow(NormalViewHolder holder) {
        super.onViewAttachedToWindow(holder);
        if (holder.itemView instanceof RecyclerView) {
            RecyclerView recyclerView = ((RecyclerView) holder.itemView);
            LinearLayoutManager manager = (LinearLayoutManager) recyclerView.getLayoutManager();
            manager.scrollToPositionWithOffset(position, xOffset);
        }
    }

如果你还内嵌了ViewPager(轮播?),同样会出现组件的状态问题。因为VLayout干预不到这一层,所以我们还是得自己解决,解决方案同上

共用复用池
当我们需要嵌套RecyclerView时,我们让子RecyclerView与父RecyclerView共用一个复用池

public HorizontalAdapter(Context context, LayoutHelper layoutHelper, RecyclerView.RecycledViewPool viewPool) {
        this.mContext = context;
        this.mLayoutHelper = layoutHelper;
        this.viewPool = viewPool;
    }

    @Override
    public void onBindViewHolder(NormalViewHolder holder, int position) {
        if (holder.itemView instanceof RecyclerView) {
            RecyclerView recyclerView = (RecyclerView) holder.itemView;
            recyclerView.setRecycledViewPool(viewPool);
            ...
       }
    }
上一篇 下一篇

猜你喜欢

热点阅读