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 |
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
There are some limitations to this approach. You can decide whether or not it's worth it.
Return values may have to be casted:
This means that path components returning primitives will never work, despite the fact that at runtime, they are boxed by KVC: