ItemTouchHleper实现RecyclerView侧滑删除

这篇博客用来简单复习一下SQLite以及实现RecyclerView的侧滑Item删除。
主界面
侧滑删除
这个Demo整体思路就是将一个人的姓名和金钱存进数据库,并将所有存进数据库的数据用RecyclerView显示出来,然后加上侧滑删除的实现。

SQLiite

数据的准备

这里首先一开始是前段时间找工作遇到的上机复试题。
点击按钮,显示出一个Dialog,然后输入15232.20,得到一万五千二百三十二元二角。
虽然很简单,但自己处理的不好。这里是后来自己又做了一次,稍微比复试好一些。思路就是首先在输入金钱额度的时候,将EditText输入的属性设置为只可以输入数字和小数点。

tie.setInputType(InputType.TYPE_CLASS_NUMBER|InputType.TYPE_NUMBER_FLAG_DECIMAL);

然后对输入的数字进行判断是否含有小数点。如果含有小数点,就利用Stirng.split()方法,将String拆分。这里需要注意的是,String.split(“.”)是无效的。
String [] s = str.split("\\.");需要使用转译字符 。split方法的参数可以为正则表达式。在正则表达式中” . ” 代表任意字符。之后便是对拆分的字符串的处理。

界面很简单就是有几个按钮,一个保存按钮,一个查看。保存就是将输入的人的名字和金额存进数据库。查看按钮会打开一个新的Activity来展示数据库中有的数据。

DBHelper

Android中提供了一个SQLiteOpenHelper类,直接继承这个类。

public class DBHelper extends SQLiteOpenHelper {
    private static final String DEFAULT_NAME = "app_1_person.db";
    private static final int DEFAULT_VERSION = 1;
    public DBHelper(Context context) {
        super(context,DEFAULT_NAME, null, DEFAULT_VERSION);
    }
    /**
     * 第一次被创建时会调用
     */
    @Override
    public void onCreate(SQLiteDatabase db) {
        db.execSQL(
                 "CREATE TABLE IF NOT EXISTS person" +
                         "(_id INTEGER PRIMARY KEY AUTOINCREMENT , name VARCHAR(20) , money TEXT)"
         );
    }
    /**
     * 版本更新时 会调用
     */
    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
         if (newVersion > oldVersion){
              //TODO
         }
    }
}

继承SQLiteOpenHelper就要实现onCreate和onUpgrade方法。其中,onCreate方法只会调用一次。构造方法有来两个,其中一个包含有4个属性。这里直接在super()方法中把一些属性写死了。

  • 第一个属性,context,上下文;
  • 第二个属性,DEFAULT_NAME,数据库文件名字以”.db”结尾,这里不是数据库表的名字,要区分。
  • 第三个属性,SQLiteDatabase.CursorFactory factory ,我这里设置为了null
  • 第四个属性, DEFAULT_VERSION,数据库的版本号,一开始一般默认为版本号为 1

onCreate()方法,用来建立数据库表单。我这里sql语句也很简单,需要注意的是一般以”_id”作为字段名,主键(PRIMARY KEY)为从0开始的整数。
onUpgrade()方法是用来更新数据库表单的。这里的更新是对表的更新。例如添加了字段,删除字段的时候。我这里暂时没有涉及,以后深入了解后,在做补充。

DBManager

这类个算作是对数据库操作的基本的一些封装。

public class DBManager {
    private DBHelper dbHelper;
    private SQLiteDatabase database;
    public DBManager(Context context) {
        dbHelper = new DBHelper(context);
        database = dbHelper.getWritableDatabase();
    }
    public void addPersonInfo(Person person){
        database.beginTransaction();//开启事务
        try{
            database.execSQL("INSERT INTO person VALUES(NULL,?,?)",new Object[]{person.getName(),person.getMoney()});
            database.setTransactionSuccessful();//设置事务完成成功
        }finally {
            database.endTransaction();//设置结束事务
        }
    }
    public boolean deleteByPerson(Person person){
            int n = database.delete("person","name =  ?",
                    new String[]{person.getName()});
            return n != -1;
    }
    /**
     * 拿到数据库中所有的Person并放入集合中
     * @return
     */
    public List<Person> getPersonListData(){
        List<Person> listData = new ArrayList<>();
        Cursor c = getAllPersonInfo();
        while (c.moveToNext()){
            Person person = new Person();
            person.setName(c.getString(c.getColumnIndex("name")));
            person.setMoney(c.getString(c.getColumnIndex("money")));
            listData.add(person);
        }
        c.close();
        return listData;
    }
    private Cursor getAllPersonInfo(){
        Cursor c = database.rawQuery("SELECT * FROM person", null);
        return c;
    }
    /**
     * 关闭  database;
     */
    public void closeDB() {
        database.close();
    }
}

构造方法中需要一个Context,通过context拿到DBHeplper对象dbHelper。dbHelper.getWritableDatabase()拿到SQLiteDatabase对象后,就可以执行sql语句,来实现增删改查。

在getPersonListData()方法中,首先拿到含有全部数据的Cursor。调用curosr.moveToNet()来判断是否还有下一条数据。通过Cursor拿到信息,需要先通过字段得到索引index,然后才能得到内容。

注意需要记得是要记得关闭数据库连接。


数据库部分的内容基本就这些。接下来就是使用RecyclerView将数据库中存储的内容显示出来。

使用ItemTouchHelper实现侧滑删除

利用Android v7包中提供的ItemTouchHelper可以实现侧滑,拖拽操作。

public class RVItemTouchHelper extends ItemTouchHelper.Callback {
    private static final float ALPHA_FULL = 1.0f;
    private ItemTouchHelperAdapter adapter;
    private boolean isLongPressDragEnabled = false, isItemViewSwipeEnabled = false;

    public RVItemTouchHelper(ItemTouchHelperAdapter adapter) {
        this.adapter = adapter;
    }

    /**
     * 支持长按开始拖拽
     * @return
     */

    public void setLongPressDragEnabled(boolean isLongPressDragEnabled) {
        this.isLongPressDragEnabled = isLongPressDragEnabled;
    }


    @Override
    public boolean isLongPressDragEnabled() {
        return isLongPressDragEnabled;
    }

    /**
     * 支持左右滑动
     *
     * @return
     */
    public void setItemViewSwipeEnabled(boolean itemViewSwipeEnabled) {
        isItemViewSwipeEnabled = itemViewSwipeEnabled;
    }

    @Override
    public boolean isItemViewSwipeEnabled() {
        return isItemViewSwipeEnabled;
    }

    /**
     * 持哪个方向的拖拽和滑动
     *
     * @param recyclerView
     * @param viewHolder
     * @return
     */
    @Override
    public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {

        if (recyclerView.getLayoutManager() instanceof GridLayoutManager) {
            int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.START | ItemTouchHelper.END;
            int swipeFlags = 0;
            return makeMovementFlags(dragFlags, swipeFlags);
        } else if (recyclerView.getLayoutManager() instanceof LinearLayoutManager) {
            int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN;
            int swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END;
            return makeMovementFlags(dragFlags, swipeFlags);
        }
        return 0;
    }

    /**
     * 拖拽后调用
     *
     * @param recyclerView
     * @param viewHolder
     * @param target
     * @return
     */
    @Override
    public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {
        if (viewHolder.getItemViewType() != target.getItemViewType()) {
            return false;
        }
        adapter.onItemMove(viewHolder.getAdapterPosition(), target.getAdapterPosition());
        return true;
    }

    /**
     * 滑动删除后调用
     *
     * @param viewHolder
     * @param direction
     */
    @Override
    public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
        adapter.onItemDelete(viewHolder.getAdapterPosition());
    }

    @Override
    public void onChildDraw(Canvas c, RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, float dX, float dY, int actionState, boolean isCurrentlyActive) {
        if (actionState == ItemTouchHelper.ACTION_STATE_SWIPE && dX > 200) {//滑动,手指没有离开屏幕
            //左右滑动时改变Item的透明度
            final float alpha = ALPHA_FULL - Math.abs(dX) / (float) viewHolder.itemView.getWidth();
            viewHolder.itemView.setAlpha(alpha);
            viewHolder.itemView.setTranslationX(dX);
        }
        super.onChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);
    }   
}

````
继承ItemTouchHelper.Callback后,要实现三个方法。

- getMovementFlags() 这个方法可以设置支持哪个方向的滑动和拖拽
- onMove()   拖拽之后会调用
- onSwiped   滑动后调用

定义一个接口,进行拖拽或者滑动后接口回调




<div class="se-preview-section-delimiter"></div>

```java
public interface ItemTouchHelperAdapter {
    void onItemMove(int formPosition, int toPosition);
    void onItemDelete(int position);
}

RecyclerView的Adapter实现ItemTouchHelperAdapter 接口

public class RVAdapter extends RecyclerView.Adapter<RVAdapter.MyViewHolder> implements ItemTouchHelperAdapter{
    private List<Person> data = new ArrayList<>();
    private Context context;
    private DBManager dbManager;
    public RVAdapter(Context context) {
        this.context = context;
    }

    public RVAdapter(Context context, DBManager dbManager) {
        this.context = context;
        this.dbManager = dbManager;
    }

    public void setData(List<Person> data) {
         if (data != null && data.size() > 0){
             this.data.clear();
             this.data.addAll(data);
             notifyDataSetChanged();
         }
    }

    @Override
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View v = LayoutInflater.from(context).inflate(R.layout.item_layout,parent,false);
        return new MyViewHolder(v);
    }

    @Override
    public void onBindViewHolder(MyViewHolder holder, int position) {
          Person p = data.get(position);
          if (p != null){
              holder.tv_name.setText(p.getName());
              holder.tv_money.setText(p.getMoney());
          }
    }

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

    @Override
    public void onItemMove(int formPosition, int toPosition) {}

    @Override
    public void onItemDelete(int position) {
        if (dbManager !=  null){
            boolean b = dbManager.deleteByPerson(data.get(position));
            if (b){
                //删除集合中的对应item的数据
                data.remove(position);
                //删除RecyclerView中对应item的子View
                notifyItemRemoved(position);
            }
        }
    }


    class MyViewHolder extends RecyclerView.ViewHolder{
        private TextView tv_name,tv_money;
        public MyViewHolder(View itemView) {
            super(itemView);
            tv_name = (TextView) itemView.findViewById(R.id.name_tv_item);
            tv_money = (TextView) itemView.findViewById(R.id.money_tv_item);
        }
    }
}

在Activity中实现ItemTouchHelper与RecyclerView,Adapter的绑定

RVItemTouchHelper callback = new RVItemTouchHelper(adapter);
        callback.setItemViewSwipeEnabled(true);//设置可以侧滑
        callback.setLongPressDragEnabled(false);//不可以拖拽
        ItemTouchHelper rvItemTouchHelper = new ItemTouchHelper(callback);
        rvItemTouchHelper.attachToRecyclerView(rv);

实现的效果很一般,基础的学习过程。

RVItemTouchHelper callback = new RVItemTouchHelper(adapter);
        callback.setItemViewSwipeEnabled(true);//设置可以侧滑
        callback.setLongPressDragEnabled(false);//不可以拖拽
        ItemTouchHelper rvItemTouchHelper = new ItemTouchHelper(callback);
        rvItemTouchHelper.attachToRecyclerView(rv);

实现的效果很一般,基础的学习过程。
代码

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 这个问题其实是一个编码问题。原本这个问句是“如何recyclerview侧滑删除item”,但是被编码转换了一下。如果要回答,应该是如下: 使用RecyclerView可以通过实现ItemTouchHelper.Callback类来实现侧滑删除item的效果。在实现ItemTouchHelper.Callback类中,可以通过onSwiped()方法实现侧滑删除item。在onSwiped()方法中,可以调用RecyclerView.Adapter的notifyItemRemoved()方法来通知RecyclerView移除该item。 ### 回答2: RecyclerViewAndroid上一个强大的列表控件。它提供了不同于ListView和GridView的Item视图回收机制以及更灵活的布局管理,同时也支持ItemTouchHelper类来实现方便的侧滑删除item的功能。 ItemTouchHelper是Android Support Library中为RecyclerView提供的一个辅助类,主要是用于支持RecyclerView中的拖拽和滑动事件。为了实现RecyclerView侧滑删除item的功能,我们需要使用ItemTouchHelper类,并实现ItemTouchHelper.Callback的各个方法(onMove、onSwiped等): 1. onMove():该方法用于处理RecyclerView中拖拽事件,通常用于实现Item的移动、排序等,我们可以在该方法中添加自己的代码,实现列表Item的移动。 2. onSwiped():该方法用于处理RecyclerView中的滑动事件,通常用于实现列表Item删除等功能,我们可以在该方法中添加自己的代码,实现Item删除的操作。 在我们实现侧滑删除Item的过程中,上述两个方法都需要实现。其中,onSwiped()方法中,我们通常会使用adapter的remove()方法将Item从列表中移除。 具体实现过程如下: 1.创建一个SwipeToDeleteCallback类,并继承自ItemTouchHelper.Callback类。 public class SwipeToDeleteCallback extends ItemTouchHelper.Callback { //实现 onMove() 方法 @Override public boolean onMove(@NonNull RecyclerView recyclerView, @NonNull RecyclerView.ViewHolder viewHolder, @NonNull RecyclerView.ViewHolder target) { return false; } //实现 onSwiped() 方法 @Override public void onSwiped(@NonNull RecyclerView.ViewHolder viewHolder, int direction) { //从数据源中移除 Item //adapter.remove(viewHolder.getAdapterPosition()) } //其他的实现方法... } 2.在onCreate()方法中创建SwipeToDeleteCallback实例,并将其绑定到RecyclerView上。 SwipeToDeleteCallback swipeHandler = new SwipeToDeleteCallback(this); ItemTouchHelper itemTouchHelper = new ItemTouchHelper(swipeHandler); itemTouchHelper.attachToRecyclerView(recyclerView); 3.在RecyclerView的Adapter中实现remove()方法,并在onBindViewHolder()方法中为每个Item添加GestureDetector。通过监听GestureDetector的滑动事件,判断当前Item是否向右滑动,并根据滑动的方向调用ItemTouchHelper.Callback的onSwiped()方法。 public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> { //数据源 List<String> mDataList = new ArrayList<>(); //添加Item操作 public void addItem(String item) { mDataList.add(item); notifyItemInserted(mDataList.size() - 1); } //移除 Item 操作 public void remove(int position) { mDataList.remove(position); notifyItemRemoved(position); } // 创建 ViewHolder @NonNull @Override public ViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) { //创建 Item 布局 View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.list_item, parent, false); return new ViewHolder(view); } // 绑定 ViewHolder @Override public void onBindViewHolder(@NonNull ViewHolder holder, int position) { //设置 Item 数据 holder.itemText.setText(mDataList.get(position)); //添加 GestureDetector 监听 Item 滑动事件 GestureDetectorCompat gestureDetector = new GestureDetectorCompat(holder.itemView.getContext(), new MyGestureListener(holder)); holder.itemView.setOnTouchListener((view, motionEvent) -> { gestureDetector.onTouchEvent(motionEvent); return true; }); } //获取数据源大小 @Override public int getItemCount() { return mDataList.size(); } //创建 ViewHolder static class ViewHolder extends RecyclerView.ViewHolder { private TextView itemText; ViewHolder(View itemView) { super(itemView); itemText = itemView.findViewById(R.id.item_text); } } //GestureDetector 监听器,用于监听 Item 的滑动事件 class MyGestureListener extends GestureDetector.SimpleOnGestureListener { private final ViewHolder mHolder; MyGestureListener(MyAdapter.ViewHolder holder) { mHolder = holder; } @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { final float xDiff = e2.getX() - e1.getX(); if (Math.abs(xDiff) > 100) { //向右滑动时调用 onSwiped()方法 if (xDiff > 0) { remove(mHolder.getAdapterPosition()); return true; } } return false; } } } 在上述功能实现过程中,我们继承了ItemTouchHelper.Callback来自定义每个Item的拖拽和滑动事件,使用ItemTouchHelper将这个Callback实例与RecyclerView绑定在一起,然后在Adapter中添加了remove()方法,以实现Item删除功能。最後,我们使用GestureDetector监听Item滑动事件,并根据滑动的方向调用ItemTouchHelper.Callback的onSwiped()方法。 在实现RecyclerView侧滑删除item的过程中,我们还可以对ItemView进行进一步的美化,例如在ItemView左右两侧添加删除和修改按钮等。总之,RecyclerView提供了足够灵活的机制,使得我们能够自由地扩展和定制列表控件的各种功能,为用户提供优秀的使用体验。 ### 回答3: RecyclerViewAndroid官方推出的一个列表控件,它比ListView具有更为灵活和高效的特性,支持数据动态更新、多种布局管理、 ItemTouchHelper操作等,而其支持的滑动操作也极大地方便了用户操作列表数据。针对RecyclerView侧滑删除item,可以通过以下方式实现。 1. ItemTouchHelper ItemTouchHelper是RecyclerView的内置工具类,可实现滑动删除、拖曳排序等操作。通过ItemTouchHelper.onSwiped()回调将删除数据源,再通过适配器的notifyItemRemoved()通知RecyclerView删除某个条目。 2. 自定义ItemDecoration 我们可以通过RecyclerViewItemDecoration来实现侧滑删除操作,通过自定义一个类继承ItemDecoration并在其onDraw()方法中绘制删除按钮,再通过onTouchEvent()方法实现删除操作,即可实现侧滑删除item。 3. 第三方库 目前市面上也有一些优秀的RecyclerView侧滑删除第三方库,如SwipeMenuRecyclerView、SwipeRecyclerView等。这些库都提供了美观且易于使用的侧滑删除控件,大大降低了开发者的工作量。 总之,RecyclerView侧滑删除功能符合用户的使用习惯,提高了用户的操作效率,是不可或缺的一个功能,希望开发者在设计列表页面时能够注重用户体验,添加RecyclerView侧滑删除功能。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值