9

我正在获取 UIButton 所属的 UITableViewCell,如下所示:

-(void)buttonHandler:(UIButton *)button {

    OrderCell *cell = [[button superview] superview];
    NSLog(@"cell.item = %@", cell.item.text);

它在 iOS 7 之前的任何东西上都可以正常工作。但是给了我:

[UITableViewCellScrollView item]:无法识别的选择器发送到实例 0x17ae2cf0

如果我在 iOS 7 中运行该应用程序。但如果我这样做:

-(void)buttonHandler:(UIButton *)button {

    OrderCell *cell = [[[button superview] superview] superview];
    NSLog(@"cell.item = %@", cell.item.text);

然后它可以在 iOS 7 中运行,但不是更早的版本?!?!?!

我通过这样做来规避这个问题:

OrderCell *cell;
if([[[UIDevice currentDevice] systemVersion] isEqualToString:@"7.0"])
    cell = [[[button superview] superview] superview];
else
    cell = [[button superview] superview];

NSLog(@"cell.item = %@", cell.item.text);

但是WTF正在进行中!?有谁知道为什么会这样?

谢谢!

4

5 回答 5

27

更好的解决方案是为 UIView(SuperView) 添加一个类别,并通过以下方式调用它:

UITableViewCell *cell = [button findSuperViewWithClass:[UITableViewCell class]]

这样,您的代码适用于所有未来和过去的 iOS 版本

@interface UIView (SuperView)

- (UIView *)findSuperViewWithClass:(Class)superViewClass;

@end


@implementation UIView (SuperView)

- (UIView *)findSuperViewWithClass:(Class)superViewClass {

    UIView *superView = self.superview;
    UIView *foundSuperView = nil;

    while (nil != superView && nil == foundSuperView) {
        if ([superView isKindOfClass:superViewClass]) {
            foundSuperView = superView;
        } else {
            superView = superView.superview;
        }
    }
    return foundSuperView;
}
@end
于 2013-09-23T15:44:58.673 回答
13

最好的方法是:

CGPoint buttonPosition = [sender convertPoint:CGPointZero toView:self.tableView];
NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:buttonPosition];
UITableViewCell *cell = (UITableViewCell*)[self.tableView cellForRowAtIndexPath:indexPath];
于 2013-12-10T02:14:06.957 回答
3

要完成@thomas-keuleers的答案,这是快速方法:

extension UIView {

    func findSuperViewWithClass<T>(superViewClass : T.Type) -> UIView? {

        var xsuperView : UIView!  = self.superview!
        var foundSuperView : UIView!

        while (xsuperView != nil && foundSuperView == nil) {

            if xsuperView.self is T {
                foundSuperView = xsuperView
            } else {
                xsuperView = xsuperView.superview
            }
        }
        return foundSuperView
    }

}

你只需这样调用:

child.findSuperViewWithClass(TableViewCell)
于 2015-06-05T15:04:16.270 回答
0

swift 5 中的较短版本

extension UIView {
   var xsuperView = self.superview

   while xsuperView != nil {
        if let superView = xsuperView.self as? T {
            return superView
        } else {
            xsuperView = xsuperView?.superview
        }
    }
}
于 2019-05-22T12:57:00.340 回答
-2
if ([[button superView] isKindOfClass:[UITableViewCell class]]) {

}

else //check next :
于 2013-09-23T15:23:50.397 回答