【问题标题】:How to Stop Swiping the item in Recycler view after reaching some distance达到一定距离后如何停止在回收站视图中滑动项目
【发布时间】:2018-05-31 12:26:09
【问题描述】:

我在 android 中有一个 Recycler View。我已将ItemTouchHelper 附加到它。我只启用了从左到右的滑动。

因此,当我滑动 Recycler 视图的任何项目时,该项目开始向右移动,在左侧我绘制一个文本。 这可以 。 我的要求是,我希望只允许用户滑动到某个距离,当达到该距离并且用户释放触摸时,被滑动的项目应该回到它向左的位置。

问题是,当我从左向右滑动时,视图会完全滑出屏幕。我怎样才能限制它只能刷到一段距离? 我该怎么做?

这是我的项目触摸回调代码:

private void initSwipe(){
    ItemTouchHelper.SimpleCallback simpleItemTouchCallback = new ItemTouchHelper.SimpleCallback(0, ItemTouchHelper.RIGHT) {
            @Override
            public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {
                return false;
            }

            @Override
            public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
                int position = viewHolder.getAdapterPosition();
            }

        @Override
        public void onChildDraw(Canvas c, RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, float dX, float dY, int actionState, boolean isCurrentlyActive) {
            ViewHolder viewHolder1 = (ViewHolder)viewHolder;
            int position = viewHolder1.getAdapterPosition();
            MyItem item = dataList.get(position);
            if (item==null )return;
            if(actionState == ItemTouchHelper.ACTION_STATE_SWIPE){

                View itemView = viewHolder.itemView;
                float height = (float) itemView.getBottom() - (float) itemView.getTop();
                float width = height / 3;
                Paint p = new Paint();
                Paint textPaint = new Paint();
                textPaint.setColor(Color.WHITE);
                textPaint.setTextSize(20);

                if(dX > 0) {
                    p.setColor(Color.parseColor("#000000"));
                    RectF background = new RectF((float) itemView.getLeft(), (float) itemView.getTop(), dX,(float) itemView.getBottom());
                    c.drawRect(background,p);
                    //RectF icon_dest = new RectF((float) itemView.getLeft() + width ,(float) itemView.getTop() + width,(float) itemView.getLeft()+ 2*width,(float)itemView.getBottom() - width);
                    //c.drawBitmap(icon,null,icon_dest,p);
                    c.drawText(item.getDate(),(float)(itemView.getLeft() + width),(float)(itemView.getBottom() - width), textPaint);
                }
            }
            super.onChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);
        }
    };
    ItemTouchHelper itemTouchHelper = new ItemTouchHelper(simpleItemTouchCallback);
    itemTouchHelper.attachToRecyclerView(myRecycler);
}`

【问题讨论】:

    标签: android android-recyclerview itemtouchhelper


    【解决方案1】:

    在onChildDraw中替换这一行:

    super.onChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive);
    

    通过以下,它将最大 x 位置设置为 100,根据需要调整此值。

    float newDx = dX;
    if (newDx >= 100f) {
        newDx = 100f
    }
    super.onChildDraw(c, recyclerView, viewHolder, newDx, dY, actionState, isCurrentlyActive);
    

    Then 在回调处理程序的 onSwiped 中,调用 RecyclerView 适配器上的 notifyItemChanged。

     @Override
     public void onSwiped(RecyclerView.ViewHolder viewHolder,
         int direction) {
             if (direction == ItemTouchHelper.RIGHT) {
            adapter.notifyItemChanged(viewHolder.getAdapterPosition())
        }
     }
    

    notifyItemChanged 的​​一个小问题是,当视图恢复到其原始位置时,您可能会注意到快速闪烁。

    【讨论】:

    • 我们可以在不眨眼的情况下将它滑回原来的位置吗?
    • 这行得通,但是如何像 WhatsApp 一样在滑动结束之前降低滑动速度?
    • 你也可以通过在你的onChildDraw方法中粘贴这段代码sn-p来控制它if (actionState == ItemTouchHelper.ActionStateSwipe) { if (dX < 0) { itemView.TranslationX = dX / 5; } } else { base.OnChildDraw(c, recyclerView, viewHolder, dX, dY, actionState, isCurrentlyActive); } //itemViewviewHolder.ItemView