-
Notifications
You must be signed in to change notification settings - Fork 84
/
Copy pathMAKVONotificationCenter.h
175 lines (140 loc) · 7.29 KB
/
MAKVONotificationCenter.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
//
// MAKVONotificationCenter.h
// MAKVONotificationCenter
//
// Created by Michael Ash on 10/15/08.
//
#import <Foundation/Foundation.h>
/******************************************************************************/
enum
{
// These constants are technically unsafe to use, as Apple could add options
// with identical values in the future. I'm hoping the highest possible
// bits are high enough for them not to bother with before a point in time
// where it won't matter anymore. Only 32 bits are used, as the definition
// of NSUInteger is 32 bits on iOS.
// Pass this flag to disable automatic de-registration of observers at
// dealloc-time of observer or target. This avoids some swizzling hackery
// on the observer and target objects.
// WARNING: Manual de-registration of observations has the same caveats as
// stardard KVO - deallocating the target or observer objects without
// removing the observation WILL throw KVO errors to the console and cause
// crashes!
MAKeyValueObservingOptionUnregisterManually = 0x80000000,
};
/******************************************************************************/
// An object representing a (potentially) active observation.
@protocol MAKVOObservation <NSObject>
@required
- (BOOL)isValid; // returns NO if the observation has been deregistered by any means
- (void)remove;
@end
/******************************************************************************/
// An object adopting this protocol can be passed as a key path, and every key
// path returned from the required method will be observed. Strings, arrays,
// sets, and ordered sets automatically get this support, as does anything else
// that can be used with for (... in ...)
@protocol MAKVOKeyPathSet <NSObject>
@required
- (id<NSFastEnumeration>)ma_keyPathsAsSetOfStrings;
@end
/******************************************************************************/
@interface MAKVONotification : NSObject
@property(copy,readonly) NSString *keyPath;
@property(assign,readonly) id observer, target;
@property(assign,readonly) NSKeyValueChange kind;
@property(strong,readonly) id oldValue;
@property(strong,readonly) id __attribute__((ns_returns_not_retained)) newValue;
@property(strong,readonly) NSIndexSet *indexes;
@property(assign,readonly) BOOL isPrior;
@end
/******************************************************************************/
// As with Apple's KVO, observer and target are NOT retained.
// An observation object (as returned by an -addObserver: method) will be
// rendered invalid when either the observer or target are deallocated. If you
// hold on to a strong reference past that point, the object will still be
// valid, but will no longer be useful for anything (however, passing it to
// -removeObservation is harmless). It is strongly recommended that
// references to observation objects be weak (or nonexistent), as this will
// make automatic deregistration 100% leak-free. Holding on to an unretained
// (non-weak) reference will cause an observation to go invalid without warning
// unless automatic deregistration is disabled.
// -addObserver:keyPath:selector:userInfo:options: is exactly identical to
// -observeTarget:keyPath:selector:userInfo:options: with the sender and target
// switched; which you use is a matter of preference.
@interface NSObject (MAKVONotification)
- (id<MAKVOObservation>)addObserver:(id)observer
keyPath:(id<MAKVOKeyPathSet>)keyPath
selector:(SEL)selector
userInfo:(id)userInfo
options:(NSKeyValueObservingOptions)options;
- (id<MAKVOObservation>)observeTarget:(id)target
keyPath:(id<MAKVOKeyPathSet>)keyPath
selector:(SEL)selector
userInfo:(id)userInfo
options:(NSKeyValueObservingOptions)options;
#if NS_BLOCKS_AVAILABLE
- (id<MAKVOObservation>)addObservationKeyPath:(id<MAKVOKeyPathSet>)keyPath
options:(NSKeyValueObservingOptions)options
block:(void (^)(MAKVONotification *notification))block;
- (id<MAKVOObservation>)addObserver:(id)observer
keyPath:(id<MAKVOKeyPathSet>)keyPath
options:(NSKeyValueObservingOptions)options
block:(void (^)(MAKVONotification *notification))block;
- (id<MAKVOObservation>)observeTarget:(id)target
keyPath:(id<MAKVOKeyPathSet>)keyPath
options:(NSKeyValueObservingOptions)options
block:(void (^)(MAKVONotification *notification))block;
#endif
- (void)removeAllObservers;
- (void)stopObservingAllTargets;
- (void)removeObserver:(id)observer keyPath:(id<MAKVOKeyPathSet>)keyPath;
- (void)stopObserving:(id)target keyPath:(id<MAKVOKeyPathSet>)keyPath;
- (void)removeObserver:(id)observer keyPath:(id<MAKVOKeyPathSet>)keyPath selector:(SEL)selector;
- (void)stopObserving:(id)target keyPath:(id<MAKVOKeyPathSet>)keyPath selector:(SEL)selector;
@end
/******************************************************************************/
@interface MAKVONotificationCenter : NSObject
+ (id)defaultCenter;
// selector should have the following signature:
// - (void)observeValueForKeyPath:(NSString *)keyPath
// ofObject:(id)target
// change:(NSDictionary *)change
// userInfo:(id)userInfo;
// If target is an NSArray, every object in the collection will be observed,
// per -addObserver:toObjectsAtIndexes:.
- (id<MAKVOObservation>)addObserver:(id)observer
object:(id)target
keyPath:(id<MAKVOKeyPathSet>)keyPath
selector:(SEL)selector
userInfo:(id)userInfo
options:(NSKeyValueObservingOptions)options;
#if NS_BLOCKS_AVAILABLE
- (id<MAKVOObservation>)addObserver:(id)observer
object:(id)target
keyPath:(id<MAKVOKeyPathSet>)keyPath
options:(NSKeyValueObservingOptions)options
block:(void (^)(MAKVONotification *notification))block;
#endif
// remove all observations registered by observer on target with keypath using
// selector. nil for any parameter is a wildcard. One of observer or target
// must be non-nil. The only way to deregister a specific block is to
// remove its particular MAKVOObservation.
- (void)removeObserver:(id)observer object:(id)target keyPath:(id<MAKVOKeyPathSet>)keyPath selector:(SEL)selector;
// remove specific registered observation
- (void)removeObservation:(id<MAKVOObservation>)observation;
@end
/******************************************************************************/
// Declarations to make the basic objects work as key paths; these are
// technically private, but need to be publically visible or the compiler will
// complain.
@interface NSString (MAKeyPath) <MAKVOKeyPathSet>
@end
@interface NSArray (MAKeyPath) <MAKVOKeyPathSet>
@end
@interface NSSet (MAKeyPath) <MAKVOKeyPathSet>
@end
#if __IPHONE_OS_VERSION_MIN_REQUIRED >= __IPHONE_5_0 || __MAC_OS_X_VERSION_MIN_REQUIRED >= __MAC_10_7
@interface NSOrderedSet (MAKeyPath) <MAKVOKeyPathSet>
@end
#endif