写一个UIControl的延展
.h文件
#import <UIKit/UIKit.h>
@interface UIControl (ZQWcontrol)
/** 按钮点击的时间间隔*/
@property(assign,nonatomic)NSTimeInterval cjr_acceptEventInterval;
@end
.m文件
#import "UIControl+ZQWcontrol.h"
#import <objc/runtime.h>
@interface UIControl()
@property(assign,nonatomic)NSTimeInterval cjr_acceptEventTime;
@end;
@implementation UIControl (ZQWcontrol)
//方法原型
id objc_getAssociatedObject(id object, const void *key);
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy);
@selector(categoryProperty) 也就是参数中的 key,其实可以使用静态指针 static void * 类型
的参数来代替,不过在这里,笔者强烈推荐使用 @selector(categoryProperty) 作为 key 传入
。因为这种方法省略了声明参数的代码,并且能很好地保证 key 的唯一性。
*****
OBJC_ASSOCIATION_RETAIN_NONATOMIC 又是什么呢?
我们在代码中实现的属性 NSTimeInterval 就相当于使用了 nonatomic 和 strong 修饰符。(其他的Command 加左键查看它的定义或者自行百度)
-(NSTimeInterval)cjr_acceptEventInterval{
return [objc_getAssociatedObject(self, @selector(cjr_acceptEventInterval)) doubleValue] ;
}
-(void)setCjr_acceptEventInterval:(NSTimeInterval)cjr_acceptEventInterval{
objc_setAssociatedObject(self, @selector(cjr_acceptEventInterval), @(cjr_acceptEventInterval), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (NSTimeInterval )cjr_acceptEventTime{
return [objc_getAssociatedObject(self, @selector(cjr_acceptEventTime)) doubleValue];
}
- (void)setCjr_acceptEventTime:(NSTimeInterval)cjr_acceptEventTime{
objc_setAssociatedObject(self, @selector(cjr_acceptEventTime), @(cjr_acceptEventTime), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
+(void)load{
//获取两个方法
Method systemMethod = class_getInstanceMethod(self, @selector(sendAction:to:forEvent:));
SEL sysSEL = @selector(sendAction:to:forEvent:);
Method myMethod = class_getInstanceMethod(self, @selector(cjr_sendAction:to:forEvent:));
SEL mySEL = @selector(cjr_sendAction:to:forEvent:);
//判断替换的方法有没有实现,如果返回成功,则说明被替换方法没有实现我们需要先把这个方法实现,然后再执行我们想要的效果,用我们自定义的方法去替换被替换的方法. 这里使用到的是class_replaceMethod这个方法. class_replaceMethod本身会尝试调用class_addMethod和method_setImplementation,所以直接调用class_replaceMethod就可以了)
BOOL didAddMethod = class_addMethod(self, sysSEL, method_getImplementation(myMethod), method_getTypeEncoding(myMethod));
if (didAddMethod) {
class_replaceMethod(self, mySEL, method_getImplementation(systemMethod), method_getTypeEncoding(systemMethod));
}else{
//交换方法
method_exchangeImplementations(systemMethod, myMethod);
}
}
- (void)cjr_sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event{
if (NSDate.date.timeIntervalSince1970 - self.cjr_acceptEventTime < self.cjr_acceptEventInterval) {
return;
}
if (self.cjr_acceptEventInterval > 0) {
self.cjr_acceptEventTime = NSDate.date.timeIntervalSince1970;
}
[self cjr_sendAction:action to:target forEvent:event];
}
@end
前方高能----------------------
最近刚刚发现,这样写有一个神坑。会导致tableview侧滑删除按钮的点击事件无效。。。不晓得你有没有中奖。
分享一下我的解决方案
//重写按钮的点击事件的方法。同时把交换方法取消掉。
-(void)sendAction:(SEL)action to:(id)target forEvent:(UIEvent *)event{
if (self. cjr_acceptEventInterval > 0) {
if (NSDate.date.timeIntervalSince1970 - self.cjr_acceptEventTime < self.cjr_acceptEventInterval) {
return;
}
if (self.cjr_acceptEventInterval > 0) {
self.cjr_acceptEventTime = NSDate.date.timeIntervalSince1970;
}
[super sendAction:action to:target forEvent:event];
}else{
[super sendAction:action to:target forEvent:event];
}
}
希望能帮助到你。我的朋友!!