iOS 防止按钮重复点击,runtime拦截按钮的点击事件

写一个UIControl的延展
.h文件

#import 

@interface UIControl (ZQWcontrol)

/** 按钮点击的时间间隔*/
@property(assign,nonatomic)NSTimeInterval cjr_acceptEventInterval;
@end

.m文件

#import "UIControl+ZQWcontrol.h"
#import 

@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];
    }
    
    
}

希望能帮助到你。我的朋友!!

你可能感兴趣的:(iOS 防止按钮重复点击,runtime拦截按钮的点击事件)