Android View 事件防抖的兩種方案
兩種方案
- 侵入式防抖處理(NoShakeClickListener) implements View.OnClickListener
- 無(wú)侵入式防抖處理(NoShakeClickListener2) not implements View.OnClickListener
侵入式防抖處理(NoShakeClickListener)
1. 既適用于單個(gè)View事件防抖, 也適用于Adapter中ItemView事件防抖
2. 如果事件為跳轉(zhuǎn)到新的Activity, 該Activity啟動(dòng)模型應(yīng)為android:launchMode="singleTop"
Java 版本
public abstract class NoShakeClickListener implements View.OnClickListener {
private long mTimeInterval = 500L;
/**
* 最近一次點(diǎn)擊的時(shí)間
*/
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 時(shí)間間期(毫秒)
* @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);
}
可簡(jiǎn)寫(xiě)為
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)擊的時(shí)間
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 時(shí)間間期(毫秒)
* @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()
}
})
無(wú)侵入式防抖處理(NoShakeClickListener2)
特點(diǎn):
1 移除了對(duì)View.OnClickListener的依賴(lài), 在不破壞原有代碼已設(shè)置好的OnClickListener基礎(chǔ)上繼續(xù)處理事件防抖;
2 支持簡(jiǎn)單處理事件和泛型回調(diào)
/**
* 事件防抖
* 注: 不僅適用于 View , 其他控件如: MenuItem 同樣適用
*
* 1.既適用于單個(gè)`View`事件防抖, 也適用于`Adapter`中`ItemView`事件防抖
* 2.如果事件為跳轉(zhuǎn)到新的`Activity`, 該`Activity`啟動(dòng)模型應(yīng)為`android:launchMode="singleTop"`
*/
open class NoShakeClickListener2 @JvmOverloads constructor(interval: Long = 500L) {
private var mTimeInterval = 500L
private var mLastClickTime: Long = 0 //最近一次點(diǎn)擊的時(shí)間
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 時(shí)間間期(毫秒)
* @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 簡(jiǎn)單使用
//快速點(diǎn)擊事件
val fastClick=object :NoShakeClickListener2(){
override fun onFastClick(item: Any?) {
super.onFastClick(item)
//此時(shí) 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)
//此時(shí) 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 事件防抖的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- Android WebView如何判定網(wǎng)頁(yè)加載的錯(cuò)誤
- Android實(shí)現(xiàn)View滑動(dòng)效果的6種方法
- Android控件View的文字周?chē)砑訄D標(biāo)
- 自己實(shí)現(xiàn)Android View布局流程
- 如何自己實(shí)現(xiàn)Android View Touch事件分發(fā)流程
- Android webView字體突然變小的原因及解決
- Android Filterable實(shí)現(xiàn)Recyclerview篩選功能的示例代碼
- Android 滑動(dòng)Scrollview標(biāo)題欄漸變效果(仿京東toolbar)
- Android自定義View用切圖顯示字符串
相關(guān)文章
Android開(kāi)發(fā)之基于RecycleView實(shí)現(xiàn)的頭部懸浮控件
RecyclerView是一種類(lèi)似于ListView的一個(gè)滑動(dòng)列表,但是RecyclerView和ListView相比,RecyclerView比ListView更好,這篇文章重點(diǎn)給大家介紹基于RecycleView實(shí)現(xiàn)的頭部懸浮控件,感興趣的朋友一起看看吧2019-10-10
Android BottomSheet實(shí)現(xiàn)可拉伸控件
這篇文章主要為大家詳細(xì)介紹了Android BottomSheet實(shí)現(xiàn)可拉伸控件,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-11-11
丟失Android系統(tǒng)庫(kù)或者Conversion to Dalvik format failed with error
這篇文章主要介紹了丟失Android系統(tǒng)庫(kù)或者Conversion to Dalvik format failed with error 1錯(cuò)誤的解決方法,分析了Android系統(tǒng)庫(kù)丟失及版本問(wèn)題的處理技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-12-12
Android實(shí)現(xiàn)觸摸移動(dòng)的懸浮窗口功能
這篇文章主要介紹了Android實(shí)現(xiàn)觸摸移動(dòng)的懸浮窗口功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-09-09
flutter 自定義websocket路由的實(shí)現(xiàn)
這篇文章主要介紹了flutter 自定義websocket路由的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12
Android實(shí)現(xiàn)文字滾動(dòng)播放效果的代碼
這篇文章主要介紹了Android實(shí)現(xiàn)文字滾動(dòng)播放效果,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07
Android開(kāi)發(fā)實(shí)現(xiàn)的獲取sdcard大小及內(nèi)存大小工具類(lèi)
這篇文章主要介紹了Android開(kāi)發(fā)實(shí)現(xiàn)的獲取sdcard大小及內(nèi)存大小工具類(lèi),涉及Android針對(duì)手機(jī)硬件SD卡及內(nèi)存相關(guān)操作技巧,需要的朋友可以參考下2017-11-11
利用SpannableString和ImageSpan在textview中插入圖片的方法
這篇文章主要為大家詳細(xì)介紹了利用SpannableString和ImageSpan在textview中插入圖片的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11

