Нещодавно я натрапив на цю велику істоту, яка дає діючу реалізацію перетягування ListView
, без зовнішньої залежності.
В основному він полягає у створенні власного адаптера, що розширюється ArrayAdapter
як внутрішній клас до діяльності, що містить ваш ListView
. На цьому адаптері тоді встановлюєтьсяonTouchListener
ваш список елементів, який буде сигналізувати про початок перетягування.
У цьому Gist вони встановлюють слухача конкретну частину макета елемента «Список» («ручку» елемента), так що випадково його не переміщують, натискаючи будь-яку його частину. Особисто я віддав перевагу onLongClickListener
замість цього, але це вирішувати саме вам. Ось уривок цієї частини:
public class MyArrayAdapter extends ArrayAdapter<String> {
private ArrayList<String> mStrings = new ArrayList<String>();
private LayoutInflater mInflater;
private int mLayout;
//constructor, clear, remove, add, insert...
@Override
public View getView(final int position, View convertView, ViewGroup parent) {
ViewHolder holder;
View view = convertView;
//inflate, etc...
final String string = mStrings.get(position);
holder.title.setText(string);
// Here the listener is set specifically to the handle of the layout
holder.handle.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View view, MotionEvent motionEvent) {
if (motionEvent.getAction() == MotionEvent.ACTION_DOWN) {
startDrag(string);
return true;
}
return false;
}
});
// change color on dragging item and other things...
return view;
}
}
Це також включає додавання onTouchListener
до ListView
, яке перевіряє, чи елемент перетягується, обробляє заміну та недійсність і зупиняє стан перетягування. Уривок цієї частини:
mListView.setOnTouchListener(new View.OnTouchListener() {
@Override
public boolean onTouch(View view, MotionEvent event) {
if (!mSortable) { return false; }
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN: {
break;
}
case MotionEvent.ACTION_MOVE: {
// get positions
int position = mListView.pointToPosition((int) event.getX(),
(int) event.getY());
if (position < 0) {
break;
}
// check if it's time to swap
if (position != mPosition) {
mPosition = position;
mAdapter.remove(mDragString);
mAdapter.insert(mDragString, mPosition);
}
return true;
}
case MotionEvent.ACTION_UP:
case MotionEvent.ACTION_CANCEL:
case MotionEvent.ACTION_OUTSIDE: {
//stop drag state
stopDrag();
return true;
}
}
return false;
}
});
Нарешті, ось, як виглядають методи stopDrag
та startDrag
методи, які керують включенням та відключенням процесу перетягування:
public void startDrag(String string) {
mPosition = -1;
mSortable = true;
mDragString = string;
mAdapter.notifyDataSetChanged();
}
public void stopDrag() {
mPosition = -1;
mSortable = false;
mDragString = null;
mAdapter.notifyDataSetChanged();
}