寫一個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];
}
}
希望能幫助到你。我的朋友!!