一个小问题

今天在公司项目中遇到一个问题,一个网络请求的按钮,点击以后我没有禁用,用户一直点击,APP频繁的请求同一个接口,而且无限弹窗。

解决历程

1.写一个按钮的分类或者子类,在- (void)sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event

这个方法重写点击事件,做一个时间的延时判断,缺点,你需要在所有的按钮创建的地方用这个类去替换原来的UIButton,项目改动太大.

2.还有一种方法,就是利用运行时,通过iOS黑魔法-Method Swizzling写按钮的祖宗(UIControl)的分类,通过方法交换,实现限制按钮被重复点击.下面是实现过程

在.h中声明一个属性,用来设置按钮延时的时间:

@property(nonatomic,assign)NSTimeIntervalclickDurationTime;

在.m文件中代码如下

#import"UIControl+extension.h"

static const NSTimeIntervaldefaultDuration =1.0f;// 记录是否忽略按钮点击事件,

默认第一次执行事件

static BOOL _isIgnoreEvent = NO;

staticvoidresetState()

{_isIgnoreEvent=NO;}

@implementationUIControl(extension)

@dynamicclickDurationTime;

+ (void)load

{

SEL originSEL = @selector(sendAction:to:forEvent:);

SEL newSEL = @selector(new_sendAction:to:forEvent:);

Method originM = class_getInstanceMethod([selfclass], originSEL);

constchar *typeEncodinds = method_getTypeEncoding(originM);

Method newM = class_getInstanceMethod([selfclass], newSEL);

IMP newIMP = method_getImplementation(newM);

if(class_addMethod([selfclass], newSEL, newIMP, typeEncodinds))

{

class_replaceMethod([selfclass], originSEL, newIMP, typeEncodinds);

}

else

{

method_exchangeImplementations(originM, newM);

}

}

- (void)my_sendAction:(SEL)action to:(id)target forEvent:(UIEvent*)event {

UITabBarButtonNSLog(@"%@",NSStringFromClass([selfclass]));// 保险起见,判断下Class类型

if([selfisKindOfClass:[UIButtonclass]])

{//1. 按钮点击间隔事件self.clickDurationTime =self.clickDurationTime ==0? defaultDuration :self.clickDurationTime;//2. 是否忽略按钮点击事件if(_isIgnoreEvent)

{

//2.1 忽略按钮事件

return;

}

else if (self.clickDurationTime >0)

{

//2.2 不忽略按钮事件

// 后续在间隔时间内直接忽略按钮事件

_isIgnoreEvent =YES;// 间隔事件后,执行按钮事件dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(self.clickDurationTime *NSEC_PER_SEC)), dispatch_get_main_queue(), ^{            resetState();        });

// 发送按钮点击消息

[selfmy_sendAction:action to:target forEvent:event];

}}

else

{

[selfmy_sendAction:action to:target forEvent:event];

}

}

#pragma mark - associate- (void)setClickDurationTime:(NSTimeInterval)clickDurationTime {objc_setAssociatedObject(self,@selector(clickDurationTime), @(clickDurationTime), OBJC_ASSOCIATION_RETAIN_NONATOMIC);}- (NSTimeInterval)clickDurationTime {

return [objc_getAssociatedObject(self,@selector(clickDurationTime)) doubleValue];

}

@end

这个demo里面的所有按钮都做了防止连续点击事件,你的项目如果想集成这个功能,只需要把demo中的UIControl+Extension的分类放到你的项目中,并在需要用的地方导入头文件,也可以直接放在PCH文件中就OK了.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,768评论 0 9
  • runtime 和 runloop 作为一个程序员进阶是必须的,也是非常重要的, 在面试过程中是经常会被问到的, ...
    made_China阅读 1,226评论 0 7
  • runtime 和 runloop 作为一个程序员进阶是必须的,也是非常重要的, 在面试过程中是经常会被问到的, ...
    SOI阅读 21,848评论 3 63
  • 在之前的开发中用到了UIMenuController这个类,发现一个小问题,记录下来,具体如下 UIMenuCon...
    镌写不一样的人生阅读 1,372评论 0 0
  • Swift2.0 1.defer译为延缓、推迟之意类似栈 注意作用域,其次是调用顺序——即一个作用域结束(注意),...
    zeqinjie阅读 3,410评论 0 50