Cocoa 使用NSPredicateEditor强制显示复合行
我正在用代码创建一个NSPredicateEditor,并试图实现我所希望的相当简单的任务。基本上,我想显示一个复合的NSPredicateEditorRowTemplate(当“以下所有/任何/无一项为真”时,为用户提供执行匹配的选项)及其下面的一些相当基本的子工作。为此,我构造NSPredicateEditor并绑定其值,以便在用户编辑谓词时保存更改 我遇到的问题是,无论怎样,我都无法使复合NSPredicateEditorRowTemplate在默认情况下显示为具有单个子代码的父行。最初创建谓词时,我传递了一个伪空谓词,如下所示:Cocoa 使用NSPredicateEditor强制显示复合行,cocoa,xcode,nspredicate,nspredicateeditor,Cocoa,Xcode,Nspredicate,Nspredicateeditor,我正在用代码创建一个NSPredicateEditor,并试图实现我所希望的相当简单的任务。基本上,我想显示一个复合的NSPredicateEditorRowTemplate(当“以下所有/任何/无一项为真”时,为用户提供执行匹配的选项)及其下面的一些相当基本的子工作。为此,我构造NSPredicateEditor并绑定其值,以便在用户编辑谓词时保存更改 我遇到的问题是,无论怎样,我都无法使复合NSPredicateEditorRowTemplate在默认情况下显示为具有单个子代码的父行。最初创
filename BEGINSWITH[cd] ''
为了强制显示复合行,我必须创建两个子视图:
filename BEGINSWITH[cd] '' && path ==[cd] ''
以下是我构建NSPredicateEditor的方法,仅供参考:
NSArray *keyPaths = [NSArray arrayWithObjects:[NSExpression expressionForKeyPath:@"filename"], [NSExpression expressionForKeyPath:@"path"], nil];
NSArray *operators = [NSArray arrayWithObjects:[NSNumber numberWithInteger:NSEqualToPredicateOperatorType],
[NSNumber numberWithInteger:NSNotEqualToPredicateOperatorType],
[NSNumber numberWithInteger:NSBeginsWithPredicateOperatorType],
[NSNumber numberWithInteger:NSEndsWithPredicateOperatorType],
[NSNumber numberWithInteger:NSContainsPredicateOperatorType],
nil];
NSPredicateEditorRowTemplate *template = [[NSPredicateEditorRowTemplate alloc] initWithLeftExpressions:keyPaths
rightExpressionAttributeType:NSStringAttributeType
modifier:NSDirectPredicateModifier
operators:operators
options:(NSCaseInsensitivePredicateOption | NSDiacriticInsensitivePredicateOption)];
NSArray *compoundTypes = [NSArray arrayWithObjects:[NSNumber numberWithInteger:NSNotPredicateType],
[NSNumber numberWithInteger:NSAndPredicateType],
[NSNumber numberWithInteger:NSOrPredicateType],
nil];
NSPredicateEditorRowTemplate *compound = [[NSPredicateEditorRowTemplate alloc] initWithCompoundTypes:compoundTypes];
NSArray *rowTemplates = [NSArray arrayWithObjects:compound, template, nil];
[template release];
[compound release];
predicateEditor = [[NSPredicateEditor alloc] init];
[predicateEditor setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable];
[predicateEditor setRowTemplates:rowTemplates];
[predicateEditor setCanRemoveAllRows:NO];
[predicateEditor setContinuous:YES];
[predicateEditor setFrame:[[scrollView contentView] bounds]];
// Create a custom binding for our NSPredicateEditor
SIPredicateStringValueTransformer *transformer = [[[SIPredicateStringValueTransformer alloc] init] autorelease];
NSMutableDictionary *bindingOptions = [NSMutableDictionary dictionaryWithObjectsAndKeys:transformer, NSValueTransformerBindingOption, [NSNumber numberWithBool:YES], NSValidatesImmediatelyBindingOption, nil];
[predicateEditor bind:@"value" toObject:self withKeyPath:@"self.filter.predicate" options:bindingOptions];
// Add our NSPredicateEditor to our NSScrollView
[scrollView setDocumentView:predicateEditor];
问题是你给它一个比较谓词。你需要给它一个复合的 我猜您的
SIPredicateStringValueTransformer
对象正在获取字符串并将其转换为谓词,对吗?它可能在做这样的事情:
- (id) transformedValue:(id)value {
return [NSPredicate predicateWithFormat:value];
}
相反,您要做的是确保值转换器不只是返回任何旧谓词,而是返回一个复合谓词(即NSCompoundPredicate
而不是NSComparisonPredicate
)。做到这一点的方法非常简单:
- (id) transformedValue:(id)value {
NSPredicate *p = [NSPredicate predicateWithFormat:value];
if ([p isKindOfClass:[NSComparisonPredicate class]]) {
p = [NSCompoundPredicate andPredicateWithSubpredicates:[NSArray arrayWithObject:p]];
}
return p;
}
将
NSCompoundPredicate
提供给谓词编辑器时,它将显示复合行。当您只给出一个比较谓词时,它只显示相应的比较行。这是完全正确的。谢谢你这么详细的回答。希望这对将来的人有所帮助。