Android listview的滑動沖突解決方法
在Android開發(fā)的過程中,有時候會遇到子控件和父控件都要滑動的情況,尤其是當(dāng)子控件為listview的時候。就比如在一個ScrollView里有一個listview,這種情況較常見,就會出現(xiàn)這種滑動沖突的情況。這種情況也比較常見,有時候就是這樣,沒法,但是,了解事件分發(fā)的我們知道應(yīng)該怎么處理這樣的事情
有兩點需要注意:
一般來說,view的onTouchEvent返回true,即消耗點擊事件,viewgroup的onInterceptTouchEvent返回false,即不攔截點擊事件,這一點從android源碼中可以看出來。但是listview的父類AbsListView重寫了onInterceptTouchEvent,返回了true,注意這里不是一定返回true,但是我覺得這一點可以先忽略。
onTouchEvent和onInterceptTouchEvent的調(diào)用順序。點擊事件從父控件向子控件傳遞,如果父控件不攔截,則交由子控件攔截,如果父控件攔截了,則交由父控件的onTouchEvent處理,如果最終處理點擊事件的控件的onTouchEvent返回了false,則將會直接調(diào)用其父控件的onTouchEvent,如此向上類推
其實解決方法也很簡單:重寫父控件的onInterceptTouchEvent函數(shù),在move的時候根據(jù)需要返回true,比如左右滑動返回true,其他情況均返回false。這樣,當(dāng)左右滑動的時候,由于onInterceptTouchEvent返回了true,父控件就能處理,其他情況,事件將傳遞到listview中,listview自身可以處理上下滑動。
@Override public boolean onInterceptTouchEvent(MotionEvent ev) { Log.d(TAG, "onInterceptTouchEvent-slop:"+mTouchSlop); final int action = ev.getAction(); if ((action == MotionEvent.ACTION_MOVE) && (mTouchState != TOUCH_STATE_REST)) { return true; } final float x = ev.getX(); final float y = ev.getY(); switch (action) { case MotionEvent.ACTION_MOVE: final int xDiff = (int)Math.abs(mLastMotionX-x); if (xDiff>mTouchSlop) { mTouchState = TOUCH_STATE_SCROLLING; } break; case MotionEvent.ACTION_DOWN: mLastMotionX = x; mLastMotionY = y; mTouchState = mScroller.isFinished()? TOUCH_STATE_REST : TOUCH_STATE_SCROLLING; break; case MotionEvent.ACTION_CANCEL: case MotionEvent.ACTION_UP: mTouchState = TOUCH_STATE_REST; break; } return mTouchState != TOUCH_STATE_REST; } 感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
新聞熱點
疑難解答
圖片精選