中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Android應用的中滑動事件出現沖突如何解決

發布時間:2020-11-25 17:01:51 來源:億速云 閱讀:185 作者:Leah 欄目:移動開發

Android應用的中滑動事件出現沖突如何解決?針對這個問題,這篇文章詳細介紹了相對應的分析和解答,希望可以幫助更多想解決這個問題的小伙伴找到更簡單易行的方法。

外部攔截法

外部攔截法是指在有點擊事件時都要經過父容器,那么在父容器時如果需要攔截就攔截自己處理,不需要則傳遞給下一層進行處理,下面看個例子:

首先定義一個水平滑動的HorizontalScrollViewEx,看主要代碼

主要的攔截是需要重寫onInterceptTouchEvent

@Override 
 public boolean onInterceptTouchEvent(MotionEvent ev) { 
 boolean intercepted = false; 
 int x = (int) ev.getX(); 
 int y = (int) ev.getY(); 
 switch (ev.getAction()) { 
 case MotionEvent.ACTION_DOWN: 
 //down事件不攔截,否則無法傳給子元素 
 intercepted = false; 
 if (!mScroller.isFinished()) { 
  mScroller.abortAnimation(); 
  intercepted = true; 
 } 
 break; 
 case MotionEvent.ACTION_MOVE: 
 int deltaX = x - mLastXIntercept; 
 int deltaY = y - mLastYIntercept; 
 //水平滑動則攔截 
 if (Math.abs(deltaX) > Math.abs(deltaY) + 5) { 
  intercepted = true; 
 } else { 
  intercepted = false; 
 } 
 break; 
 case MotionEvent.ACTION_UP: 
 //不攔截,否則子元素無法收到 
 intercepted = false; 
 break; 
 } 
 //因為當ViewGroup中的子View可能消耗了down事件,在onTouchEvent無法獲取, 
 // 無法對mLastX賦初值,所以在這里賦值一次 
 mLastX = x; 
 mLastY = y; 
 mLastYIntercept = y; 
 mLastXIntercept = x; 
 return intercepted; 
 } 

在down事件不需要攔截,返回false,否則的話子view無法收到事件,將全部會由父容器處理,這不是希望的;up事件也要返回false,否則最后子view收不到。

看看move事件,當水平滑動距離大于豎直距離時,代表水平滑動,返回true,由父類來進行處理,否則交由子view處理。這里move事件就是主要的攔截條件判斷,如果你遇到的不是水平和豎直的條件這么簡單,就可以在這里進行改變,比如,ScrollView嵌套了ListView,條件就變成,當ListView滑動到底部或頂部時,返回true,交由父類滑動處理,否則自身ListView滑動。

在onTouchEvent中主要是做的滑動切換的處理

@Override 
 public boolean onTouchEvent(MotionEvent event) { 
 mVelocityTracker.addMovement(event); 
 int x = (int) event.getX(); 
 int y = (int) event.getY(); 
 switch (event.getAction()) { 
 case MotionEvent.ACTION_DOWN: 
 if (!mScroller.isFinished()) { 
  mScroller.abortAnimation(); 
 } 
 break; 
 case MotionEvent.ACTION_MOVE: 
 int deltaX = x - mLastX; 
 int deltaY = y - mLastY; 
 if (getScrollX() < 0) { 
  scrollTo(0, 0); 
 } 
 scrollBy(-deltaX, 0); 
 break; 
 case MotionEvent.ACTION_UP: 
 int scrollX = getScrollX(); 
 mVelocityTracker.computeCurrentVelocity(1000); 
 float xVelocityTracker = mVelocityTracker.getXVelocity(); 
 if (Math.abs(xVelocityTracker) > 50) {//速度大于50則滑動到下一個 
  mChildIndex = xVelocityTracker > 0 &#63; mChildIndex - 1 : mChildIndex + 1; 
 } else { 
  mChildIndex = (scrollX + mChildWith / 2) / mChildWith; 
 } 
 mChildIndex = Math.max(0, Math.min(mChildIndex, mChildrenSize - 1)); 
 int dx = mChildIndex * mChildWith - scrollX; 
 smoothScrollBy(dx, 0); 
 mVelocityTracker.clear(); 
 break; 
 } 
 mLastY = y; 
 mLastX = x; 
 return true; 
 } 

在這個嵌套一個普通的ListView,這樣就可以解決水平和豎直滑動沖突的問題了。

<com.example.lzy.customview.HorizontalScrollViewEx 
 android:layout_width="match_parent" 
 android:layout_height="200dp"> 
 
 <ListView 
 android:id="@+id/listView" 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" /> 
 
 <Button 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" 
 android:background="@android:color/holo_blue_bright" 
 android:text="2" /> 
 
 <Button 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" 
 android:background="@android:color/holo_green_dark" 
 android:text="3" /> 
 </com.example.lzy.customview.HorizontalScrollViewEx> 

其他的部分代碼如果需要可以下載源碼來看

內部攔截法

內部攔截法是父容器不攔截任何事件,所有事件都傳遞給子view,如果需要就直接消耗掉,不需要再傳給父容器處理

下面重寫一個ListView,只需要重寫一個dispatchTouchEvent方法就OK

public class ListViewEx extends ListView { 
 
 private static final String TAG = "lzy"; 
 private int mLastX; 
 private int mLastY; 
 
 public ListViewEx(Context context) { 
 super(context); 
 } 
 
 public ListViewEx(Context context, AttributeSet attrs) { 
 super(context, attrs); 
 } 
 
 public ListViewEx(Context context, AttributeSet attrs, int defStyleAttr) { 
 super(context, attrs, defStyleAttr); 
 } 
 
 
 @Override 
 public boolean dispatchTouchEvent(MotionEvent ev) { 
 int x = (int) ev.getX(); 
 int y = (int) ev.getY(); 
 
 switch (ev.getAction()) { 
 case MotionEvent.ACTION_DOWN: 
 //子View的所有父ViewGroup都會跳過onInterceptTouchEvent的回調 
 getParent().requestDisallowInterceptTouchEvent(true); 
 break; 
 case MotionEvent.ACTION_MOVE: 
 int deltaX = x - mLastX; 
 int deltaY = y - mLastY; 
 if (Math.abs(deltaX) > Math.abs(deltaY) + 5) {//水平滑動,使得父類可以執行onInterceptTouchEvent 
  getParent().requestDisallowInterceptTouchEvent(false); 
 } 
 break; 
 } 
 mLastX = x; 
 mLastY = y; 
 return super.dispatchTouchEvent(ev); 
 } 
} 

在down事件調用getParent().requestDisallowInterceptTouchEvent(true),這句代碼的意思是使這個view的父容器都會跳過onInterceptTouchEvent,在move中判斷如果是水平滑動就由父容器去處理,父容器只需要把之前的onInterceptTouchEvent改為下面那樣,其他不變。

@Override 
 public boolean onInterceptTouchEvent(MotionEvent ev) { 
 int x = (int) ev.getX(); 
 int y = (int) ev.getY(); 
 if (ev.getAction() == MotionEvent.ACTION_DOWN) { 
 mLastX = x; 
 mLastY = y; 
 if (!mScroller.isFinished()) { 
 mScroller.abortAnimation(); 
 return true; 
 } 
 return false; 
 } else { 
 //如果是非down事件,說明子View并沒有攔截父類的onInterceptTouchEvent 
 //說明該事件交由父類處理,所以不需要再傳遞給子類,返回true 
 return true; 
 } 
 } 

關于Android應用的中滑動事件出現沖突如何解決問題的解答就分享到這里了,希望以上內容可以對大家有一定的幫助,如果你還有很多疑惑沒有解開,可以關注億速云行業資訊頻道了解更多相關知識。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

西昌市| 泽州县| 安福县| 彰化县| 额尔古纳市| 万州区| 东平县| 安康市| 隆尧县| 三门峡市| 莆田市| 拉萨市| 星座| 怀远县| 梁山县| 宾川县| 房产| 孝感市| 遂宁市| 海口市| 宕昌县| 赤水市| 海盐县| 巴林左旗| 玉龙| 甘谷县| 修水县| 大化| 莒南县| 军事| 来宾市| 景宁| 三门县| 牙克石市| 手机| 民县| 承德市| 芷江| 京山县| 舞阳县| 南岸区|