1// [The "BSD licence"]
2// Copyright (c) 2006-2007 Kay Roepke 2010 Alan Condit
3// All rights reserved.
4//
5// Redistribution and use in source and binary forms, with or without
6// modification, are permitted provided that the following conditions
7// are met:
8// 1. Redistributions of source code must retain the above copyright
9//    notice, this list of conditions and the following disclaimer.
10// 2. Redistributions in binary form must reproduce the above copyright
11//    notice, this list of conditions and the following disclaimer in the
12//    documentation and/or other materials provided with the distribution.
13// 3. The name of the author may not be used to endorse or promote products
14//    derived from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
17// IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
18// OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
19// IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
20// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
21// NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
22// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
23// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
25// THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
27#import "RewriteRuleElementStream.h"
28
29@implementation RewriteRuleElementStream
30
31@synthesize cursor;
32@synthesize dirty;
33@synthesize isSingleElement;
34@synthesize singleElement;
35@synthesize elements;
36@synthesize elementDescription;
37@synthesize treeAdaptor;
38
39+ (RewriteRuleElementStream *) newRewriteRuleElementStream:(id<TreeAdaptor>)aTreeAdaptor
40                                                         description:(NSString *)anElementDescription
41{
42    return [[RewriteRuleElementStream alloc] initWithTreeAdaptor:aTreeAdaptor
43                                                          description:anElementDescription];
44}
45
46+ (RewriteRuleElementStream *) newRewriteRuleElementStream:(id<TreeAdaptor>)aTreeAdaptor
47                                                         description:(NSString *)anElementDescription
48                                                             element:(id)anElement
49{
50    return [[RewriteRuleElementStream alloc] initWithTreeAdaptor:aTreeAdaptor
51                                                          description:anElementDescription
52                                                              element:anElement];
53}
54
55+ (RewriteRuleElementStream *) newRewriteRuleElementStream:(id<TreeAdaptor>)aTreeAdaptor
56                                                         description:(NSString *)anElementDescription
57                                                            elements:(NSArray *)theElements;
58{
59    return [[RewriteRuleElementStream alloc] initWithTreeAdaptor:aTreeAdaptor
60                                                          description:anElementDescription
61                                                             elements:theElements];
62}
63
64- (id) initWithTreeAdaptor:(id<TreeAdaptor>)aTreeAdaptor description:(NSString *)anElementDescription
65{
66    if ((self = [super init]) != nil) {
67        cursor = 0;
68        dirty = NO;
69        [self setDescription:anElementDescription];
70        [self setTreeAdaptor:aTreeAdaptor];
71        dirty = NO;
72        isSingleElement = YES;
73        singleElement = nil;
74        elements = nil;
75    }
76    return self;
77}
78
79- (id) initWithTreeAdaptor:(id<TreeAdaptor>)aTreeAdaptor description:(NSString *)anElementDescription element:(id)anElement
80{
81    if ((self = [super init]) != nil) {
82        cursor = 0;
83        dirty = NO;
84        [self setDescription:anElementDescription];
85        [self setTreeAdaptor:aTreeAdaptor];
86        dirty = NO;
87        isSingleElement = YES;
88        singleElement = nil;
89        elements = nil;
90        [self addElement:anElement];
91    }
92    return self;
93}
94
95- (id) initWithTreeAdaptor:(id<TreeAdaptor>)aTreeAdaptor description:(NSString *)anElementDescription elements:(NSArray *)theElements
96{
97    self = [super init];
98    if (self) {
99        cursor = 0;
100        dirty = NO;
101        [self setDescription:anElementDescription];
102        [self setTreeAdaptor:aTreeAdaptor];
103        dirty = NO;
104        singleElement = nil;
105        isSingleElement = NO;
106        elements = [[AMutableArray arrayWithArray:theElements] retain];
107    }
108    return self;
109}
110
111- (void) dealloc
112{
113#ifdef DEBUG_DEALLOC
114    NSLog( @"called dealloc in RewriteRuleElementStream" );
115#endif
116    if ( singleElement && isSingleElement ) [singleElement release];
117    else if ( elements && !isSingleElement ) [elements release];
118    [self setDescription:nil];
119    [self setTreeAdaptor:nil];
120    [super dealloc];
121}
122
123- (void)reset
124{
125    cursor = 0;
126    dirty = YES;
127}
128
129- (id<TreeAdaptor>) getTreeAdaptor
130{
131    return treeAdaptor;
132}
133
134- (void) setTreeAdaptor:(id<TreeAdaptor>)aTreeAdaptor
135{
136    if (treeAdaptor != aTreeAdaptor) {
137        if ( treeAdaptor ) [treeAdaptor release];
138        treeAdaptor = aTreeAdaptor;
139        [treeAdaptor retain];
140    }
141}
142
143- (void) addElement: (id)anElement
144{
145    if (anElement == nil)
146        return;
147    if (elements != nil) {
148        [elements addObject:anElement];
149        return;
150        }
151    if (singleElement == nil) {
152        singleElement = anElement;
153        singleElement = [anElement retain];
154        return;
155    }
156    isSingleElement = NO;
157    elements = [[AMutableArray arrayWithCapacity:5] retain];
158    [elements addObject:singleElement];
159    singleElement = nil;  // balance previous retain in initializer/addElement
160    [elements addObject:anElement];
161}
162
163- (void) setElement: (id)anElement
164{
165    if (anElement == nil)
166        return;
167    if (elements != nil) {
168        [elements addObject:anElement];
169        return;
170        }
171    if (singleElement == nil) {
172        singleElement = anElement;
173        singleElement = [anElement retain];
174        return;
175    }
176    isSingleElement = NO;
177    elements = [[AMutableArray arrayWithCapacity:5] retain];
178    [elements addObject:singleElement];
179    singleElement = nil;  // balance previous retain in initializer/addElement
180    [elements addObject:anElement];
181}
182
183- (id<BaseTree>) nextTree
184{
185    NSInteger n = [self size];
186    if ( dirty && (cursor >= 0 && n == 1)) {
187        // if out of elements and size is 1, dup
188        id element = [self _next];
189        return [self copyElement:element];
190    }
191    // test size above then fetch
192    id element = [self _next];
193    return element;
194}
195
196- (id) _next       // internal: TODO: redesign if necessary. maybe delegate
197{
198    NSInteger n = [self size];
199    if (n == 0) {
200        @throw [NSException exceptionWithName:@"RewriteEmptyStreamException" reason:nil userInfo:nil];// TODO: fill in real exception
201    }
202    if ( cursor >= n ) {
203        if ( n == 1 ) {
204            return [self toTree:singleElement]; // will be dup'ed in -next
205        }
206        @throw [NSException exceptionWithName:@"RewriteCardinalityException" reason:nil userInfo:nil];// TODO: fill in real exception
207    }
208    if (singleElement != nil) {
209        cursor++;
210        return [self toTree:singleElement];
211    }
212    id el = [elements objectAtIndex:cursor];
213    cursor++;
214    return [self toTree:el];
215}
216
217- (BOOL) hasNext
218{
219    return (singleElement != nil && cursor < 1) ||
220            (elements != nil && cursor < [elements count]);
221}
222
223- (NSInteger) size
224{
225    NSInteger n = 0;
226    if (singleElement != nil)
227        n = 1;
228    if (elements != nil)
229        return [elements count];
230    return n;
231}
232
233- (id) copyElement:(id)element
234{
235    [self doesNotRecognizeSelector:_cmd];   // subclass responsibility
236    return nil;
237}
238
239- (id<BaseTree>) toTree:(id)element
240{
241    return element;
242}
243
244- (NSString *) getDescription
245{
246    return elementDescription;
247}
248
249- (void) setDescription:(NSString *) description
250{
251    if ( description != nil && description != elementDescription ) {
252        if (elementDescription != nil) [elementDescription release];
253        elementDescription = [NSString stringWithString:description];
254        [elementDescription retain];
255    }
256}
257
258@end
259