我正在编写UITableView的子类,我希望我的子类在将它们传递给“真实”委托之前,先处理一些UITableViewDelegate方法本身,并转发未由我的子类实现的所有UITableViewDelegate方法。
在子类中,我有一个私有财产:
@property (nonatomic, assign) id <UITableViewDelegate> trueDelegate;
其中包含所有未实现的方法应转发给的“真实委托”。在我的两种初始化方法中,
self.delegate = self;
我重写-(void)setDelegate:{id)这样
-(void)setDelegate:(id<UITableViewDelegate>)delegate {
if (delegate != self) {
_trueDelegate = delegate;
} else {
[super setDelegate:self];
}
}
然后我覆盖这些以处理消息转发
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
NSMethodSignature *sig;
sig = [[self.delegate class] instanceMethodSignatureForSelector:aSelector];
if (sig == nil) {
sig = [NSMethodSignature signatureWithObjCTypes:"@^v^c"];
}
return sig;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
SEL selector = anInvocation.selector;
if ([self respondsToSelector:selector]) {
[anInvocation invokeWithTarget:self];
} else {
[anInvocation invokeWithTarget:_trueDelegate];
}
}
问题是未实现的委托方法永远不会在表视图上被调用,因此没有机会将它们转发到_trueDelegate对象。
我尝试在这里检查它们:
- (BOOL)respondsToSelector:(SEL)aSelector {
}
但是,尽管可以捕获其他方法,但从未为UITableViewDelegate方法调用该方法。
最佳答案
为了提高性能,UITableView
检查并记住设置了委托后立即可用的委托方法。首先设置委托self
,然后设置trueDelegate
。因此,在将委托设置在UITableView
上时,trueDelegate
是nil
,因此该代表上的-respondsToSelector:
始终返回NO
。
要解决此问题,请在设置trueDelegate
之后设置委托。另外,您可以简化转发代码。删除上面除属性之外的所有代码,并替换为:
- (void)setDelegate:(id <UITableViewDelegate>)delegate
{
if (delegate == self) return;
self.trueDelegate = delegate;
[super setDelegate:self];
}
- (BOOL)respondsToSelector:(SEL)aSelector
{
if ([super respondsToSelector:aSelector]) return YES;
return [self.trueDelegate respondsToSelector:aSelector];
}
- (id)forwardingTargetForSelector:(SEL)aSelector
{
return self.trueDelegate;
}