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对于组行和其他行,都会调用code>。感谢您的详细回答。我将尝试您发布的代码,但我想补充一点,在我的代码中,viewForTableColumn
中的断点也会被击中,但仅针对父项,而不是子项。是的,我也使用了该断点,我注意到所有人都会被击中。祝你好运谢谢你的提示。事实证明,真正的问题是我没有将我创建的列outlineTableColumn
添加到tableColumns
列表中,尽管它已设置为outlineView.outlineTableColumn
。我一添加行[outlineView addTableColumn:outlineTableColumn]
,视图就开始出现。在苹果批准的例子中,这是在XIB中完成的,我现在可以通过仔细检查看到。这让我非常好奇,如果不手动添加设置outline列并将其设置为outlineTableColumn
属性,代码是如何工作的?有什么想法吗?@AlexZielinski因为你的例子是对基于视图的非XIB大纲视图的一个很好的独立解释,我很乐意接受这个答案,特别是因为你花了时间编写了完整的代码。如果你能在你的回答中加入我之前的评论,我很乐意回答。@jkcl我在xib中添加了视图,并将其保留在基于单元格的视图上,没有手动设置,这解释了为什么我没有看到你的outlineTableColumn问题。