Created
May 13, 2012 02:55
-
-
Save jonsterling/2672219 to your computer and use it in GitHub Desktop.
Safe keypaths without macros!
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
NSString *safe = self.keys.url.port.stringValue; | |
NSString *unsafe = @"url.port.stringValue"; | |
assert([safe isEqualToString:unsafe]); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@interface NSObject (SafeKeypaths) | |
+ (instancetype)keys; | |
- (instancetype)keys; | |
@end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@interface JSKeypathRecorder : NSString { | |
NSMutableString *_path; | |
} | |
- (void)appendPathComponent:(NSString *)component; | |
@end | |
@implementation NSObject (SafeKeypaths) | |
+ (instancetype)keys { return [JSKeypathRecorder new]; } | |
- (instancetype)keys { return self.class.keys; } | |
@end | |
@implementation JSKeypathRecorder { | |
NSMutableString *_path; | |
} | |
- (id)init { | |
if ((self = [super init])) | |
_path = [NSMutableString new]; | |
return self; | |
} | |
#pragma mark - NSString Primitives | |
- (NSUInteger)length { | |
return [_path length]; | |
} | |
- (unichar)characterAtIndex:(NSUInteger)index { | |
return [_path characterAtIndex:index]; | |
} | |
#pragma mark - Auxiliary | |
- (void)appendPathComponent:(NSString *)component { | |
[_path appendFormat:@"%@%@", _path.length > 0 ? @"." : @"",component]; | |
} | |
#pragma mark - Invocation Recording | |
- (void)forwardInvocation:(NSInvocation *)anInvocation { | |
[self appendPathComponent:NSStringFromSelector(anInvocation.selector)]; | |
anInvocation.returnValue = (id __strong*)(&self); | |
} | |
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector { | |
NSMethodSignature *signature = [super methodSignatureForSelector:aSelector]; | |
if (signature) | |
return signature; | |
// if we don't already have a signature, let's use one that is the | |
// same as all the messages we intend to accept | |
return [NSNumber instanceMethodSignatureForSelector:@selector(stringValue)]; | |
} | |
@end |
There are some limitations to this approach. You can decide whether or not it's worth it.
Return values may have to be casted:
// will fail, because NSString* != NSNumber*
NSString *path1 = NSURL.keys.port;
// will succeed
NSString *path2 = (id)NSURL.keys.port;
This means that path components returning primitives will never work, despite the fact that at runtime, they are boxed by KVC:
// will fail, because NSString* != NSUInteger
NSString *path3 = NSURL.keys.port.stringValue.length;
// will also fail, because there is no safe cast NSUInteger => NSString*
NSString *path4 = (id)NSURL.keys.port.stringValue.length;
By the way, this also makes your keypaths refactor-safe.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Since all the correct messages will have signatures like
-(id)message
, we can do away with the class lookup and just fudge the method signatures.In fact, since KVO does automatic boxing, the result will be more correct anyway if we just create the signature by hand.