Android使用RecycleView實現(xiàn)拖拽交換item位置
本文實例為大家分享了RecycleView實現(xiàn)拖拽交換item位置的具體代碼,供大家參考,具體內容如下
老規(guī)矩,先來一張效果圖:

相比起ListView而言,RecycleView實現(xiàn)拖拽交換位置的效果要簡單很多,因為通過SDK中的ItemTouchHelper工具類可以輕松的實現(xiàn)這種效果,并且一套代碼支持所有布局方式;而ListView的話則需要通過生成View的緩存鏡像設置到ImageView中,然后通過WindowManager來操作該ImageView,具體怎么實現(xiàn)這里就不展開講解了.回歸到ItemTouchHelper話題上,這個工具類我們需要關心的方法只有一個,即:
public void attachToRecyclerView(@Nullable RecyclerView recyclerView) {}
通過名字也可以知道其作用就是和RecyclerView 建立關系.
而我們真正需要關心的地方就是ItemTouchHelper的內部類ItemTouchHelper.Callback,它是一個抽象類,需要我們去實現(xiàn)下面這幾個關鍵的抽象方法以及重寫幾個非抽象的方法:
//決定拖拽/滑動的方向
public abstract int getMovementFlags(RecyclerView recyclerView,
ViewHolder viewHolder);
//和位置交換有關,可用于實現(xiàn)drag功能
public abstract boolean onMove(RecyclerView recyclerView,
ViewHolder viewHolder, ViewHolder target);
//和滑動有關,可用于實現(xiàn)swipe功能
public abstract void onSwiped(ViewHolder viewHolder, int direction);
//是否長按啟用拖拽功能,默認是true
public boolean isLongPressDragEnabled() {return true;}
//是否支持滑動,默認true
public boolean isItemViewSwipeEnabled() {return true;}
//和目標View的狀態(tài)改變有關,例如drag,swipe,ide
public void onSelectedChanged(ViewHolder viewHolder, int actionState) {}
//和移除View的狀態(tài)有關,通常用于清除在onSelectedChanged,onChildDraw中對View設置的動畫
public void clearView(RecyclerView recyclerView, ViewHolder viewHolder) {}
當創(chuàng)建完ItemTouchHelper.Callback的實現(xiàn)類,我這里稱之為SimpleItemTouchHelperCallback后,還需要將數(shù)據(jù)的變化以及View的狀態(tài)通知到RecycleView的Adapter中,為了達到解耦的目的,通常可以通過定義接口來實現(xiàn),在SimpleItemTouchHelperCallback的構造方法中傳入該解耦接口的引用,并讓RecycleView的Adapter實現(xiàn)該解耦的接口,這樣就實現(xiàn)了這2個類的通信問題了.
來看看我定義的解耦接口:
/**
* 定義RecycleView的Adapter和SimpleItemTouchHelperCallback直接交互的接口方法
* Created by mChenys on 2017/2/16.
*/
public interface ItemTouchHelperAdapter {
//數(shù)據(jù)交換
void onItemMove(RecyclerView.ViewHolder source, RecyclerView.ViewHolder target);
//數(shù)據(jù)刪除
void onItemDissmiss(RecyclerView.ViewHolder source);
//drag或者swipe選中
void onItemSelect(RecyclerView.ViewHolder source);
//狀態(tài)清除
void onItemClear(RecyclerView.ViewHolder source);
}
完整的SimpleItemTouchHelperCallback代碼如下:
/**
* 處理RecycleView的選中,拖拽移動,拖拽刪除的實現(xiàn)類
* Created by mChenys on 2017/2/16.
*/
public class SimpleItemTouchHelperCallback extends ItemTouchHelper.Callback {
private ItemTouchHelperAdapter mAdapter;
public SimpleItemTouchHelperCallback(ItemTouchHelperAdapter adapter) {
mAdapter = adapter;
}
@Override
public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {
//int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN; //允許上下的拖動
//int dragFlags =ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT; //允許左右的拖動
//int swipeFlags = ItemTouchHelper.LEFT; //只允許從右向左側滑
//int swipeFlags = ItemTouchHelper.DOWN; //只允許從上向下側滑
//一般使用makeMovementFlags(int,int)或makeFlag(int, int)來構造我們的返回值
//makeMovementFlags(dragFlags, swipeFlags)
int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN | ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT; //允許上下左右的拖動
return makeMovementFlags(dragFlags, 0);
}
@Override
public boolean isLongPressDragEnabled() {
return true;//長按啟用拖拽
}
@Override
public boolean isItemViewSwipeEnabled() {
return false; //不啟用拖拽刪除
}
@Override
public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder source, RecyclerView.ViewHolder target) {
//通過接口傳遞拖拽交換數(shù)據(jù)的起始位置和目標位置的ViewHolder
mAdapter.onItemMove(source, target);
return true;
}
@Override
public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
//移動刪除回調,如果不用可以不用理
// mAdapter.onItemDissmiss(viewHolder);
}
@Override
public void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) {
super.onSelectedChanged(viewHolder, actionState);
if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) {
//當滑動或者拖拽view的時候通過接口返回該ViewHolder
mAdapter.onItemSelect(viewHolder);
}
}
@Override
public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {
super.clearView(recyclerView, viewHolder);
if (!recyclerView.isComputingLayout()) {
//當需要清除之前在onSelectedChanged或者onChildDraw,onChildDrawOver設置的狀態(tài)或者動畫時通過接口返回該ViewHolder
mAdapter.onItemClear(viewHolder);
}
}
}
RecycleView.Adapter實現(xiàn)類代碼
/**
* Created by mChenys on 2017/2/15.
*/
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyViewHolder> implements ItemTouchHelperAdapter {
...
@Override
public MyAdapter.MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
...
}
@Override
public void onBindViewHolder(final MyAdapter.MyViewHolder holder, int position) {
...
}
@Override
public int getItemCount() {
//注意:這里最少有一個,因為有多了一個添加按鈕
return null == mData ? 1 : mData.size() + 1;
}
@Override
public void onItemMove(RecyclerView.ViewHolder source,
RecyclerView.ViewHolder target) {
int fromPosition = source.getAdapterPosition();
int toPosition = target.getAdapterPosition();
if (fromPosition < mData.size() && toPosition < mData.size()) {
//交換數(shù)據(jù)位置
Collections.swap(mData, fromPosition, toPosition);
//刷新位置交換
notifyItemMoved(fromPosition, toPosition);
}
//移動過程中移除view的放大效果
onItemClear(source);
}
@Override
public void onItemDissmiss(RecyclerView.ViewHolder source) {
int position = source.getAdapterPosition();
mData.remove(position); //移除數(shù)據(jù)
notifyItemRemoved(position);//刷新數(shù)據(jù)移除
}
@Override
public void onItemSelect(RecyclerView.ViewHolder viewHolder) {
//當拖拽選中時放大選中的view
viewHolder.itemView.setScaleX(1.2f);
viewHolder.itemView.setScaleY(1.2f);
}
@Override
public void onItemClear(RecyclerView.ViewHolder viewHolder) {
//拖拽結束后恢復view的狀態(tài)
viewHolder.itemView.setScaleX(1.0f);
viewHolder.itemView.setScaleY(1.0f);
}
public class MyViewHolder extends RecyclerView.ViewHolder {
...
public MyViewHolder(View itemView) {
super(itemView);
...
}
}
}
MainActivity的使用方式
/**
* Created by mChenys on 2017/2/16.
*/
public class MainActivity extends AppCompatActivity {
...
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
...
mRecyclerView = (RecyclerView) findViewById(R.id.recycleView);
//創(chuàng)建adapter
MyAdapter myAdapter = new MyAdapter(this, mData);
//設置默認的布局方式
mRecyclerView.setLayoutManager(new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false));
//設置adapter
mRecyclerView.setAdapter(myAdapter);
//創(chuàng)建SimpleItemTouchHelperCallback
ItemTouchHelper.Callback callback = new SimpleItemTouchHelperCallback(myAdapter);
//用Callback構造ItemtouchHelper
ItemTouchHelper touchHelper = new ItemTouchHelper(callback);
//調用ItemTouchHelper的attachToRecyclerView方法建立聯(lián)系
touchHelper.attachToRecyclerView(mRecyclerView);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
...
return super.onCreateOptionsMenu(menu);
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
...
return super.onOptionsItemSelected(item);
}
}
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Win10下android studio開發(fā)環(huán)境配置圖文教程
這篇文章主要為大家詳細介紹了Win10下android studio開發(fā)環(huán)境配置圖文教程,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-07-07
Flutter改變狀態(tài)變量是否必須寫在setState回調詳解
這篇文章主要為大家介紹了Flutter改變狀態(tài)變量是否必須寫在setState回調里的原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-11-11
Android通過HTTP協(xié)議實現(xiàn)上傳文件數(shù)據(jù)
這篇文章主要為大家詳細介紹了Android通過HTTP協(xié)議實現(xiàn)上傳文件數(shù)據(jù),具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-09-09
Android自定義Drawable之在Drawable中部指定透明區(qū)域方法示例
對于不同的屏幕密度、不同的設備方向,不同的語言和區(qū)域,都會涉及到備選 drawable 資源,下面這篇文章主要給你大家介紹了關于Android自定義Drawable之在Drawable中部指定透明區(qū)域的相關資料,需要的朋友可以參考下2018-07-07
Android Drawerlayout實現(xiàn)側滑菜單效果
這篇文章主要為大家詳細介紹了Android Drawerlayout實現(xiàn)側滑菜單效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-10-10
Android實現(xiàn)類似IOS右滑返回的效果(原因分析及解決辦法)
這篇文章主要介紹了Android實現(xiàn)類似IOS右滑返回的效果,非常不錯,具有參考借鑒價值,需要的朋友參考下2017-03-03
Android開發(fā)之進度條ProgressBar的示例代碼
本篇文章主要介紹了Android開發(fā)之進度條ProgressBar的示例代碼,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-03-03
分享10個很棒的學習Android開發(fā)的網(wǎng)站
我推薦的網(wǎng)站,都是我在學習Android 開發(fā)過程中發(fā)現(xiàn)的好網(wǎng)站,給初學者一些建議,少走一些彎路2015-03-03

