我正在编写UITableView的子类,我希望我的子类在将它们传递给“真正的”委托之前处理一些UITableViewDelegate方法,并转发我的子类未实现的所有UITableViewDelegate方法。
在子类中我有一个私有属性:
@property (nonatomic, assign) id <UITableViewDelegate> trueDelegate;
保存所有未实现的方法应该转发的“真实委托”。在我的两个init方法中,我设置了
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];
}
}
问题是未实现的委托方法永远不会在tableview上调用,因此它们没有机会被转发到_trueDelegate对象。
我试着在这里检查它们:
- (BOOL)respondsToSelector:(SEL)aSelector {
}
但是从来没有为UITableViewDelegate方法调用该方法,尽管它可以很好地捕获其他方法。
答案 0 :(得分:12)
对于性能,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;
}