aboutsummaryrefslogtreecommitdiff
path: root/Foundation/GTMLightweightProxy.m
blob: aa2d1a1c85aa6a6a5f0bdcdaa6fc10677afbed4a (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
//
//  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];
}

- (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