当前位置: 首页 > news >正文

【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);}@Overridepublic int getItemViewType(int position) {return mAdapterData.get(position).first;}@NonNull@Overridepublic 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 lastif (adapterPosition < expandPosition + count) {break;}count += getChildList(expandPosition).size();//check thisif (adapterPosition <= expandPosition + count) {return expandPosition;}}// find 绝对位置return adapterPosition - count;}@Overridepublic 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 = rAdapterthis.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)

效果如下:

可折叠列表

http://www.xdnf.cn/news/866161.html

相关文章:

  • 基于大模型的结节性甲状腺肿智能诊疗系统技术方案
  • NLP学习路线图(二十三):长短期记忆网络(LSTM)
  • constexpr 是 C++11 引入的关键字
  • PocketFlow 快速入门指南
  • 阿里内推-6月新出HC
  • App 上线后还能加固吗?iOS 应用的动态安全补强方案实战分享(含 Ipa Guard 等工具组合)
  • 数学复习笔记 26
  • 雷卯针对易百纳G610Q-IPC-38E 模组防雷防静电方案
  • 2025年大模型平台落地实践研究报告|附75页PDF文件下载
  • MySQL 索引底层原理剖析:B+ 树结构、索引创建维护与性能优化策略全解读
  • x86 汇编逻辑运算全解析:从【位操作】到实际应用(AND,OR,NOT,XOR,TEST)
  • 缓存控制HTTP标头设置为“无缓存、无存储、必须重新验证”
  • Cursor 工具项目构建指南: Web Vue-Element UI 环境下的 Prompt Rules 约束(new Vue 方式)
  • 杰发科技AC7801——使用内部晶振
  • 极客时间-《搞定音频技术》-学习笔记
  • 大数据学习(128)-数据分析实例
  • Linux开发工具(apt,vim,gcc)
  • Fluence推出“Pointless计划”:五种方式参与RWA算力资产新时代
  • ISO 17387——解读自动驾驶相关标准法规(LCDAS)
  • 网络寻路--图论
  • DeepSeek+SpringAI实现流式对话
  • 读文献先读图:GO弦图怎么看?
  • 概念全解析:结构化数据,半结构化数据,非结构化数据分别是什么意思?
  • 中国区域30m/15天植被覆盖度数据集(2010-2022)
  • 【PDF提取表格】如何提取发票内容文字并导出到Excel表格,并将发票用发票号改名,基于pdf电子发票的应用实现
  • 基于若依前后分离版-用户密码错误锁定
  • 第二章 2.3 数据存储安全风险之数据存储风险防范
  • 湖北理元理律师事务所:债务化解中的心理重建与法律护航
  • 缓存击穿 缓存穿透 缓存雪崩
  • 强制刷新页面和改变当前地址栏地址而不刷新页面