前言
android 防止重复点击是一个非常常见的需求,每个人都有各自的点击事件的处理习惯,有的喜欢使用匿名内部类,有的activity、fragment、自定义View等继承点击事件然后在onClick()方法中根据id用switch实现各自View的点击事件。
在开发中我们经常需要这样的需求,比如一个验证码发送按钮,我们只想让它响应500毫秒中的第一次点击事件,该如何处理呢?你可能会说这个简单,在点击事件中获取当前时间与上次的比较下,如果小于500毫秒就return掉。是的,这样可以解决,但是如果现在整个项目的所有按钮点击事件都需要这样的需求,该如何处理?不可能内个点击事件中都加入这几行代码吧。
这里先放上我写的一个响应第一次点击的工具类,可实现2种模式:
- 第一种:无论点击的哪个View,仅响应第一次点击
- 第二章:同一个View上仅响应第一次点击,不同View间无影响
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
|
public class ClickHelper { private static long DELAY = 500L; private static long lastTime = 0L; private static List<Integer> viewIds = null ; private static final int SAME_VIEW_SIZE = 10 ; private ClickHelper() { } public static void setDelay( long delay) { ClickHelper.DELAY = delay; } public static long getDelay() { return DELAY; } public static void onlyFirstIgnoreView( final View target, @NonNull final Callback callback) { long nowTime = System.currentTimeMillis(); if (nowTime - lastTime > DELAY) { callback.onClick(target); lastTime = nowTime; } } public static void onlyFirstSameView( final View target, @NonNull final Callback callback) { long nowTime = System.currentTimeMillis(); int id = target.getId(); if (viewIds == null ) { viewIds = new ArrayList<>(SAME_VIEW_SIZE); } if (viewIds.contains(id)) { if (nowTime - lastTime > DELAY) { callback.onClick(target); lastTime = nowTime; } } else { if (viewIds.size() >= SAME_VIEW_SIZE) { viewIds.remove( 0 ); } viewIds.add(id); callback.onClick(target); lastTime = nowTime; } } public interface Callback { void onClick(View view); } } |
那如何才能让它对整个项目的所有点击事件生效呢?我的解决办法是这样的。
第一步,继承View.OnClickListener在onClick方法中调用工具类回调到抽象方法,项目中所有的点击事件都去继承这个抽象类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public abstract class OnFirstClickListener implements View.OnClickListener { @Override public final void onClick( final View v) { ClickHelper.onlyFirstSameView(v, new ClickHelper.Callback() { @Override public void onClick(View view) { onFirstClick(view); } }); } public abstract void onFirstClick(View v); } |
第二步,在项目的BaseActivity中实现View.OnClickListener接口,在onClick方法中调用工具类,回调出屏蔽后的点击事件,子类复写onClickWithoutLogin或者onClickCheckLogin方法就可以了。至于为什么会有2个,看名字就知道了,一个验证了用户登录状态,仅在登录状态响应事件,未登录则跳转登录界面,多封装了一层罢了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
|
/** * 用注解绑定点击事件时,在该方法绑定 */ @Override public void onClick( final View v) { ClickHelper.onlyFirstSameView(v, new ClickHelper.Callback() { @Override public void onClick(View view) { if (!onClickWithoutLogin(view)) { if (UserUtils.doIfLogin(getContext())) { onClickCheckLogin(view); } } } }); } /** * 不需要登录的点击事件 */ public boolean onClickWithoutLogin(View v) { return false ; } /** * 必须登录的点击事件 * 如果已经登录直接执行,没有登录时跳转登录界面 */ public void onClickCheckLogin(View v) { } |
以上只是个人开发中用到的,如果大家有更好的方法,欢迎留言讨论。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对服务器之家的支持。
原文链接:http://www.jianshu.com/p/f2610851b260