iOS 防止按鈕重復點擊,runtime攔截按鈕的點擊事件

寫一個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];
    }
    
    
}

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

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容