]> err.no Git - sope/blob - sope-xml/DOM/DOMTreeWalker.m
fixed some comments
[sope] / sope-xml / DOM / DOMTreeWalker.m
1 /*
2   Copyright (C) 2000-2005 SKYRIX Software AG
3
4   This file is part of SOPE.
5
6   SOPE is free software; you can redistribute it and/or modify it under
7   the terms of the GNU Lesser General Public License as published by the
8   Free Software Foundation; either version 2, or (at your option) any
9   later version.
10
11   SOPE is distributed in the hope that it will be useful, but WITHOUT ANY
12   WARRANTY; without even the implied warranty of MERCHANTABILITY or
13   FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
14   License for more details.
15
16   You should have received a copy of the GNU Lesser General Public
17   License along with SOPE; see the file COPYING.  If not, write to the
18   Free Software Foundation, 59 Temple Place - Suite 330, Boston, MA
19   02111-1307, USA.
20 */
21
22 #include <DOM/DOMTreeWalker.h>
23 #include <DOM/DOMNodeFilter.h>
24 #include <DOM/DOMNode.h>
25 #include "common.h"
26
27 @implementation NGDOMTreeWalker
28
29 - (id)initWithRootNode:(id)_rootNode
30   whatToShow:(unsigned long)_whatToShow
31   filter:(id)_filter
32   expandEntityReferences:(BOOL)_flag
33 {
34   NSAssert(_rootNode, @"missing root-node !");
35
36   if ((self = [super init])) {
37     self->root       = [_rootNode retain];
38     self->whatToShow = _whatToShow;
39     self->filter     = [_filter retain];
40     self->expandEntityReferences = _flag;
41     
42     [self setCurrentNode:_rootNode];
43   }
44   return self;
45 }
46
47 - (void)dealloc {
48   [self->visibleChildren release];
49   [self->currentNode release];
50   [self->root        release];
51   [self->filter      release];
52   [super dealloc];
53 }
54                  
55 /* attributes */
56
57 - (id)root {
58   return self->root;
59 }
60 - (unsigned long)whatToShow {
61   return self->whatToShow;
62 }
63 - (id)filter {
64   return self->filter;
65 }
66 - (BOOL)expandEntityReferences {
67   return self->expandEntityReferences;
68 }
69
70 - (void)setCurrentNode:(id)_node {
71   if (_node == self->currentNode)
72     /* same node */
73     return;
74   
75   ASSIGN(self->currentNode, _node);
76
77   /* clear state caches */
78   [self->visibleChildren release]; self->visibleChildren = nil;
79 }
80 - (id)currentNode {
81   return self->currentNode;
82 }
83
84 /* internals */
85
86 - (BOOL)_shouldShowNode:(id)_node {
87   if (self->whatToShow == DOM_SHOW_ALL)
88     return YES;
89   
90   switch([_node nodeType]) {
91     case DOM_ATTRIBUTE_NODE:
92       return (self->whatToShow & DOM_SHOW_ATTRIBUTE) != 0 ? YES : NO;
93     case DOM_CDATA_SECTION_NODE:
94       return (self->whatToShow & DOM_SHOW_CDATA_SECTION) != 0 ? YES : NO;
95     case DOM_COMMENT_NODE:
96       return (self->whatToShow & DOM_SHOW_COMMENT) != 0 ? YES : NO;
97     case DOM_DOCUMENT_NODE:
98       return (self->whatToShow & DOM_SHOW_DOCUMENT) != 0 ? YES : NO;
99     case DOM_DOCUMENT_FRAGMENT_NODE:
100       return (self->whatToShow & DOM_SHOW_DOCUMENT_FRAGMENT) != 0 ? YES : NO;
101     case DOM_ELEMENT_NODE:
102       return (self->whatToShow & DOM_SHOW_ELEMENT) != 0 ? YES : NO;
103     case DOM_PROCESSING_INSTRUCTION_NODE:
104       return (self->whatToShow & DOM_SHOW_PROCESSING_INSTRUCTION) != 0 ? YES:NO;
105     case DOM_TEXT_NODE:
106       return (self->whatToShow & DOM_SHOW_TEXT) != 0 ? YES : NO;
107     case DOM_DOCUMENT_TYPE_NODE:
108       return (self->whatToShow & DOM_SHOW_DOCUMENT_TYPE) != 0 ? YES : NO;
109     case DOM_ENTITY_NODE:
110       return (self->whatToShow & DOM_SHOW_ENTITY) != 0 ? YES : NO;
111     case DOM_ENTITY_REFERENCE_NODE:
112       return (self->whatToShow & DOM_SHOW_ENTITY_REFERENCE) != 0 ? YES : NO;
113     case DOM_NOTATION_NODE:
114       return (self->whatToShow & DOM_SHOW_NOTATION) != 0 ? YES : NO;
115     default:
116       return YES;
117   }
118 }
119
120 - (BOOL)_isVisibleNode:(id)_node {
121   if (![self _shouldShowNode:_node])
122     return NO;
123   if (self->filter)
124     return [self->filter acceptNode:_node] == DOM_FILTER_ACCEPT ? YES : NO;
125   return YES;
126 }
127 - (unsigned short)_navTypeOfNode:(id)_node {
128   if (![self _shouldShowNode:_node])
129     return DOM_FILTER_SKIP;
130   if (self->filter)
131     return [self->filter acceptNode:_node] == DOM_FILTER_ACCEPT ? YES : NO;
132   return DOM_FILTER_ACCEPT;
133 }
134
135 - (NSArray *)_ensureVisibleChildren {
136   static NSArray *emptyArray = nil;
137   id children;
138   unsigned count;
139
140   if (self->visibleChildren)
141     return self->visibleChildren;
142   
143   children = [[self currentNode] childNodes];
144
145   if ((count = [children count]) > 0) {
146     unsigned i;
147     NSMutableArray *ma;
148     
149     ma = [[NSMutableArray alloc] initWithCapacity:(count + 1)];
150     
151     for (i = 0; i < count; i++) {
152       id childNode;
153       
154       childNode = [children objectAtIndex:i];
155       
156       if ([self _isVisibleNode:childNode])
157         [ma addObject:childNode];
158     }
159     
160     self->visibleChildren = [ma copy];
161     [ma release]; ma = nil;
162   }
163   else {
164     if (emptyArray == nil) emptyArray = [[NSArray alloc] init];
165     self->visibleChildren = [emptyArray retain];
166   }
167   return self->visibleChildren;
168 }
169
170 - (BOOL)_hasVisibleChildren {
171   return [[self _ensureVisibleChildren] count] > 0 ? YES : NO;
172 }
173 - (id)_visibleChildren {
174   return [self _ensureVisibleChildren];
175 }
176 - (id)_firstVisibleChild {
177   NSArray *a = [self _ensureVisibleChildren];
178   if ([a count] == 0)
179     return nil;
180   return [a objectAtIndex:0];
181 }
182 - (id)_lastVisibleChild {
183   NSArray *a = [self _ensureVisibleChildren];
184   unsigned count;
185   
186   if ((count = [a count]) == 0)
187     return nil;
188   return [a objectAtIndex:(count - 1)];
189 }
190
191 - (id<NSObject,DOMNode>)_visibleParentNode {
192   id<NSObject,DOMNode> node;
193
194   for (node = [[self currentNode] parentNode]; node; node =[node parentNode]) {
195     if ([self _isVisibleNode:node])
196       return node;
197     
198     if (node == [self root])
199       /* do not step above root */
200       break;
201   }
202   return nil;
203 }
204
205 - (id<NSObject,DOMNode>)_nextVisibleSibling {
206   id<NSObject,DOMNode> node;
207
208   for (node = [(id<NSObject,DOMNode>)[self currentNode] nextSibling];
209        node != nil;
210        node = [node nextSibling]) {
211     if ([self _isVisibleNode:node])
212       return node;
213   }
214   return nil;
215 }
216 - (id<NSObject,DOMNode>)_previousVisibleSibling {
217   id<NSObject,DOMNode> node;
218
219   for (node = [(id<NSObject,DOMNode>)[self currentNode] previousSibling];
220        node != nil;
221        node = [node previousSibling]) {
222     if ([self _isVisibleNode:node])
223       return node;
224   }
225   return nil;
226 }
227
228 /* operations */
229
230 - (id<NSObject,DOMNode>)parentNode {
231   id parent;
232
233   if ((parent = [self _visibleParentNode])) {
234     [self setCurrentNode:parent];
235     return parent;
236   }
237   else
238     return nil;
239 }
240
241 - (id<NSObject,DOMNode>)firstChild {
242   if ([self _hasVisibleChildren]) {
243     id child;
244
245     child = [self _firstVisibleChild];
246     [self setCurrentNode:child];
247     return child;
248   }
249   else
250     return nil;
251 }
252
253 - (id<NSObject,DOMNode>)lastChild {
254   if ([self _hasVisibleChildren]) {
255     id child;
256
257     child = [self _lastVisibleChild];
258     [self setCurrentNode:child];
259     return child;
260   }
261   else
262     return nil;
263 }
264
265 - (id<NSObject,DOMNode>)previousSibling {
266   id node;
267
268   if ((node = [self _previousVisibleSibling])) {
269     [self setCurrentNode:node];
270     return node;
271   }
272   else
273     return nil;
274 }
275
276 - (id<NSObject,DOMNode>)nextSibling {
277   id node;
278
279   if ((node = [self _nextVisibleSibling])) {
280     [self setCurrentNode:node];
281     return node;
282   }
283   else
284     return nil;
285 }
286
287 - (id<NSObject,DOMNode>)previousNode {
288   [self doesNotRecognizeSelector:_cmd];
289   return nil;
290 }
291 - (id<NSObject,DOMNode>)nextNode {
292   [self doesNotRecognizeSelector:_cmd];
293   return nil;
294 }
295
296 @end /* NGDOMTreeWalker */