Android如何優(yōu)雅的處理重復(fù)點擊
一般手機上的 Android App,主要的交互方式是點擊。用戶在點擊后,App 可能做出在頁面內(nèi)更新 UI、新開一個頁面或者發(fā)起網(wǎng)絡(luò)請求等操作。Android 系統(tǒng)本身沒有對重復(fù)點擊做處理,如果用戶在短時間內(nèi)多次點擊,則可能出現(xiàn)新開多個頁面或者重復(fù)發(fā)起網(wǎng)絡(luò)請求等問題。因此,需要對重復(fù)點擊有影響的地方,增加處理重復(fù)點擊的代碼。
之前的處理方式
之前在項目中使用的是 RxJava 的方案,利用第三方庫 RxBinding 實現(xiàn)了防止重復(fù)點擊:
fun View.onSingleClick(interval: Long = 1000L, listener: (View) -> Unit) { RxView.clicks(this) .throttleFirst(interval, TimeUnit.MILLISECONDS) .subscribe({ listener.invoke(this) }, { LogUtil.printStackTrace(it) }) }
但是這樣有一個問題,比如使用兩個手指同時點擊兩個不同的按鈕,按鈕的功能都是新開頁面,那么有可能會新開兩個頁面。因為 Rxjava 這種方式是針對單個控件實現(xiàn)防止重復(fù)點擊,不是多個控件。
現(xiàn)在的處理方式
現(xiàn)在使用的是時間判斷,在時間范圍內(nèi)只響應(yīng)一次點擊,通過將上次單擊時間保存到 Activity Window 中的 decorView 里,實現(xiàn)一個 Activity 中所有的 View 共用一個上次單擊時間。
fun View.onSingleClick( interval: Int = SingleClickUtil.singleClickInterval, isShareSingleClick: Boolean = true, listener: (View) -> Unit ) { setOnClickListener { val target = if (isShareSingleClick) getActivity(this)?.window?.decorView ?: this else this val millis = target.getTag(R.id.single_click_tag_last_single_click_millis) as? Long ?: 0 if (SystemClock.uptimeMillis() - millis >= interval) { target.setTag( R.id.single_click_tag_last_single_click_millis, SystemClock.uptimeMillis() ) listener.invoke(this) } } } private fun getActivity(view: View): Activity? { var context = view.context while (context is ContextWrapper) { if (context is Activity) { return context } context = context.baseContext } return null }
參數(shù) isShareSingleClick 的默認值為 true,表示該控件和同一個 Activity 中其他控件共用一個上次單擊時間,也可以手動改成 false,表示該控件自己獨享一個上次單擊時間。
mBinding.btn1.onSingleClick { // 處理單次點擊 } mBinding.btn2.onSingleClick(interval = 2000, isShareSingleClick = false) { // 處理單次點擊 }
其他場景處理重復(fù)點擊
間接設(shè)置點擊
除了直接在 View 上設(shè)置的點擊監(jiān)聽外,其他間接設(shè)置點擊的地方也存在需要處理重復(fù)點擊的場景,比如說富文本和列表。
為此將判斷是否觸發(fā)單次點擊的代碼抽離出來,單獨作為一個方法:
fun View.onSingleClick( interval: Int = SingleClickUtil.singleClickInterval, isShareSingleClick: Boolean = true, listener: (View) -> Unit ) { setOnClickListener { determineTriggerSingleClick(interval, isShareSingleClick, listener) } } fun View.determineTriggerSingleClick( interval: Int = SingleClickUtil.singleClickInterval, isShareSingleClick: Boolean = true, listener: (View) -> Unit ) { ... }
直接在點擊監(jiān)聽回調(diào)中調(diào)用 determineTriggerSingleClick 判斷是否觸發(fā)單次點擊。下面拿富文本和列表舉例。
富文本
繼承 ClickableSpan,在 onClick 回調(diào)中判斷是否觸發(fā)單次點擊:
inline fun SpannableStringBuilder.onSingleClick( listener: (View) -> Unit, isShareSingleClick: Boolean = true, ... ): SpannableStringBuilder = inSpans( object : ClickableSpan() { override fun onClick(widget: View) { widget.determineTriggerSingleClick(interval, isShareSingleClick, listener) } ... }, builderAction = builderAction )
這樣會有一個問題, onClick 回調(diào)中的 widget,就是設(shè)置富文本的控件,也就是說如果富文本存在多個單次點擊的地方, 就算 isShareSingleClick 值為 false,這些單次點擊還是會共用設(shè)置富文本控件的上次單擊時間。
因此,這里需要特殊處理,在 isShareSingleClick 為 false 的時候,創(chuàng)建一個假的 View 來觸發(fā)單擊事件,這樣富文本中多個單次點擊 isShareSingleClick 為 false 的地方都有一個自己的假的 View 來獨享上次單擊時間。
class SingleClickableSpan( ... ) : ClickableSpan() { private var mFakeView: View? = null override fun onClick(widget: View) { if (isShareSingleClick) { widget } else { if (mFakeView == null) { mFakeView = View(widget.context) } mFakeView!! }.determineTriggerSingleClick(interval, isShareSingleClick, listener) } ... }
在設(shè)置富文本的地方,使用設(shè)置 onSingleClick 實現(xiàn)單次點擊:
mBinding.tvText.movementMethod = LinkMovementMethod.getInstance() mBinding.tvText.highlightColor = Color.TRANSPARENT mBinding.tvText.text = buildSpannedString { append("normalText") onSingleClick({ // 處理單次點擊 }) { color(Color.GREEN) { append("clickText") } } }
列表
列表使用 RecyclerView 控件,適配器使用第三方庫 BaseRecyclerViewAdapterHelper。
Item 點擊:
adapter.setOnItemClickListener { _, view, _ -> view.determineTriggerSingleClick { // 處理單次點擊 } }
Item Child 點擊:
adapter.addChildClickViewIds(R.id.btn1, R.id.btn2) adapter.setOnItemChildClickListener { _, view, _ -> when (view.id) { R.id.btn1 -> { // 處理普通點擊 } R.id.btn2 -> view.determineTriggerSingleClick { // 處理單次點擊 } } }
數(shù)據(jù)綁定
使用 DataBinding 的時候,有時會在布局文件中直接設(shè)置點擊事件,于是在 View.onSingleClick 上增加 @BindingAdapte 注解,實現(xiàn)在布局文件中設(shè)置單次點擊事件,并對代碼做出調(diào)整,這個時候需要將項目中 listener: (View) -> Unit 替換成 listener: View.OnClickListener。
@BindingAdapter( *["singleClickInterval", "isShareSingleClick", "onSingleClick"], requireAll = false ) fun View.onSingleClick( interval: Int? = SingleClickUtil.singleClickInterval, isShareSingleClick: Boolean? = true, listener: View.OnClickListener? = null ) { if (listener == null) { return } setOnClickListener { determineTriggerSingleClick( interval ?: SingleClickUtil.singleClickInterval, isShareSingleClick ?: true, listener ) } }
在布局文件中設(shè)置單次點擊:
<androidx.appcompat.widget.AppCompatButton android:layout_width="match_parent" android:layout_height="wrap_content" android:text="@string/btn" app:isShareSingleClick="@{false}" app:onSingleClick="@{()->viewModel.handleClick()}" app:singleClickInterval="@{2000}" />
在代碼中處理單次點擊:
class YourViewModel : ViewModel() { fun handleClick() { // 處理單次點擊 } }
總結(jié)
對于直接在 View 上設(shè)置點擊的地方,如果需要處理重復(fù)點擊使用 onSingleClick,不需要處理重復(fù)點擊則使用原來的 setOnClickListener。
對于間接設(shè)置點擊的地方,如果需要處理重復(fù)點擊,則使用 determineTriggerSingleClick 判斷是否觸發(fā)單次點擊。
項目地址
single-click,覺得用起來很爽的,請不要吝嗇你的 Star !
以上就是Android如何優(yōu)雅的處理重復(fù)點擊的詳細內(nèi)容,更多關(guān)于Android 處理重復(fù)點擊的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
另外兩種Android沉浸式狀態(tài)欄實現(xiàn)思路
這篇文章主要為大家介紹了另外兩種Android沉浸式狀態(tài)欄實現(xiàn)思路,android5.0及以后版本都支持給狀態(tài)欄著色,而目前android主流版本還是4.4,想要深入了解的朋友可以參考一下2016-01-01詳解Android studio 3+版本apk安裝失敗問題
這篇文章主要介紹了詳解Android studio 3+版本apk安裝失敗問題,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04Android 和 windows C/C++/QT通訊時字節(jié)存儲
本篇文章主要介紹 Android和Windows 通訊時數(shù)據(jù)地址的理解,這里提供代碼實例進行分析,有需要參考的朋友可以看下2016-07-07Android編程之ListView和EditText發(fā)布帖子隱藏軟鍵盤功能詳解
這篇文章主要介紹了Android編程之ListView和EditText發(fā)布帖子隱藏軟鍵盤功能,結(jié)合實例形式分析了Android控件調(diào)用、隱藏軟鍵盤的原理與具體實現(xiàn)技巧,需要的朋友可以參考下2017-08-08Android HTTP發(fā)送請求和接收響應(yīng)的實例代碼
Android HTTP請求和接收響應(yīng)實例完整的Manifest文件如下,感興趣的朋友可以參考下哈,希望對大家有所幫助2013-06-06android系統(tǒng)在靜音模式下關(guān)閉camera拍照聲音的方法
本文為大家詳細介紹下android系統(tǒng)如何在靜音模式下關(guān)閉camera拍照聲音,具體的實現(xiàn)方法如下,感興趣的朋友可以參考下哈2013-07-07