Android scrollview如何監(jiān)聽滑動狀態(tài)
ScrollView
視圖的滾動過程,其實是在不斷修改原點坐標。當手指觸摸后,ScrollView會暫時攔截觸摸事件,使用一個計時器。假如在計時器到點后沒有發(fā)生手指移動事件,那么ScrollView發(fā)送tracking events到被點擊的subView;若是在計時器到點后發(fā)生了移動事件,那么ScrollView取消tracking自己促發(fā)滾動。
首先說一下 NestedScrollView 的滑動事件的監(jiān)聽,
如果使用
nestedScrollView.setOnScrollChangeListener(new View.OnScrollChangeListener() { @Override public void onScrollChange(View v, int scrollX, int scrollY, int oldScrollX, int oldScrollY) { } });
這個方法在 API >= 23 時才可以使用,怎么解決呢 。我們可以自己定義一個ScrollView
public class MyScrollView extends NestedScrollView { private OnScrollChanged mOnScrollChanged; public MyScrollView(Context context) { this(context, null); } public MyScrollView(Context context, AttributeSet attributeSet) { this(context, attributeSet, 0); } public MyScrollView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } @Override protected void onScrollChanged(int l, int t, int oldl, int oldt) { super.onScrollChanged(l, t, oldl, oldt); if (mOnScrollChanged != null) { mOnScrollChanged.onScroll(l, t, oldl, oldt); } } public void setOnScrollChanged(OnScrollChanged onScrollChanged) { this.mOnScrollChanged = onScrollChanged; } public interface OnScrollChanged { void onScroll(int l, int t, int oldl, int oldt); } }
這樣我們就可以通過實現(xiàn) onScrollChanged() 監(jiān)聽滑動事件了 ,其中可以監(jiān)測到滑動距離,這樣就可以做好多事情了;
但是現(xiàn)在有一個需求就是【滑動的時候隱藏 一個靠邊的懸浮框,不滑動是懸浮框顯示出來】,這樣的話就需要監(jiān)測滑動狀態(tài)了。scrollview 不像recyclerview一樣可以監(jiān)測滑動狀態(tài)。
以下是我的一個實現(xiàn)方案,通過CountDownTimer 來實現(xiàn)
在剛才的onScrollChanged 接口中增加方法
public interface OnScrollChanged { void onScroll(int l, int t, int oldl, int oldt); void onTouch(boolean isDown); }
然后重寫onTouchEvent方法
@Override public boolean onTouchEvent(MotionEvent ev) { switch (ev.getAction()) { case MotionEvent.ACTION_UP: case MotionEvent.ACTION_CANCEL: if (mOnScrollChanged != null) { mOnScrollChanged.onTouch(false); } break; case MotionEvent.ACTION_DOWN: case MotionEvent.ACTION_MOVE: if (mOnScrollChanged != null) { mOnScrollChanged.onTouch(true); } break; } return super.onTouchEvent(ev); }
這里的isDown=true代表是按下或者滑動的狀態(tài),對應(yīng)ACTION_DOWN和ACTION_MOVE,fale代表ACTION_UP和ACTION_CANCEL
下面使用這個自定義的scrollerview
//靜止狀態(tài) private final static int SCROLL_STATE_IDLE = 1; //拖動或者慣性滑動狀態(tài) private final static int SCROLL_STATE_SCROLL = 2; //判斷是否是拖動狀態(tài) boolean isDragState = false; int currentState = SCROLL_STATE_IDLE; //這里采用100ms來判斷是否已經(jīng)是靜止狀態(tài),100ms結(jié)束后證明是靜止狀態(tài) private CountDownTimer scrollCountTimer = new CountDownTimer(100, 1) { @Override public void onTick(long millisUntilFinished) { } @Override public void onFinish() { setScrollState(SCROLL_STATE_IDLE); } }; private void initScrollView() { scrollView.setOnScrollChanged(new MyScrollView.OnScrollChanged() { @Override public void onScroll(int l, int t, int oldl, int oldt) { if (isDragState) {//拖動狀態(tài)單獨處理不再進行滾動狀態(tài)監(jiān)測 return; } //滑動時先取消倒計時,設(shè)置滑動狀態(tài) scrollCountTimer.cancel(); if(currentState != SCROLL_STATE_SCROLL) { setScrollState(SCROLL_STATE_SCROLL); } scrollCountTimer.start(); } @Override public void onTouch(boolean isDown) { isDragState = isDown; //我這里把按下的狀態(tài)默認為了滾動的狀態(tài),當然你也可以分開定義 if (isDown) { scrollCountTimer.cancel(); setScrollState(SCROLL_STATE_SCROLL); } else { scrollCountTimer.start(); } } }); //最后記得頁面銷毀時,cancel掉timer
總結(jié)
以上所述是小編給大家介紹的Android scrollview如何監(jiān)聽滑動狀態(tài),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
如果你覺得本文對你有幫助,歡迎轉(zhuǎn)載,煩請注明出處,謝謝!
相關(guān)文章
Android RecyclerView四級緩存源碼層詳細分析
RecyclerView是Android一個更強大的控件,其不僅可以實現(xiàn)和ListView同樣的效果,還有優(yōu)化了ListView中的各種不足。其可以實現(xiàn)數(shù)據(jù)縱向滾動,也可以實現(xiàn)橫向滾動(ListView做不到橫向滾動)。接下來講解RecyclerView的用法2022-11-11Android使用RotateImageView 旋轉(zhuǎn)ImageView
這篇文章主要介紹了Android使用RotateImageView 旋轉(zhuǎn)ImageView 的相關(guān)資料,需要的朋友可以參考下2016-01-01Android防止點擊過快造成多次響應(yīng)事件的解決方法
btn點擊用戶可能只點擊了一次但是后臺響應(yīng)了多次,像一些表單的提交出現(xiàn)這種問題比較棘手,本篇文章主要介紹Android防止點擊過快造成多次響應(yīng)事件的解決方法,有興趣的可以了解一下。2016-12-12Android實現(xiàn)MVVM架構(gòu)數(shù)據(jù)刷新詳解流程
MVVM架構(gòu)模式,即Model-View-ViewModel三個層級,MVVM模式出來的時間已經(jīng)很長了,網(wǎng)上關(guān)于MVVM模式的解析也有很多,我這里只說一下我自己的理解,基本上是和MVP模式相比較的一個差異2021-10-10