Как прекратить смахивание предмета в представлении Recycler после достижения некоторого расстояния
У меня есть Recycler View в Android. Я приложил ItemTouchHelper
к нему. Я включил только смахивание слева направо.
Поэтому, когда я проводю пальцем по любому предмету в представлении Recycler, предмет начинает двигаться вправо, а слева я рисую текст. Это отлично. Мое требование заключается в том, что я хочу разрешить пользователю проводить пальцем только на некоторое расстояние, и когда это расстояние будет достигнуто и пользователь отпустит касание, предмет, который будет проведен, должен вернуться в свое положение влево.
Проблема в том, что, когда я провожу пальцем слева направо, изображение полностью удаляется с экрана. Как я могу ограничить, чтобы он был проведен только на некоторое расстояние? Как мне это сделать?
Вот мой код для обратного вызова элемента touch:
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);
}`
2 ответа
Замените эту строку в 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);
Затем в onSwiped обработчика обратного вызова вызовите notifyItemChanged на адаптере RecyclerView.
@Override
public void onSwiped(RecyclerView.ViewHolder viewHolder,
int direction) {
if (direction == ItemTouchHelper.RIGHT) {
adapter.notifyItemChanged(viewHolder.getAdapterPosition())
}
}
Небольшая проблема с notifyItemChanged заключается в том, что вы можете заметить быстрое мигание, когда представление восстанавливается в исходное положение.
В super.onChildDraw вы можете использовать это: dX/5
override fun onChildDraw(
c: Canvas,
recyclerView: RecyclerView,
viewHolder: RecyclerView.ViewHolder,
dX: Float,
dY: Float,
actionState: Int,
isCurrentlyActive: Boolean
) {
if (actionState == ItemTouchHelper.ACTION_STATE_SWIPE) {
val itemView = viewHolder.itemView
val height = itemView.bottom.toFloat() - itemView.top.toFloat()
val width = height / 3
val p = Paint()
if (dX < 0) {
p.color = Color.RED
val background = RectF(
itemView.right.toFloat() + dX,
itemView.top.toFloat(),
itemView.right.toFloat(),
itemView.bottom.toFloat()
)
c.drawRect(background, p)
val icon = BitmapFactory.decodeResource(
this@TimeLineFragment.resources,
R.drawable.ic_photo_delete
)
val margin = (dX / 5 - width) / 2
val iconDest = RectF(
itemView.right.toFloat() + margin,
itemView.top.toFloat() + width,
itemView.right.toFloat() + (margin + width),
itemView.bottom.toFloat() - width
)
c.drawBitmap(icon, null, iconDest, p)
}
} else {
c.drawColor(Color.TRANSPARENT, PorterDuff.Mode.CLEAR)
}
super.onChildDraw(
c,
recyclerView,
viewHolder,
dX / 5,
dY,
actionState,
isCurrentlyActive
)
}
}).attachToRecyclerView(binding.timeLineList)