blob: 643478aaa4b0198fc21af1eea9a50fac7e917b19 (
plain)
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
|
//
// GTMLightweightProxy.m
//
// Copyright 2006-2008 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License"); you may not
// use this file except in compliance with the License. You may obtain a copy
// of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
// License for the specific language governing permissions and limitations under
// the License.
//
#import "GTMLightweightProxy.h"
#import "GTMDefines.h"
@implementation GTMLightweightProxy
- (id)initWithRepresentedObject:(id)object {
// it's weak, we don't retain
representedObject_ = object;
return self;
}
- (id)init {
return [self initWithRepresentedObject:nil];
}
#if MAC_OS_X_VERSION_MIN_REQUIRED >= MAC_OS_X_VERSION_10_5
#if GTM_SUPPORT_GC
// -[NSProxy finalize] is only in 10.5 or later
- (void)finalize {
representedObject_ = nil;
[super finalize];
}
#endif
#endif
- (void)dealloc {
// it's weak, we don't release
representedObject_ = nil;
[super dealloc];
}
- (id)representedObject {
// Use a local variable to avoid a bogus compiler warning.
id repObject = nil;
@synchronized(self) {
// Even though we don't retain this object, we hang it on the lifetime
// of the calling threads pool so it's lifetime is safe for at least that
// long.
repObject = [representedObject_ retain];
}
return [repObject autorelease];
}
- (void)setRepresentedObject:(id)object {
@synchronized(self) {
representedObject_ = object;
}
}
// Passes any unhandled method to the represented object if it responds to that
// method.
- (void)forwardInvocation:(NSInvocation*)invocation {
id target = [self representedObject];
// Silently discard all messages when there's no represented object
if (!target)
return;
SEL aSelector = [invocation selector];
if ([target respondsToSelector:aSelector])
[invocation invokeWithTarget:target];
}
// Gets the represented object's method signature for |selector|; necessary for
// forwardInvocation.
- (NSMethodSignature*)methodSignatureForSelector:(SEL)selector {
id target = [self representedObject];
if (target) {
return [target methodSignatureForSelector:selector];
} else {
// Apple's underlying forwarding code crashes if we return nil here.
// Since we are not going to use the invocation being constructed
// if there's no representedObject, a random valid NSMethodSignature is fine.
return [NSObject methodSignatureForSelector:@selector(alloc)];
}
}
// Prevents exceptions from unknown selectors if there is no represented
// object, and makes the exception come from the right place if there is one.
- (void)doesNotRecognizeSelector:(SEL)selector {
id target = [self representedObject];
if (target)
[target doesNotRecognizeSelector:selector];
}
// Checks the represented object's selectors to allow clients of the proxy to
// do respondsToSelector: tests.
- (BOOL)respondsToSelector:(SEL)selector {
if ([super respondsToSelector:selector] ||
selector == @selector(initWithRepresentedObject:) ||
selector == @selector(representedObject) ||
selector == @selector(setRepresentedObject:))
{
return YES;
}
id target = [self representedObject];
return target && [target respondsToSelector:selector];
}
@end
|