Search code examples
objective-ccocoamacosxcode4nspredicateeditor

NSPredicateEditor in Xcode 4


Bit of an issue with Xcode 4's predicate editor controls - I think I'm doing everything right, and it really seems like the IDE itself is busted.

I'm having this issue in an app I'm writing, but to isolate it, I did the following:

Create a new project with a window. In the XIB editor, add an NSPredicateEditor to it, and add one line. Leave it as key paths/strings and add two key paths - "title" and "writer". Make it case and diacritical insensitive.

Create a subclass of NSWindowController and add an IBOutlet for the predicate editor.

In awakeFromNib, put the following code:

NSPredicate *myPredicate = [NSPredicate predicateWithFormat: @"(title CONTAINS[CD] %@) AND (writer CONTAINS[CD] %@)", @"", @""];

[_predicateEditor setObjectValue:myPredicate];

This yields the following in the console:

2011-04-12 15:59:37.709 PredicateTest[38419:903] Warning - unable to find template matching predicate title CONTAINS[cd] ""

2011-04-12 15:59:37.710 PredicateTest[38419:903] Warning - unable to find template matching predicate writer CONTAINS[cd] ""

When I click the (+) button to add a new row, I get the following:

2011-04-12 15:59:40.044 PredicateTest[38419:903] Cannot create a comparison predicate with nil operator or expression.

Am I right in thinking I'm not doing anything wrong here, and this should work? If I change the predicate editor row template away from key paths, then switch it back to key paths, and edit the list of key paths, Xcode crashes with an internal consistency exception, which makes me think maybe Xcode 4 is not up to snuff when it comes to predicate editing.

Anyone got any ideas? I've tried creating an NSPredicateEditor in code rather than the XIB editor, and endless messing around to no avail. The predicate editing functionality is all that's holding me up releasing an app to the store, so this is kind of an annoyance.


Solution

  • I know this isn't the answer you want to hear, but I highly recommend setting up the the predicate editor programmatically. Setting it up in IB, in my experience, isn't very intuitive. At least in code you can explicitly see what's going on.

    NSArray *keyPaths = @[[NSExpression expressionForKeyPath:@"title"],
                          [NSExpression expressionForKeyPath:@"writer"]];
    NSArray *operators = @[@(NSEqualToPredicateOperatorType),
                           @(NSNotEqualToPredicateOperatorType),
                           @(NSBeginsWithPredicateOperatorType),
                           @(NSEndsWithPredicateOperatorType),
                           @(NSContainsPredicateOperatorType)];
    
    NSPredicateEditorRowTemplate *template = [[NSPredicateEditorRowTemplate alloc] initWithLeftExpressions:keyPaths
                                                                              rightExpressionAttributeType:NSStringAttributeType
                                                                                                  modifier:NSDirectPredicateModifier 
                                                                                                 operators:operators 
                                                                                                   options:(NSCaseInsensitivePredicateOption | NSDiacriticInsensitivePredicateOption)];
    
    NSArray *compoundTypes = @[@(NSNotPredicateType),
                               @(NSAndPredicateType),
                               @(NSOrPredicateType)];
    NSPredicateEditorRowTemplate *compound = [[NSPredicateEditorRowTemplate alloc] initWithCompoundTypes:compoundTypes];
    
    [myPredicateEditor setRowTemplates:@[template, compound]];