亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

Android如何優雅的處理重復點擊

瀏覽:4日期:2022-09-20 13:58:53

一般手機上的 Android App,主要的交互方式是點擊。用戶在點擊后,App 可能做出在頁面內更新 UI、新開一個頁面或者發起網絡請求等操作。Android 系統本身沒有對重復點擊做處理,如果用戶在短時間內多次點擊,則可能出現新開多個頁面或者重復發起網絡請求等問題。因此,需要對重復點擊有影響的地方,增加處理重復點擊的代碼。

之前的處理方式

之前在項目中使用的是 RxJava 的方案,利用第三方庫 RxBinding 實現了防止重復點擊:

fun View.onSingleClick(interval: Long = 1000L, listener: (View) -> Unit) { RxView.clicks(this) .throttleFirst(interval, TimeUnit.MILLISECONDS) .subscribe({ listener.invoke(this) }, { LogUtil.printStackTrace(it) })}

但是這樣有一個問題,比如使用兩個手指同時點擊兩個不同的按鈕,按鈕的功能都是新開頁面,那么有可能會新開兩個頁面。因為 Rxjava 這種方式是針對單個控件實現防止重復點擊,不是多個控件。

現在的處理方式

現在使用的是時間判斷,在時間范圍內只響應一次點擊,通過將上次單擊時間保存到 Activity Window 中的 decorView 里,實現一個 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}

參數 isShareSingleClick 的默認值為 true,表示該控件和同一個 Activity 中其他控件共用一個上次單擊時間,也可以手動改成 false,表示該控件自己獨享一個上次單擊時間。

mBinding.btn1.onSingleClick { // 處理單次點擊}mBinding.btn2.onSingleClick(interval = 2000, isShareSingleClick = false) { // 處理單次點擊}其他場景處理重復點擊間接設置點擊

除了直接在 View 上設置的點擊監聽外,其他間接設置點擊的地方也存在需要處理重復點擊的場景,比如說富文本和列表。

為此將判斷是否觸發單次點擊的代碼抽離出來,單獨作為一個方法:

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) { ...}

直接在點擊監聽回調中調用 determineTriggerSingleClick 判斷是否觸發單次點擊。下面拿富文本和列表舉例。

富文本

繼承 ClickableSpan,在 onClick 回調中判斷是否觸發單次點擊:

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 回調中的 widget,就是設置富文本的控件,也就是說如果富文本存在多個單次點擊的地方, 就算 isShareSingleClick 值為 false,這些單次點擊還是會共用設置富文本控件的上次單擊時間。

因此,這里需要特殊處理,在 isShareSingleClick 為 false 的時候,創建一個假的 View 來觸發單擊事件,這樣富文本中多個單次點擊 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) } ...}

在設置富文本的地方,使用設置 onSingleClick 實現單次點擊:

mBinding.tvText.movementMethod = LinkMovementMethod.getInstance()mBinding.tvText.highlightColor = Color.TRANSPARENTmBinding.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 { // 處理單次點擊 } }}數據綁定

使用 DataBinding 的時候,有時會在布局文件中直接設置點擊事件,于是在 View.onSingleClick 上增加 @BindingAdapte 注解,實現在布局文件中設置單次點擊事件,并對代碼做出調整,這個時候需要將項目中 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 ) }}

在布局文件中設置單次點擊:

<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() { // 處理單次點擊 }}總結

對于直接在 View 上設置點擊的地方,如果需要處理重復點擊使用 onSingleClick,不需要處理重復點擊則使用原來的 setOnClickListener。

對于間接設置點擊的地方,如果需要處理重復點擊,則使用 determineTriggerSingleClick 判斷是否觸發單次點擊。

項目地址

single-click,覺得用起來很爽的,請不要吝嗇你的 Star !

以上就是Android如何優雅的處理重復點擊的詳細內容,更多關于Android 處理重復點擊的資料請關注好吧啦網其它相關文章!

標簽: Android
相關文章:
主站蜘蛛池模板: 尤物在线播放 | 久久综合网久久综合 | 久久这里只有精品2 | 日韩中文视频 | 一区二区三区毛片免费 | 免费摸碰碰视频在线观看 | 最刺激黄a大片免费观看 | 国产精品99精品久久免费 | 国产免费小视频 | 久久国产区 | 中文字幕播放 | 网友自拍区一区二区三区 | 中文精品久久久久国产网址 | 成人影院在线观看kkk4444 | 国内精品一区二区在线观看 | 成人高清毛片a | 国产丝袜第一页 | 国内自拍视频在线看免费观看 | 国产一区二区影院 | 在线欧美色 | 国产女人毛片 | 伊人久久综合视频 | 青青青爽国产在线视频 | 国产1区在线观看 | 免费欧洲毛片a级视频老妇女 | 久久久久一区二区三区 | 成人污 | 久久爰www免费人成 久久狠狠干 | 国产精品成人免费视频99 | 色一级| 国产精品亚洲精品日韩已满 | 黄色一级一级 | 51国产午夜精品免费视频 | 色网站在线免费观看 | 国产国语特级一级aa毛片 | 日本96在线精品视频免费观看 | 亚洲伦理一区二区三区 | 亚洲国产精品看片在线观看 | 99久久99久久精品免费看蜜桃 | 日本黄色一区 | 99视频在线观看视频一区 |