【Android】RV折叠适配器

1,折叠List,除了使用ListView中已有的ExpandAdapter,笔者定制化了此类,用户折叠列表的RV通用实现,仅供参考。

BaseExpandRecyclerAdapter,

​

import android.annotation.SuppressLint;
import android.util.Pair;
import android.view.ViewGroup;

import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;


import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.stream.Collectors;


public abstract class BaseExpandRecyclerAdapter<Parent, Child> extends RecyclerView.Adapter<RecyclerView.ViewHolder> {

    protected static final int PARENT_TYPE = "BaseExpandRecyclerAdapter".hashCode() + 1;
    protected static final int CHILD_TYPE = "BaseExpandRecyclerAdapter".hashCode() + 2;
    private final List<Pair<Parent, List<Child>>> mData = new ArrayList<>();

    private final List<Pair<Integer, Object>> mAdapterData = new ArrayList<>();

    private final List<Integer> mExpandParentPositionList = new ArrayList<>();

    public BaseExpandRecyclerAdapter() {
    }

    @SuppressLint("NotifyDataSetChanged")
    public void notifyData(List<Pair<Parent, List<Child>>> data) {
        mData.clear();
        mData.addAll(data);
        notifyAdapterDataSourcesChange();
        notifyDataSetChanged();
    }

    private void notifyAdapterDataSourcesChange() {
        mAdapterData.clear();
        if (mExpandParentPositionList.isEmpty()) {
            mAdapterData.addAll(mData.stream().map(it -> Pair.create(getItemViewTypeFromObject(it.first, PARENT_TYPE), (Object) it.first)).collect(Collectors.toList()));
            return;
        }
        for (int i = 0; i < mData.size(); i++) {
            mAdapterData.add(Pair.create(PARENT_TYPE, getParentData(i)));

            if (mExpandParentPositionList.contains(i)) {
                List<Child> childList = getChildList(i);
                mAdapterData.addAll(childList.stream().map(it -> Pair.create(getItemViewTypeFromObject(it, CHILD_TYPE), (Object) it)).collect(Collectors.toList()));
            }
        }
    }

    private int notifyAdapterDataSourcesInsert(int newExpandPosition) {
        int insertStart = getAdapterPosition(newExpandPosition) + 1;
        mAdapterData.addAll(insertStart, getChildList(newExpandPosition).stream().map(it -> Pair.create(getItemViewTypeFromObject(it, CHILD_TYPE), (Object) it)).collect(Collectors.toList()));
        return insertStart;
    }

    private int notifyAdapterDataSourcesRemove(int oldExpandPosition) {
        int removeStart = getAdapterPosition(oldExpandPosition) + 1;
        int count = getChildList(oldExpandPosition).size();

        Iterator<Pair<Integer, Object>> iterator = mAdapterData.iterator();

        int i = 0;
        while (iterator.hasNext()) {
            iterator.next();
            if (i >= removeStart && i < removeStart + count) {
                iterator.remove();
            }
            i++;
            if (i >= removeStart + count) {
                break;
            }
        }

        return removeStart;
    }


    protected int getAdapterPosition(int parentPosition) {
        if (mExpandParentPositionList.isEmpty()) {
            return parentPosition;
        }
        int count = mExpandParentPositionList.stream().filter(it -> parentPosition > it).mapToInt(it -> getChildList(it).size()).sum();

        return parentPosition + count;
    }


    /**
     * 展开
     */
    public void notifyExpand(int parentPosition) {
        boolean nextExpandStatus = !isExpand(parentPosition);
        notifyExpand(parentPosition, nextExpandStatus);
    }


    public void notifyExpand(int parentPosition, boolean expand) {
        if (!onSupportParentExpand(parentPosition, expand)) {
            return;
        }
        int index = mExpandParentPositionList.indexOf(parentPosition);
        int size = getChildList(parentPosition).size();
        if (index >= 0) {
            if (expand) {
                return;
            }
            mExpandParentPositionList.remove(index);
            int removeStart = notifyAdapterDataSourcesRemove(parentPosition);
            notifyItemRangeRemoved(removeStart, size);
            notifyItemRangeChanged(removeStart, mAdapterData.size() - removeStart);
        } else {
            if (!expand) {
                return;
            }
            mExpandParentPositionList.add(parentPosition);
            int insertStart = notifyAdapterDataSourcesInsert(parentPosition);
            notifyItemRangeInserted(insertStart, size);
            notifyItemRangeChanged(insertStart, mAdapterData.size() - insertStart);
        }
    }

    public boolean isExpand(int parentPosition) {
        return mExpandParentPositionList.contains(parentPosition);
    }

    public List<Pair<Parent, List<Child>>> getData() {
        return mData;
    }

    public List<Pair<Integer, Object>> getAdapterData() {
        return mAdapterData;
    }

    public Parent getParentData(int parentPosition) {
        return mData.get(parentPosition).first;
    }

    public Child getChildData(int parentPosition, int childPosition) {
        return mData.get(parentPosition).second.get(childPosition);
    }

    @Override
    public int getItemViewType(int position) {
        return mAdapterData.get(position).first;
    }

    @NonNull
    @Override

    public RecyclerView.ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        if (viewType == PARENT_TYPE) {
            return onParentViewHolder(parent);
        }

        if (viewType == CHILD_TYPE) {
            return onChildViewHolder(parent);
        }

        return onCreateCommonViewHolder(parent, viewType);
    }

    @Override
    @SuppressWarnings("unchecked")
    public void onBindViewHolder(@NonNull RecyclerView.ViewHolder holder, int position) {
        Pair<Integer, Object> data = mAdapterData.get(position);
        int type = data.first;
        if (type == PARENT_TYPE) {
            holder.itemView.setOnClickListener(v -> {
                int parentPosition = getParentPosition(position);
                notifyExpand(parentPosition);

                onParentClick(holder, parentPosition);
            });

            onBindParentViewHolder(holder, (Parent) data.second);
        } else if (type == CHILD_TYPE) {
            holder.itemView.setOnClickListener(v -> {
                int parentPosition = getParentPosition(position);
                onChildClick(holder, parentPosition, position - getAdapterPosition(parentPosition) - 1);
            });
            onBindChildViewHolder(holder, (Child) data.second);
        } else {
            onBindCommonViewHolder(holder, position, data.second);
        }

    }

    private List<Child> getChildList(int parentPosition) {
        return mData.get(parentPosition).second;
    }

    protected int getParentPosition(int adapterPosition) {
        if (mExpandParentPositionList.isEmpty()) {
            return adapterPosition;
        }

        List<Integer> validExpandPosition = mExpandParentPositionList.stream().filter(it -> it < adapterPosition).sorted().collect(Collectors.toList());

        if (validExpandPosition.isEmpty()) {
            return adapterPosition;
        }

        int count = 0;
        for (int expandPosition : validExpandPosition) {
            //check last
            if (adapterPosition < expandPosition + count) {
                break;
            }
            count += getChildList(expandPosition).size();
            //check this
            if (adapterPosition <= expandPosition + count) {
                return expandPosition;
            }
        }

        // find 绝对位置
        return adapterPosition - count;
    }

    @Override
    public int getItemCount() {
        return mAdapterData.size();
    }


    protected abstract RecyclerView.ViewHolder onParentViewHolder(ViewGroup parent);

    protected abstract RecyclerView.ViewHolder onChildViewHolder(ViewGroup parent);

    /**
     * 存在其他类型的viewHolder时,需要重写的方法,
     *
     * @see #getItemViewType(int)
     */
    protected RecyclerView.ViewHolder onCreateCommonViewHolder(ViewGroup parent, int viewType) {
        throw new IllegalArgumentException("unSupport type " + viewType);
    }

    protected void onBindCommonViewHolder(RecyclerView.ViewHolder holder, int position, Object data) {
        throw new IllegalArgumentException("unSupport  CommonViewHolder " + holder);
    }

    /**
     * 根据object返回自定义Type
     *
     * @param object:数据
     * @param defaultType :默认类型{@link #CHILD_TYPE} {@link #PARENT_TYPE}
     */
    protected int getItemViewTypeFromObject(Object object, int defaultType) {
        return defaultType;
    }


    protected abstract void onBindChildViewHolder(RecyclerView.ViewHolder holder, Child data);

    protected abstract void onBindParentViewHolder(RecyclerView.ViewHolder holder, Parent data);

    /**
     * 是否支持parentPosition的下一个展开状态
     *
     * @param parentPosition   目标位置
     * @param nextExpandStatus 下一个展开状态
     */
    protected abstract boolean onSupportParentExpand(int parentPosition, boolean nextExpandStatus);

    protected void onParentClick(RecyclerView.ViewHolder holder, int parentPosition) {

    }

    protected void onChildClick(RecyclerView.ViewHolder holder, int parentPosition, int childPosition) {

    }
}

​

实例如下

class ExpandListFragment : BaseComposeFragment() {

    private val rAdapter = RAdapter()

    override fun onContentView(rootView: ComposeView) {
        rootView.apply {
            doOnAttach {
                rAdapter.notifyData(
                    listOf(
                        Pair(Parent(1), listOf(Child(2, 1))),
                        Pair(
                            Parent(2), listOf(
                                Child(2, 2), Child(3, 2), Child(4, 2),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 3), Child(3, 3), Child(4, 3),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                        Pair(
                            Parent(3), listOf(
                                Child(3, 4), Child(3, 4), Child(4, 4),
                            )
                        ),
                    )
                )

                "notify data".shortToast()
            }
        }.setContent {
            MyApplicationTheme {
                Surface(modifier = Modifier.fillMaxSize()) {
                    AndroidView(factory = {
                        RecyclerView(it).apply {
                            this.adapter = rAdapter
                            this.layoutManager = LinearLayoutManager(it)
                        }
                    }, modifier = Modifier.fillMaxSize())
                }
            }
        }
    }

    override fun onSubscribeValue() {

    }


    private class RAdapter : BaseExpandRecyclerAdapter<Parent, Child>() {

        override fun onChildClick(
            holder: RecyclerView.ViewHolder, parentPosition: Int, childPosition: Int
        ) {
            "click $parentPosition $childPosition ".shortToast()
        }

        override fun onParentClick(holder: RecyclerView.ViewHolder, parentPosition: Int) {
            "click $parentPosition ".shortToast()
        }

        override fun onBindChildViewHolder(holder: RecyclerView.ViewHolder, data: Child?) {
            data?.let {
                (holder.itemView as TextView).text = "parentId${data.parentId} selfId:${data.id} ${holder.javaClass.simpleName}"
            }
        }

        override fun onBindParentViewHolder(holder: RecyclerView.ViewHolder, data: Parent?) {
            data?.let {
                (holder.itemView as TextView).text = "selfId:${data.id} ${holder.javaClass.simpleName}"
            }
        }

        override fun onSupportParentExpand(
            parentPosition: Int, nextExpandStatus: Boolean
        ): Boolean {
            return true
        }

        override fun onParentViewHolder(parent: ViewGroup): RecyclerView.ViewHolder {
            return ParentViewHolder(TextView(parent.context))
        }

        override fun onChildViewHolder(parent: ViewGroup): RecyclerView.ViewHolder {
            return ChildViewHolder(TextView(parent.context))
        }
    }

}


data class Parent(val id: Int)
data class Child(val id: Int, val parentId: Int)

class ParentViewHolder(val view: View) : RecyclerView.ViewHolder(view)
class ChildViewHolder(val view: View) : RecyclerView.ViewHolder(view)

效果如下:

可折叠列表

### 实现类似微博评论回复功能 为了实现在 Android 应用中类似于微博的评论回复功能,主要涉及几个方面: #### 1. 数据结构设计 定义数据模型来存储每条评论及其子评论的信息。通常情况下,会创建一个 `Comment` 类表示单条评论,并且该类应该能够容纳多个子评论。 ```java public class Comment { private String id; private String content; private List<Comment> replies; public Comment(String id, String content) { this.id = id; this.content = content; this.replies = new ArrayList<>(); } // Getters and Setters... } ``` #### 2. 用户界面布局 构建用于显示评论列表以及输入框的 XML 布局文件。这里展示的是简化版的布局,实际项目可能更复杂一些。 ```xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="wrap_content"> <!-- RecyclerView to display comments --> <androidx.recyclerview.widget.RecyclerView android:id="@+id/recycler_view_comments" android:layout_width="match_parent" android:layout_height="0dp" android:layout_weight="1"/> <!-- Input area for writing a comment or reply --> <RelativeLayout android:layout_width="match_parent" android:layout_height="wrap_content"> <EditText android:id="@+id/edit_text_comment" android:hint="Write your comment here..." android:inputType="textMultiLine" android:maxLines="4" android:minHeight="?attr/listPreferredItemHeightSmall" android:paddingLeft="8dp" android:paddingRight="8dp" android:layout_toStartOf="@+id/button_send" android:layout_alignParentStart="true"/> <Button android:id="@+id/button_send" android:text="Send" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentEnd="true" android:layout_centerVertical="true"/> </RelativeLayout> </LinearLayout> ``` #### 3. Adapter 编写 编写适配器 (Adapter),以便于将评论对象绑定至视图组件并支持展开/折叠子评论的功能。 ```java public class CommentsAdapter extends RecyclerView.Adapter<CommentsAdapter.CommentViewHolder> { private final Context context; private final List<Comment> commentsList; public static class CommentViewHolder extends RecyclerView.ViewHolder { TextView tvContent; Button btnReply; RecyclerView rvReplies; ImageButton ibExpandCollapse; public CommentViewHolder(@NonNull View itemView) { super(itemView); tvContent = itemView.findViewById(R.id.tv_content); btnReply = itemView.findViewById(R.id.btn_reply); rvReplies = itemView.findViewById(R.id.rv_replies); ibExpandCollapse = itemView.findViewById(R.id.ib_expand_collapse); // Initialize the nested recyclerview adapter. RepliesAdapter repliesAdapter = new RepliesAdapter(); LinearLayoutManager layoutManager = new LinearLayoutManager(context); rvReplies.setLayoutManager(layoutManager); rvReplies.setAdapter(repliesAdapter); // Expand/Collapse button click listener ibExpandCollapse.setOnClickListener(v -> toggleVisibility(rvReplies)); } void bind(final Comment item){ tvContent.setText(item.getContent()); // Bind other properties as needed... // Handle Reply Click Event Here // Update visibility of child items based on current state updateChildItemsVisibility(item.isExpanded(),rvReplies); } } @NonNull @Override public CommentViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { LayoutInflater inflater = LayoutInflater.from(parent.getContext()); View view = inflater.inflate(R.layout.item_comment,parent,false); return new CommentViewHolder(view); } @Override public void onBindViewHolder(@NonNull CommentViewHolder holder, int position) { holder.bind(commentsList.get(position)); } @Override public int getItemCount() {return commentsList.size();} } // Similar implementation would be done for RepliesAdapter which handles individual replies within each main comment. ``` 上述代码片段展示了如何设置基本的评论系统框架[^1]。需要注意的是,在真实的应用开发过程中还需要考虑更多细节问题,比如网络请求管理、错误处理机制等。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值