Cocoa 未为基于视图的NSOutlineView中的子项调用viewForTableColumn

Cocoa 未为基于视图的NSOutlineView中的子项调用viewForTableColumn,cocoa,nsoutlineview,Cocoa,Nsoutlineview,我有一个问题,那就是方法 -(NSView *)outlineView:(NSOutlineView *)outlineView viewForTableColumn:(NSTableColumn *)tableColumn item:(id)item; 当项目不是组项目时,不会调用。方法 -(BOOL)outlineView:(NSOutlineView *)outlineView isGroupItem:(id)item; -(BOOL)outlineView:(NSOutlineView

我有一个问题,那就是方法

-(NSView *)outlineView:(NSOutlineView *)outlineView viewForTableColumn:(NSTableColumn *)tableColumn item:(id)item;
当项目不是组项目时,不会调用。方法

-(BOOL)outlineView:(NSOutlineView *)outlineView isGroupItem:(id)item;
-(BOOL)outlineView:(NSOutlineView *)outlineView isItemExpandable:(id)item;
对于(相同且)正确的标题/组项目集,实现返回“是”。以下方法

-(NSInteger)outlineView:(NSOutlineView *)outlineView numberOfChildrenOfItem:(id)item;
-(id)outlineView:(NSOutlineView *)outlineView child:(NSInteger)childIndex ofItem:(id)item;
还分别返回子项和有效子项的合理计数。但是上面的
outlineView:viewForTableColumn:item:
方法从未为子项调用。这些行是为子级而布置的,但为空(如下面的屏幕截图所示),因为从未为子级调用视图生成方法
outlineView:viewForTableColumn:item:

请注意,我没有任何XIB,所有视图都是以编程方式布置的。特别是,这意味着我无法通过XIB在outline视图的基础表视图上为Content Mode设置“基于视图”的值,正如在的Apple示例代码中所做的那样

如果有一种方法可以通过编程实现这一点,那将是一个很大的帮助。虽然由于视图方法是为父项调用的,但不确定这是否是真正的问题

对我来说,另一个谜是以下两种方法返回的视图如何相互作用:

- (NSView *)outlineView:(NSOutlineView *)outlineView viewForTableColumn:(NSTableColumn *)tableColumn item:(id)item;
- (NSTableRowView *)outlineView:(NSOutlineView *)outlineView rowViewForItem:(id)item;

关于修复此问题的任何线索都会有很大帮助?

以下是我整理的一些代码,以显示
-(NSView*)outlineView:(NSOutlineView*)outlineView viewForTableColumn:(NSTableColumn*))tableColumn项:(id)项;
确实会被调用。也许通过查看它,您可以看到您在某处犯下的错误:

// AppDelegate.m

#import "AppDelegate.h"
#import "Item.h"

@interface AppDelegate ()

@property (strong) NSMutableArray *items;
@property (weak) IBOutlet NSWindow *window;

@end

@implementation AppDelegate

- (void)applicationDidFinishLaunching:(NSNotification *)aNotification {
    // Insert code here to initialize your application
    self.items = [NSMutableArray array];

    Item *group1 = [Item itemWithName:@"GROUP 1"];
    [group1 addChild:[Item itemWithName:@"0,0"]];
    [group1 addChild:[Item itemWithName:@"1,0"]];
    [group1 addChild:[Item itemWithName:@"2,0"]];

    Item *group2 = [Item itemWithName:@"GROUP 2"];
    [group2 addChild:[Item itemWithName:@"0,1"]];
    [group2 addChild:[Item itemWithName:@"1,1"]];
    [group2 addChild:[Item itemWithName:@"2,1"]];

    Item *group3 = [Item itemWithName:@"GROUP 3"];
    [group3 addChild:[Item itemWithName:@"0,2"]];
    [group3 addChild:[Item itemWithName:@"1,2"]];
    [group3 addChild:[Item itemWithName:@"2,2"]];

    [self.items addObject: group1];
    [self.items addObject: group2];
    [self.items addObject: group3];

    [self.souceList reloadData];
}

- (BOOL)outlineView:(NSOutlineView *)outlineView isGroupItem:(Item *)item {
    return !item.parent;
}

- (BOOL)outlineView:(NSOutlineView *)outlineView isItemExpandable:(Item *)item {
    return item.children.count;
}

- (id)outlineView:(NSOutlineView *)outlineView child:(NSInteger)index ofItem:(Item *)item {
    if (!item) {
        return self.items[index];
    }

    if (item.children.count > index)
        return item.children[index];
    return nil;
}

- (NSInteger)outlineView:(NSOutlineView *)outlineView numberOfChildrenOfItem:(Item *)item {
    if (!item) {
        return self.items.count;
    }
    return item.children.count;
}

- (NSView *)outlineView:(NSOutlineView *)outlineView viewForTableColumn:(NSTableColumn *)tableColumn item:(Item *)item {
    // Breakpoint set here to test
    return [[NSView alloc] init];
}

@end

// Item.h
@interface Item : NSObject
@property (weak) Item *parent;
@property (copy) NSString *name;

+ (instancetype)itemWithName:(NSString *)name;
- (instancetype)initWithName:(NSString *)name;

- (void)addChild:(Item *)child;
- (void)removeChild:(Item *)child;

@end

@interface Item (Property)
@property (readonly, strong) NSArray *children;
@end

// Item.m

#import "Item.h"

@interface Item ()
@property (readwrite, strong) NSMutableArray *children;
@end

@implementation Item

+ (instancetype)itemWithName:(NSString *)name {
    return [[self alloc] initWithName:name];
}

- (instancetype)initWithName:(NSString *)name {
    if ((self = [self init])) {
        self.name = name;
    }
    return self;
}

- (id)init {
    if ((self = [super init])) {
        self.name = @"No Name";
        self.children = [NSMutableArray array];
    }
    return self;
}

- (void)addChild:(Item *)child {
    child.parent = self;
    [self.children addObject: child];
}
- (void)removeChild:(Item *)child {
    if (child.parent == self)
        child.parent = nil;
    [self.children removeObject:child];
}

@end
关于NSTableView/NSOutlineView的
基于视图的状态的第二个问题:该值根据实现的委托/数据源方法自动设置。如果您正在实现
-(NSView*)outlineView:(NSOutlineView*)outlineView viewForTableColumn:(NSTableColumn*)tableColumn项:(id)项,则该视图将被视为基于视图

关于这两个问题,你的最后一个问题是:

- (NSView *)outlineView:(NSOutlineView *)outlineView viewForTableColumn:(NSTableColumn *)tableColumn item:(id)item;
- (NSTableRowView *)outlineView:(NSOutlineView *)outlineView rowViewForItem:(id)item;

第二种方法涉及跨越表格宽度的视图,它包含每个表格列的单元格视图。这意味着您有多个单元格视图——每列一个——由第一个方法返回,这些视图是由第二个方法为每行返回的一个视图的子视图。

这很奇怪。是否调用了
-tableView:viewForTableColumn:row:
?我在发布之前尝试过,只是通过添加该方法再次进行了检查,但没有,没有调用它。我还登录了
-outlineView:shouldExpandItem
,当组项的显示按钮被按下时,它确实会被调用。我只是快速组装了一个示例应用程序,它可以按照您的描述工作,并且
-(NSView*)outlineView:(NSOutlineView*)outlineView viewForTableColumn:(NSTableColumn*)tableColumn item:(id)item。感谢您的详细回答。我将尝试您发布的代码,但我想补充一点,在我的代码中,
viewForTableColumn
中的断点也会被击中,但仅针对父项,而不是子项。是的,我也使用了该断点,我注意到所有人都会被击中。祝你好运谢谢你的提示。事实证明,真正的问题是我没有将我创建的列
outlineTableColumn
添加到
tableColumns
列表中,尽管它已设置为
outlineView.outlineTableColumn
。我一添加行
[outlineView addTableColumn:outlineTableColumn]
,视图就开始出现。在苹果批准的例子中,这是在XIB中完成的,我现在可以通过仔细检查看到。这让我非常好奇,如果不手动添加设置outline列并将其设置为
outlineTableColumn
属性,代码是如何工作的?有什么想法吗?@AlexZielinski因为你的例子是对基于视图的非XIB大纲视图的一个很好的独立解释,我很乐意接受这个答案,特别是因为你花了时间编写了完整的代码。如果你能在你的回答中加入我之前的评论,我很乐意回答。@jkcl我在xib中添加了视图,并将其保留在基于单元格的视图上,没有手动设置,这解释了为什么我没有看到你的outlineTableColumn问题。