Android程序開(kāi)發(fā)之ListView 與PopupWindow實(shí)現(xiàn)從左向右滑動(dòng)刪除功能
文章實(shí)現(xiàn)的功能是:在ListView的Item上從右向左滑時(shí),出現(xiàn)刪除按鈕,點(diǎn)擊刪除按鈕把Item刪除。
看過(guò)文章后,感覺(jué)沒(méi)有必要把dispatchTouchEvent()和onTouchEvent()兩個(gè)方法都重寫,只要重寫onTouchEvent就好了。于是對(duì)代碼作了一些調(diào)整:
public class MyListView extends ListView { private static final String TAG = "MyListView"; private int mTouchSlop; private int mXDown; private int mYDown; private int mCurrentPosition; private View mCurrentView; private PopupWindow mPopupWindow; private LayoutInflater mInflater; private boolean isSliding = false; // 為刪除按鈕提供一個(gè)回調(diào)接口 private DelButtonClickListener mListener; private Button mDelBtn; private int mPopupWindowHeight; private int mPopupWindowWidth; public MyListView(Context context, AttributeSet attrs) { super(context, attrs); mInflater = LayoutInflater.from(context); mTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop(); View view = mInflater.inflate(R.layout.delete_btn, null); mDelBtn = (Button) view.findViewById(R.id.id_item_btn); mPopupWindow = new PopupWindow(view, LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT); // 如果需要通過(guò)點(diǎn)擊PopupWindow之外的地方使其消失,則需要setFocusable(true). mPopupWindow.setFocusable(true); // Android 6.0以前的版本需要setBackgroundDrawable(), // 才能實(shí)現(xiàn)通過(guò)點(diǎn)擊PopupWindow之外的地方使其消失的功能。 mPopupWindow.setBackgroundDrawable(new ColorDrawable(0)); // 先調(diào)用下measure,否則拿不到寬和高 mPopupWindow.getContentView().measure(0, 0); mPopupWindowHeight = mPopupWindow.getContentView().getMeasuredHeight(); mPopupWindowWidth = mPopupWindow.getContentView().getMeasuredWidth(); } @Override public boolean onTouchEvent(MotionEvent ev) { int action = ev.getAction(); int x = (int) ev.getX(); int y = (int) ev.getY(); switch (action){ case MotionEvent.ACTION_DOWN: isSliding = false; mXDown = x; mYDown = y; mCurrentPosition = pointToPosition(mXDown, mYDown); View view = getChildAt(mCurrentPosition - getFirstVisiblePosition()); mCurrentView = view; break; case MotionEvent.ACTION_MOVE: int dx = x - mXDown; int dy = y - mYDown; Log.d(TAG, "mTouchSlop = " + mTouchSlop + ", dx = " + dx + ", dy = " + dy); if(mXDown > x && Math.abs(dx) > mTouchSlop && Math.abs(dy) < mTouchSlop){ Log.d(TAG, "isSliding"); isSliding = true; int[] location = new int[2]; mCurrentView.getLocationOnScreen(location); mPopupWindow.setAnimationStyle(R.style.popwindow_delete_btn_anim_style); mPopupWindow.update(); Log.d(TAG, "Height: " + mCurrentView.getHeight() + "," + mPopupWindow.getHeight()); mPopupWindow.showAtLocation(mCurrentView, Gravity.NO_GRAVITY, location[0] + mCurrentView.getWidth(), location[1] + mCurrentView.getHeight() / 2 - mPopupWindowHeight / 2); mDelBtn.setOnClickListener(new OnClickListener() { @Override public void onClick(View v) { mListener.clickHappend(mCurrentPosition); mPopupWindow.dismiss(); } }); } case MotionEvent.ACTION_UP: // isSliding 如果這里恢復(fù)為false,則后面會(huì)執(zhí)行super.onTouchEvent事件, // 而AbsListView的onTouchEvent調(diào)用了onTouchUp方法,在onTouchUp方法中有可能執(zhí)行 // performClick.run() --> performItemClick() --> super.performItemClick // --> mOnItemClickListener.onItemClick,這樣最終觸發(fā)Item的點(diǎn)擊。 // 因此此處依舊保持isSliding為true的狀態(tài),而在ACTION_DOWN事件中恢復(fù)isSliding為false, // 畢竟每個(gè)事件都以ACTION_DOWN開(kāi)始。 //isSliding = false; } if(isSliding){ return true; } return super.onTouchEvent(ev); } public void setDelButtonClickListener(DelButtonClickListener listener){ mListener = listener; } interface DelButtonClickListener{ public void clickHappend(int position); } }
通過(guò)這個(gè)例子學(xué)習(xí)到:
1、ListView的Item點(diǎn)擊事件的觸發(fā)過(guò)程:
自定義ListView的onTouchEvent() ---調(diào)用super.onTouchEvent()---> AbsListView.onTouchEvent() ---MotionEvent.ACTION_UP---> AbsListView.onTouchUp()
---(有可能)調(diào)用performClick.run()---> AbsListView.PerformClick.run() ---調(diào)用performItemClick()---> AbsListView.performItemClick()
---(有可能)調(diào)用super.performItemClick()---> AdapterView.performItemClick() ---mOnItemClickListener.onItemClick---> OnItemClickListener.onItemClick()
也就是Item的點(diǎn)擊事件是在MotionEvent.ACTION_UP事件完成的,這樣在自定義ListView的onTouchEvent()中,對(duì)MotionEvent.ACTION_UP直接return true消費(fèi)掉事件,而不要調(diào)用super.onTouchEvent。這樣就避免了刪除按鈕與Item點(diǎn)擊事件的沖突。
2、PopupWindow--通過(guò)點(diǎn)擊PopupWindow之外的地方使其消失
a、需要調(diào)用setFocusable()方法;
b、Android 6.0以前的版本需要setBackgroundDrawable()(具體原因見(jiàn):PopupWindow的使用)。
以上所述是小編給大家介紹的Android程序開(kāi)發(fā)之ListView 與PopupWindow實(shí)現(xiàn)滑動(dòng)刪除功能,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
基于RecyclerView實(shí)現(xiàn)橫向GridView效果
這篇文章主要為大家詳細(xì)介紹了基于RecyclerView實(shí)現(xiàn)橫向GridView效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07Android fragment實(shí)現(xiàn)按鈕點(diǎn)擊事件的示例講解
下面小編就為大家分享一篇Android fragment實(shí)現(xiàn)按鈕點(diǎn)擊事件的示例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-01-01Android 系統(tǒng)相機(jī)拍照后相片無(wú)法在相冊(cè)中顯示解決辦法
這篇文章主要介紹了Android 系統(tǒng)相機(jī)拍照后相片無(wú)法在相冊(cè)中顯示解決辦法的相關(guān)資料,需要的朋友可以參考下2016-12-12android圖像繪制(四)自定義一個(gè)SurfaceView控件
自定義控件(類似按鈕等)的使用,自定義一個(gè)SurfaceView。如某一塊的動(dòng)態(tài)圖(自定義相應(yīng)),或者類似UC瀏覽器下面的工具欄,感興趣的朋友可以了解下2013-01-01Android ListView優(yōu)化之提高android應(yīng)用效率
android listview優(yōu)化做的好是提高androoid應(yīng)用效率的前提條件,本文給大家介紹Android ListView優(yōu)化之提高android應(yīng)用效率,對(duì)android listview優(yōu)化相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)吧2015-12-12Android開(kāi)發(fā)中的幾種網(wǎng)絡(luò)請(qǐng)求方式詳解
本篇文章主要包括Android中的幾種網(wǎng)絡(luò)請(qǐng)求方式詳解,具有一定的參考價(jià)值,有需要的可以了解一下。2016-11-11Flutter多項(xiàng)選擇彈窗實(shí)現(xiàn)詳解
這篇文章介紹了Flutter多項(xiàng)選擇彈窗實(shí)現(xiàn)詳解,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧<BR>2021-11-11