现象描述:
快速多次点击一个View,实际行为可能和期望的不一样,或多次打开同一个页面(Activity or Fragment),或多次 Toast等等,虽然我们可以 有其他途径去补救(如手动管理Fragment、设置 Activity 为 singleTask、每次 Toast 时把前一个取消),但是我们希望可以从根上解决这个问题。
解决方法
方法一:古老的点击计时法:
记录每次点击时的时间,如果两次点击的时间间隔小于某个数值,就认为当前发生的双击,这个时候就需要手动过滤掉第二次事件。上代码:
public void onClickSth(){
if (doubleClick()) {
return;
} else {
//do someThing
}
}
public boolean doubleClick(){
if (System.currentTimeMillis() - lastTime < CLICK_THRESHOLD){
return true;
}
lastTime = System.currentTimeMillis();
return false;
}
方法二:RxView 来帮忙:
JakeWharton 的大作,可以响应式的方式解决 UI 的响应问题。
RxBinding 直通车,使用的话在 build.gradle 文件中添加如下依赖就好了。
dependencies {
......
compile 'com.jakewharton.rxbinding2:rxbinding:2.0.0'
compile 'com.jakewharton:butterknife:8.8.1'
annotationProcessor 'com.jakewharton:butterknife-compiler:8.8.1'
}
具体使用也挺简单的,不多说了
RxView.clicks(mView).throttleFirst(1,TimeUnit.SECONDS).subscribe(o -> {//do sth});
方法三:RxJava 大法好:
有同学说了,我已经添加了 RxJava 依赖库,但是不想使 RxBinding 的其余共功能,只想做一个防止点击的工具类,这可咋整?
上代码:
//使用举例
DoubleClickHelper.click(mView, v -> {// do sth};
源代码:
public class DoubleClickHelper {
private static final int windowDuration = 1;
public static void click(View view, Runnable r) {
click(view, r, windowDuration);
}
public static void click(View view, Runnable r, int durationSeconds) {
if (view == null || r == null) {
return;
}
Observable.create(new ClickObservable(view)).throttleFirst(durationSeconds, TimeUnit.SECONDS).subscribe(o -> r.run());
}
public static void click(View view, View.OnClickListener listener) {
click(view, listener, windowDuration);
}
public static void click(View view, View.OnClickListener listener, int durationSeconds) {
if (view == null || listener == null) {
return;
}
Observable.create(new ClickObservable(view)).throttleFirst(durationSeconds, TimeUnit.SECONDS).subscribe(o -> listener.onClick(view));
}
private DoubleClickHelper() {
}
}
public class ClickObservable implements ObservableOnSubscribe<View> {
private ObservableEmitter mEmitter;
public ClickObservable(View view) {
view.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
if (mEmitter != null) {
mEmitter.onNext(view);
}
}
});
}
@Override
public void subscribe(ObservableEmitter<View> e) throws Exception {
mEmitter = e;
}
}
使用起来是不是特别方便呢,一行代码就解决了。
方法四:databingding 如何实现防止两次点击呢?
首先看一下 databingding 的使用,
<View
android:id="@+id/id_xxx"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:onClick="@{viewModel.xxxxClicklistener}"/>
平时我们写点击事件是直接这么操作的,那在代码中 xxx.setOnclick(()->{}) 是在哪设置的呢?我们可以编译下,xxxBinding executeBindings() 方法中可以看到
this.xxxView1.setOnClickListener(xxxListener1);
this.xxxView2.setOnClickListener(xxxListener2);
this.xxxView3.setOnClickListener(xxxListener3);
既然这样,在 View 的 onClick 这做文章是不可能了,但是我们还是可以手动设置点击事件的,需要依赖 databinding 的注解 @BindingAdapter,
在一个工具类中类似这样,
@BindingAdapter({"bindOnClick"})
public static void bindOnClick(View view, View.OnClickListener listener) {
//可以用上面三种任意一种形式防止二次点击
DoubleClickHelper.click(view, listener);
}
调用的地方类似这样,viewModel.xxxlistener 为一个 View.OnClickListener 的一个实现类,里面写我们自己的点击事件逻辑。
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:bindOnClick="@{viewModel.xxxlistener}"
iknow:category="兄嘚,看到这了点个赞吧" />
好了,到这就大功告成啦,有问题欢迎交流。