热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

RecyclerView实现拖拽排序效果

这篇文章主要为大家详细介绍了RecyclerView实现拖拽排序效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

效果就是这样,RecyclerView列表可拖拽排序,可删除,可添加;

RecyclerView给我们提供了一个手势器:

ItemTouchHelper helper = new ItemTouchHelper(new ItemTouchHelper.Callback() { 
  @Override 
  public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { 
   int dragFrlg = 0; 
   if (recyclerView.getLayoutManager() instanceof GridLayoutManager){ 
    dragFrlg = ItemTouchHelper.UP|ItemTouchHelper.DOWN|ItemTouchHelper.LEFT|ItemTouchHelper.RIGHT; 
   }else if(recyclerView.getLayoutManager() instanceof LinearLayoutManager){ 
    dragFrlg = ItemTouchHelper.UP|ItemTouchHelper.DOWN; 
   } 
   return makeMovementFlags(dragFrlg,0); 
  } 
 
  @Override 
  public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) { 
   //滑动事件 下面注释的代码,滑动后数据和条目错乱,被舍弃 
//   Collections.swap(datas,viewHolder.getAdapterPosition(),target.getAdapterPosition()); 
//   ap.notifyItemMoved(viewHolder.getAdapterPosition(),target.getAdapterPosition()); 
 
   //得到当拖拽的viewHolder的Position 
   int fromPosition = viewHolder.getAdapterPosition(); 
   //拿到当前拖拽到的item的viewHolder 
   int toPosition = target.getAdapterPosition(); 
   if (fromPosition  toPosition; i--) { 
     Collections.swap(datas, i, i - 1); 
    } 
   } 
   ap.notifyItemMoved(fromPosition, toPosition); 
   return true; 
  } 
 
  @Override 
  public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) { 
    //侧滑删除可以使用; 
  } 
 
  @Override 
  public boolean isLongPressDragEnabled() { 
   return true; 
  } 
  /** 
   * 长按选中Item的时候开始调用 
   * 长按高亮 
   * @param viewHolder 
   * @param actionState 
   */ 
  @Override 
  public void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) { 
   if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) { 
    viewHolder.itemView.setBackgroundColor(Color.RED); 
    //获取系统震动服务//震动70毫秒 
    Vibrator vib = (Vibrator) getSystemService(Service.VIBRATOR_SERVICE); 
    vib.vibrate(70); 
   } 
   super.onSelectedChanged(viewHolder, actionState); 
  } 
 
  /** 
   * 手指松开的时候还原高亮 
   * @param recyclerView 
   * @param viewHolder 
   */ 
  @Override 
  public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { 
   super.clearView(recyclerView, viewHolder); 
   viewHolder.itemView.setBackgroundColor(0); 
   ap.notifyDataSetChanged(); //完成拖动后刷新适配器,这样拖动后删除就不会错乱 
  } 
 }); 

设置给RecyclerView即可:

helper.attachToRecyclerView(rv); 

下面是完整的代码和Xml布局文件:

public class MainActivity extends AppCompatActivity implements View.OnClickListener { 
 private final String TAG = this.getClass().getSimpleName(); 
 private List d = Arrays.asList( 
   "A","B","C","D","E","F","G" 
   ,"H","I","J","K","L","M","N" 
   ,"O","P","Q","R","S","T" 
   ,"U","V","W","X","Y","Z"); 
 private RecyclerView rv ; 
 private Ap ap; 
 private List datas; 
 private EditText edAdd; 
 
 @Override 
 protected void onCreate(Bundle savedInstanceState) { 
  super.onCreate(savedInstanceState); 
  setContentView(R.layout.activity_main); 
  initData(); 
  rv = findViewById(R.id.rv); 
  edAdd =findViewById(R.id.et_add); 
  rv.setLayoutManager(new GridLayoutManager(this,3)); 
//  rv.setLayoutManager(new LinearLayoutManager(this)); 
  ap = new Ap(this, datas); 
  rv.setAdapter(ap); 
  helper.attachToRecyclerView(rv); 
  findViewById(R.id.tv).setOnClickListener(this); 
  findViewById(R.id.tv_add).setOnClickListener(this); 
 } 
 
 private void initData() { 
  datas = new ArrayList<>(); 
//  直接用d操作集合会崩溃,Arrays.asList集合不可增删改;详细可以看我的博客 
  for (int i = 0; i  extends RecyclerView.Adapter{ 
  private Context context; 
  public List stringList; 
  public Ap(Context context, List stringList) { 
   this.cOntext= context; 
   this.stringList = stringList; 
  } 
 
  @Override 
  public Ap.Vh onCreateViewHolder(ViewGroup parent, int viewType) { 
   return new Vh(LayoutInflater.from(context).inflate(R.layout.item_rv,null)); 
  } 
 
  @Override 
  public void onBindViewHolder(Ap.Vh holder, final int position) { 
   holder.tv.setText(stringList.get(position).toString()); 
   holder.iv.setOnClickListener(new View.OnClickListener() { 
    @Override 
    public void onClick(View view) { 
     remove(position); 
    } 
   }); 
  } 
 
  @Override 
  public int getItemCount() { 
   return stringList.size(); 
  } 
 
  public void add(T item){ 
   int position = stringList.size(); 
   stringList.add(item); 
   notifyItemInserted(position); 
  } 
 
  public void add(int position,T item){ 
   stringList.add(position,item); 
   notifyItemInserted(position); 
  } 
//  public void remove(T item) { 
//   final int position = stringList.indexOf(item); 
//   if (-1 == position) 
//    return; 
//   stringList.remove(item); 
//   notifyItemRemoved(position); 
//  } 
 
  public void remove(int position) { 
   stringList.remove(position); 
   notifyItemRemoved(position); 
   notifyItemRangeChanged(position,stringList.size()); 
  } 
 
  class Vh extends RecyclerView.ViewHolder { 
 
   public Vh(View itemView) { 
    super(itemView); 
    tv = itemView.findViewById(R.id.tv); 
    iv = itemView.findViewById(R.id.iv_delete); 
   } 
   public TextView tv; 
   public ImageView iv; 
  } 
 } 

 ItemTouchHelper helper = new ItemTouchHelper(new ItemTouchHelper.Callback() { 
  @Override 
  public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { 
   int dragFrlg = 0; 
   if (recyclerView.getLayoutManager() instanceof GridLayoutManager){ 
    dragFrlg = ItemTouchHelper.UP|ItemTouchHelper.DOWN|ItemTouchHelper.LEFT|ItemTouchHelper.RIGHT; 
   }else if(recyclerView.getLayoutManager() instanceof LinearLayoutManager){ 
    dragFrlg = ItemTouchHelper.UP|ItemTouchHelper.DOWN; 
   } 
   return makeMovementFlags(dragFrlg,0); 
  } 
 
  @Override 
  public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) { 
   //滑动事件 下面注释的代码,滑动后数据和条目错乱,被舍弃 
//   Collections.swap(datas,viewHolder.getAdapterPosition(),target.getAdapterPosition()); 
//   ap.notifyItemMoved(viewHolder.getAdapterPosition(),target.getAdapterPosition()); 
 
   //得到当拖拽的viewHolder的Position 
   int fromPosition = viewHolder.getAdapterPosition(); 
   //拿到当前拖拽到的item的viewHolder 
   int toPosition = target.getAdapterPosition(); 
   if (fromPosition  toPosition; i--) { 
     Collections.swap(datas, i, i - 1); 
    } 
   } 
   ap.notifyItemMoved(fromPosition, toPosition); 
   return true; 
  } 
 
  @Override 
  public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) { 
    //侧滑删除可以使用; 
  } 
 
  @Override 
  public boolean isLongPressDragEnabled() { 
   return true; 
  } 
  /** 
   * 长按选中Item的时候开始调用 
   * 长按高亮 
   * @param viewHolder 
   * @param actionState 
   */ 
  @Override 
  public void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) { 
   if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) { 
    viewHolder.itemView.setBackgroundColor(Color.RED); 
    //获取系统震动服务//震动70毫秒 
    Vibrator vib = (Vibrator) getSystemService(Service.VIBRATOR_SERVICE); 
    vib.vibrate(70); 
   } 
   super.onSelectedChanged(viewHolder, actionState); 
  } 
 
  /** 
   * 手指松开的时候还原高亮 
   * @param recyclerView 
   * @param viewHolder 
   */ 
  @Override 
  public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) { 
   super.clearView(recyclerView, viewHolder); 
   viewHolder.itemView.setBackgroundColor(0); 
   ap.notifyDataSetChanged(); //完成拖动后刷新适配器,这样拖动后删除就不会错乱 
  } 
 }); 
} 

Xml布局文件:

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
 
  
   
   
   
  
  
 
 

RecyclerView的Item:

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
  
  
 
  
  
 

资源链接Github

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • Android性能优化检测App卡顿
    在移动APP性能评测-流畅度评测中,我们介绍了如何准确客观评价APP的流畅度,最终采用SM指标来评价应用的流畅度,在知道如何评价流畅度之后 ... [详细]
  • 本文分析和介绍了GLo ... [详细]
  • Git(1)
    安装Git完毕(在开始菜单打开的话,打开的不是你想要的路径,切换路径很麻烦)1.D盘新建GitTest文件夹2.打开GitTest,在空白的地方右键,3.单击GitBashHere ... [详细]
  • MyBatis模糊查询和多条件查询一、ISmbmsUserDao层根据姓名模糊查询publicListgetUser();多条件查询publicList ... [详细]
  • 在MirrorAPI中,我们可以使用以下内容: ... [详细]
  • 一、基本Tag1.Tag定义了Item实例,有三个基本的属性来定义一个item实例,大小写敏感。id-每个Item实例的唯一标识type ... [详细]
  • Android的四种启动模式
     对Android的启动模式不是很了解,这里记录下简单的理解内容以便日后查看。 Androi的四种启动模式分别为:standard,singleTop,singleTask,sing ... [详细]
  • 成功入职字节跳动Android岗,定级22,入职就是30K16薪
    Android线程间切换用什么,Handler的运行机制是什么?Android处理异步任务用什么,AsyncTask线程池溢出是怎么回事& ... [详细]
  • 一、使用ContentProvider(内容提供者)共享数据ContentProvider在android中的作用是对外共享数据,也就是说 ... [详细]
  • 安全加密C语言库OpenSSL,在Android中服务器和客户端之间的签名验证和数据加密通信等。OpenSSL系列文章:一、AndroidCMake轻松 ... [详细]
  • nvmw安装,用于控制node版本;
    之前一直使用的是nodev2.2.0版本,挺说新版本的node解决了npm安装插件产生文件夹结构过深的问题,所以就想更新试试;上网一看才发现,尼玛的node已经到了6.+版本了,好 ... [详细]
  • 在ROS系统中,参数读写一般通过xml或者yaml格式的文件,其中yaml用得比较多。这是一种可读性高,轻量级的标记语言,简单好用。对于yaml文件,ros中用的较早版本的yaml- ... [详细]
  • 前端微服务二
    为了解决庞大的一整块后端服务带来的变更与扩展方面的限制,出现了微服务架构(Microservices):微服务是面向服务架构(SOA)的一种变体,把应用程序设计成一系列松耦合的细粒 ... [详细]
  • 开发笔记:(源码开放) React + webpack3 多页面应用 及 常见问题解答
    本文由编程笔记#小编为大家整理,主要介绍了(源码开放)React+webpack3多页面应用及常见问题解答相关的知识,希望对你有一定的参考价值。 ... [详细]
  • 以下内容|尾部_quarkus实战之一:准备工作
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了quarkus实战之一:准备工作相关的知识,希望对你有一定的参考价值。欢迎访问我的GitHub ... [详细]
author-avatar
手机用户2502885897
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有