forked from OSchip/llvm-project
148 lines
3.1 KiB
Objective-C
148 lines
3.1 KiB
Objective-C
#import <Foundation/Foundation.h>
|
|
|
|
// SourceBase will be the base class of Source. We'll pass a Source object into a
|
|
// function as a SourceBase, and then see if the dynamic typing can get us through the KVO
|
|
// goo and all the way back to Source.
|
|
|
|
@interface SourceBase: NSObject
|
|
{
|
|
uint32_t _value;
|
|
}
|
|
- (SourceBase *) init;
|
|
- (uint32_t) getValue;
|
|
@end
|
|
|
|
@implementation SourceBase
|
|
- (SourceBase *) init
|
|
{
|
|
[super init];
|
|
_value = 10;
|
|
return self;
|
|
}
|
|
- (uint32_t) getValue
|
|
{
|
|
return _value;
|
|
}
|
|
@end
|
|
|
|
// Source is a class that will be observed by the Observer class below.
|
|
// When Observer sets itself up to observe this property (in initWithASource)
|
|
// the KVO system will overwrite the "isa" pointer of the object with the "kvo'ed"
|
|
// one.
|
|
|
|
@interface Source : SourceBase
|
|
{
|
|
int _property;
|
|
}
|
|
- (Source *) init;
|
|
- (void) setProperty: (int) newValue;
|
|
@end
|
|
|
|
@implementation Source
|
|
- (Source *) init
|
|
{
|
|
[super init];
|
|
_property = 20;
|
|
return self;
|
|
}
|
|
- (void) setProperty: (int) newValue
|
|
{
|
|
_property = newValue; // This is the line in setProperty, make sure we step to here.
|
|
}
|
|
@end
|
|
|
|
@interface SourceDerived : Source
|
|
{
|
|
int _derivedValue;
|
|
}
|
|
- (SourceDerived *) init;
|
|
- (uint32_t) getValue;
|
|
@end
|
|
|
|
@implementation SourceDerived
|
|
- (SourceDerived *) init
|
|
{
|
|
[super init];
|
|
_derivedValue = 30;
|
|
return self;
|
|
}
|
|
- (uint32_t) getValue
|
|
{
|
|
return _derivedValue;
|
|
}
|
|
@end
|
|
|
|
// Observer is the object that will watch Source and cause KVO to swizzle it...
|
|
|
|
@interface Observer : NSObject
|
|
{
|
|
Source *_source;
|
|
}
|
|
+ (Observer *) observerWithSource: (Source *) source;
|
|
- (Observer *) initWithASource: (Source *) source;
|
|
- (void) observeValueForKeyPath: (NSString *) path
|
|
ofObject: (id) object
|
|
change: (NSDictionary *) change
|
|
context: (void *) context;
|
|
@end
|
|
|
|
@implementation Observer
|
|
|
|
+ (Observer *) observerWithSource: (Source *) inSource;
|
|
{
|
|
Observer *retval;
|
|
|
|
retval = [[Observer alloc] initWithASource: inSource];
|
|
return retval;
|
|
}
|
|
|
|
- (Observer *) initWithASource: (Source *) source
|
|
{
|
|
[super init];
|
|
_source = source;
|
|
[_source addObserver: self
|
|
forKeyPath: @"property"
|
|
options: (NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld)
|
|
context: NULL];
|
|
return self;
|
|
}
|
|
|
|
- (void) observeValueForKeyPath: (NSString *) path
|
|
ofObject: (id) object
|
|
change: (NSDictionary *) change
|
|
context: (void *) context
|
|
{
|
|
printf ("Observer function called.\n");
|
|
return;
|
|
}
|
|
@end
|
|
|
|
uint32_t
|
|
handle_SourceBase (SourceBase *object)
|
|
{
|
|
return [object getValue]; // Break here to check dynamic values.
|
|
}
|
|
|
|
int main ()
|
|
{
|
|
SourceDerived *mySource;
|
|
Observer *myObserver;
|
|
|
|
NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
|
|
|
|
mySource = [[SourceDerived alloc] init];
|
|
myObserver = [Observer observerWithSource: mySource];
|
|
|
|
[mySource setProperty: 5]; // Break here to see if we can step into real method.
|
|
|
|
uint32_t return_value = handle_SourceBase (mySource);
|
|
|
|
SourceDerived *unwatchedSource = [[SourceDerived alloc] init];
|
|
|
|
return_value = handle_SourceBase (unwatchedSource);
|
|
|
|
[pool release];
|
|
return 0;
|
|
|
|
}
|