Android View 事件防抖的兩種方案
兩種方案
- 侵入式防抖處理(NoShakeClickListener) implements View.OnClickListener
- 無侵入式防抖處理(NoShakeClickListener2) not implements View.OnClickListener
侵入式防抖處理(NoShakeClickListener)
1. 既適用于單個View事件防抖, 也適用于Adapter中ItemView事件防抖
2. 如果事件為跳轉(zhuǎn)到新的Activity, 該Activity啟動模型應(yīng)為android:launchMode="singleTop"
Java 版本
public abstract class NoShakeClickListener implements View.OnClickListener { private long mTimeInterval = 500L; /** * 最近一次點(diǎn)擊的時間 */ private long mLastClickTime; /** * 最近一次點(diǎn)擊的控件ID */ private int mLastClickViewId; public NoShakeClickListener() { } public NoShakeClickListener(long interval) { this.mTimeInterval = interval; } @Override public void onClick(View v) { final boolean isFastClick = isFastDoubleClick(v, this.mTimeInterval); if (isFastClick) { onFastClick(v); } else { onSingleClick(v); } } /** * 是否是快速點(diǎn)擊 * * @param v 點(diǎn)擊的控件 * @param interval 時間間期(毫秒) * @return true:是,false:不是 */ private boolean isFastDoubleClick(View v, long interval) { int viewId = v.getId(); long nowTime = System.currentTimeMillis(); long timeInterval = Math.abs(nowTime - mLastClickTime); if (timeInterval < interval && viewId == mLastClickViewId) { // 快速點(diǎn)擊事件 return true; } else { // 單次點(diǎn)擊事件 mLastClickTime = nowTime; mLastClickViewId = viewId; return false; } } protected void onFastClick(View v) {} protected abstract void onSingleClick(View v); }
可簡寫為
public abstract class NoShakeListener implements OnClickListener { private long mLastClickTime = 0; private boolean isFastDoubleClick() { long nowTime = System.currentTimeMillis(); if (Math.abs(nowTime - mLastClickTime) < 500) { return true; // 快速點(diǎn)擊事件 } else { mLastClickTime = nowTime; return false; // 單次點(diǎn)擊事件 } } @Override public void onClick(View v) { if (isFastDoubleClick()) { onFastClick(v); } else { onSingleClick(v); } } protected void onFastClick(View v) { } protected abstract void onSingleClick(View v); }
Kotlin版本
abstract class NoShakeClickListener @JvmOverloads constructor(interval: Long = 500L) : View.OnClickListener { private var mTimeInterval = 500L private var mLastClickTime: Long = 0 //最近一次點(diǎn)擊的時間 private var mLastClickViewId = 0 //最近一次點(diǎn)擊的控件ID init { mTimeInterval = interval } override fun onClick(v: View) { if (isFastDoubleClick(v, mTimeInterval)) onFastClick(v) else onSingleClick(v) } /** * 是否是快速點(diǎn)擊 * * @param v 點(diǎn)擊的控件 * @param interval 時間間期(毫秒) * @return true:是,false:不是 */ private fun isFastDoubleClick(v: View, interval: Long): Boolean { val viewId = v.id val nowTime = System.currentTimeMillis() val timeInterval = abs(nowTime - mLastClickTime) return if (timeInterval < interval && viewId == mLastClickViewId) { // 快速點(diǎn)擊事件 true } else { // 單次點(diǎn)擊事件 mLastClickTime = nowTime mLastClickViewId = viewId false } } protected open fun onFastClick(v: View?) {} protected abstract fun onSingleClick(v: View?) }
差點(diǎn)忘了... 相應(yīng)的擴(kuò)展函數(shù):
fun View?.noShake(block: (v: View?) -> Unit) { this?.apply { setOnClickListener(object : NoShakeClickListener() { override fun onSingleClick(v: View?) { block.invoke(v) } }) } }
RxJava2 clickExt.kt
inline fun <T : View> T.noShake(crossinline listener: T.() -> Unit) = this.noShake(1000, listener) inline fun <T : View> T.noShake(windowDuration: Long = 500, crossinline listener : T.() -> Unit) = RxView.clicks(this) .throttleFirst(windowDuration, TimeUnit.MILLISECONDS) .subscribe(object : Observer<Any> { override fun onSubscribe(d: Disposable) {} override fun onError(e: Throwable) {} override fun onComplete() {} override fun onNext(o: Any) { listener() } })
無侵入式防抖處理(NoShakeClickListener2)
特點(diǎn):
1 移除了對View.OnClickListener的依賴, 在不破壞原有代碼已設(shè)置好的OnClickListener基礎(chǔ)上繼續(xù)處理事件防抖;
2 支持簡單處理事件和泛型回調(diào)
/** * 事件防抖 * 注: 不僅適用于 View , 其他控件如: MenuItem 同樣適用 * * 1.既適用于單個`View`事件防抖, 也適用于`Adapter`中`ItemView`事件防抖 * 2.如果事件為跳轉(zhuǎn)到新的`Activity`, 該`Activity`啟動模型應(yīng)為`android:launchMode="singleTop"` */ open class NoShakeClickListener2 @JvmOverloads constructor(interval: Long = 500L) { private var mTimeInterval = 500L private var mLastClickTime: Long = 0 //最近一次點(diǎn)擊的時間 private var mLastClick: Any? = null //最近一次點(diǎn)擊的控件 View or MenuItem ... init { mTimeInterval = interval } fun proceedClick() { if (isFastClick(null, mTimeInterval)) onFastClick(null) else onSingleClick(null) } fun <T> proceedClick(item: T?) { if (isFastClick(item, mTimeInterval)) onFastClick(item) else onSingleClick(item) } /** * 是否是快速點(diǎn)擊 * * @param item 點(diǎn)擊的控件 View or MenuItem ... * @param interval 時間間期(毫秒) * @return true:是,false:不是 */ private fun <T> isFastClick(item: T?, interval: Long): Boolean { val nowTime = System.currentTimeMillis() val timeInterval = abs(nowTime - mLastClickTime) return if (timeInterval < interval && item == mLastClick) { // 快速點(diǎn)擊事件 true } else { // 單次點(diǎn)擊事件 mLastClickTime = nowTime mLastClick = item false } } protected open fun onFastClick(item: Any?) {} protected open fun onSingleClick(item: Any?) {} }
實(shí)用案例
1 簡單使用
//快速點(diǎn)擊事件 val fastClick=object :NoShakeClickListener2(){ override fun onFastClick(item: Any?) { super.onFastClick(item) //此時 item == null Log.e("123", "onFastClick Click") } } BottomNavigationView.setOnNavigationItemSelectedListener { switchPage(it.itemId) fastClick.proceedClick() true }
2 回傳參數(shù)
//快速點(diǎn)擊事件 val fastClick=object :NoShakeClickListener2(){ override fun onFastClick(item: Any?) { super.onFastClick(item) //此時 item == null 為 proceedClick(it) 中的 it Log.e("123", "onFastClick Click") } } BottomNavigationView.setOnNavigationItemSelectedListener { switchPage(it.itemId) fastClick.proceedClick(it) true }
以上就是Android View 事件防抖的兩種方案的詳細(xì)內(nèi)容,更多關(guān)于Android View 事件防抖的資料請關(guān)注腳本之家其它相關(guān)文章!
- Android WebView如何判定網(wǎng)頁加載的錯誤
- Android實(shí)現(xiàn)View滑動效果的6種方法
- Android控件View的文字周圍添加圖標(biāo)
- 自己實(shí)現(xiàn)Android View布局流程
- 如何自己實(shí)現(xiàn)Android View Touch事件分發(fā)流程
- Android webView字體突然變小的原因及解決
- Android Filterable實(shí)現(xiàn)Recyclerview篩選功能的示例代碼
- Android 滑動Scrollview標(biāo)題欄漸變效果(仿京東toolbar)
- Android自定義View用切圖顯示字符串
相關(guān)文章
Android開發(fā)之基于RecycleView實(shí)現(xiàn)的頭部懸浮控件
RecyclerView是一種類似于ListView的一個滑動列表,但是RecyclerView和ListView相比,RecyclerView比ListView更好,這篇文章重點(diǎn)給大家介紹基于RecycleView實(shí)現(xiàn)的頭部懸浮控件,感興趣的朋友一起看看吧2019-10-10Android BottomSheet實(shí)現(xiàn)可拉伸控件
這篇文章主要為大家詳細(xì)介紹了Android BottomSheet實(shí)現(xiàn)可拉伸控件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2019-11-11丟失Android系統(tǒng)庫或者Conversion to Dalvik format failed with error
這篇文章主要介紹了丟失Android系統(tǒng)庫或者Conversion to Dalvik format failed with error 1錯誤的解決方法,分析了Android系統(tǒng)庫丟失及版本問題的處理技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-12-12Android實(shí)現(xiàn)觸摸移動的懸浮窗口功能
這篇文章主要介紹了Android實(shí)現(xiàn)觸摸移動的懸浮窗口功能,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-09-09flutter 自定義websocket路由的實(shí)現(xiàn)
這篇文章主要介紹了flutter 自定義websocket路由的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12Android實(shí)現(xiàn)文字滾動播放效果的代碼
這篇文章主要介紹了Android實(shí)現(xiàn)文字滾動播放效果,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-07-07Android開發(fā)實(shí)現(xiàn)的獲取sdcard大小及內(nèi)存大小工具類
這篇文章主要介紹了Android開發(fā)實(shí)現(xiàn)的獲取sdcard大小及內(nèi)存大小工具類,涉及Android針對手機(jī)硬件SD卡及內(nèi)存相關(guān)操作技巧,需要的朋友可以參考下2017-11-11利用SpannableString和ImageSpan在textview中插入圖片的方法
這篇文章主要為大家詳細(xì)介紹了利用SpannableString和ImageSpan在textview中插入圖片的方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-11-11