3 function XML_Template_process_file($filename, $obj, $clean = 1)
5 $doc = new DOMDocument;
7 XML_Template_process($doc, $obj, $clean);
11 function XML_Template_process($node, $obj, $clean = 1)
13 if (is_a($obj, 'DOMNode')) { # overwrite
14 while ($node->childNodes->length > 0) {
15 $child = $node->childNodes->item(0);
16 $node->removeChild($child);
19 if (is_a($obj, 'DOMDocument')) {
20 $obj = $obj->documentElement;
23 $frag = $node->ownerDocument->createDocumentFragment();
24 $frag->appendChild($node->ownerDocument->importNode($obj, true));
25 XML_Template_process($frag, array(), $clean);
26 $node->appendChild($frag);
27 } else if (!is_array($obj)) { # overwrite
28 while ($node->childNodes->length > 0) {
29 $node->removeChild($node->firstChild);
31 $doc = $node->ownerDocument;
32 $node->appendChild($doc->createTextNode($obj));
33 } else if (is_associative_array($obj)) { # substitute
34 $num_children = ($node->childNodes == null) ? 0 : $node->childNodes->length;
36 for ($i = 0; $i < $num_children; ++$i) {
37 $children[] = $node->childNodes->item($i);
40 for ($i = 0; $i < $num_children; ++$i) {
41 $child = $children[$i];
44 if ($child->nodeType == XML_ELEMENT_NODE) {
47 $tag = $child->localName;
48 $attrs = $child->attributes;
52 foreach ($child->attributes as $attr) {
53 if ($attr->namespaceURI == 'http://template.sesse.net/' && $attr->name == 'id') {
56 $child->removeAttributeNode($attr);
62 # This seems to be the only way to reliably get rid of the excess
63 # XML namespace declarations.
64 if ($replace_child == 1) {
65 $newchild = own_clone_element($child, $child->ownerDocument);
66 while ($child->childNodes->length > 0) {
67 $newchild->appendChild($child->firstChild);
69 $node->replaceChild($newchild, $child);
74 # check all substitutions to see if we found anything
76 foreach (array_keys($obj) as $key) {
77 # FIXME: we would want something like \Q and \E here...
78 if (preg_match('/^' . $tag . '\/(.*)$/', $key, $matches) ||
79 (isset($id) && preg_match('/^#' . $id . '\/(.*)$/', $key, $matches))) {
80 $child->setAttribute($matches[1], $obj[$key]);
86 if ($key == $tag || (isset($id) && $key == ('#'.$id))) {
87 XML_Template_process($child, $obj[$key], $clean);
94 XML_Template_process($child, $obj, $clean);
98 $doc = $node->ownerDocument;
99 $frag = $doc->createDocumentFragment();
101 while ($node->childNodes->length > 0) {
102 $child = $node->childNodes->item(0);
103 $node->removeChild($child);
104 $frag->appendChild($child);
107 foreach ($obj as $instance) {
108 if (!isset($instance)) {
112 $newnode = own_clone_node($frag, $frag->ownerDocument);
113 XML_Template_process($newnode, $instance, $clean);
115 $node->appendChild($newnode);
117 XML_Template_clean($newnode);
123 XML_Template_clean($node);
127 function XML_Template_clean($node)
129 if ($node->nodeType == XML_ELEMENT_NODE) {
130 if ($node->namespaceURI != 'http://template.sesse.net/') {
134 # as this is a dummy node, we want to remove it and move everything further up
135 # after we've done any required replacements
136 $doc = $node->ownerDocument;
137 $parent = $node->parentNode;
139 while ($node->childNodes->length > 0) {
140 $child = $node->childNodes->item(0);
141 $node->removeChild($child);
142 $parent->insertBefore(own_clone_node($child, $doc), $node);
144 $parent->removeChild($node);
148 # FIXME: use varargs here
149 function XML_Template_alternate($tag, $array, $elems)
151 $num = count($elems);
153 for ($i = 0, $j = 0; $i < count($array); $i++) {
154 if (isset($array[$i])) {
155 $array[$i][$tag] = $elems[$j++ % $num];
162 # Ideally, this would be "return $obj->clone_node(true)". But surprise,
163 # surprise, PHP is buggy; it does not preserve the prefix information on
164 # attributes properly during a clone. (PHP4 and PHP5 are broken in different
165 # ways here...) IOW, we'll have to clone everything ourselves.
166 function own_clone_node($node, $doc)
168 // we only need these two
169 if ($node->nodeType == XML_DOCUMENT_FRAG_NODE) {
170 $newnode = $doc->createDocumentFragment();
172 for ($i = 0; $i < $node->childNodes->length; ++$i) {
173 $child = $node->childNodes->item($i);
174 $newnode->appendChild(own_clone_node($child, $doc));
177 } else if ($node->nodeType == XML_ELEMENT_NODE) {
178 $newnode = own_clone_element($node, $doc);
179 for ($i = 0; $i < $node->childNodes->length; ++$i) {
180 $child = $node->childNodes->item($i);
181 $newnode->appendChild(own_clone_node($child, $doc));
185 return $node->cloneNode(true);
189 function own_clone_element($node, $doc)
191 $nsuri = $node->namespaceURI;
192 if (isset($nsuri) && $node->prefix != "default") {
193 $newnode = $doc->createElementNS($node->namespaceURI, $node->nodeName, $node->prefix);
195 // remove useless empty text child
196 $newnode->removeChild($newnode->firstChild);
198 $newnode = $doc->createElement($node->localName);
201 $attrs = $node->attributes;
203 foreach ($node->attributes as $attr) {
204 $nsuri = $attr->namespaceURI;
205 if (isset($nsuri) && $attr->prefix != "default") {
206 $attr2 = $doc->createAttributeNS($nsuri, $attr->prefix . ":" . $attr->name);
208 $attr2 = $doc->createAttribute($attr->localName);
211 # You've got to be kidding me...
212 $attr2->value = preg_replace("/&/", "&", $attr->value);
213 $newnode->appendChild($attr2);
219 function is_associative_array($arr)
221 if (!is_array($arr)) {
224 $diff = array_diff(range(0, count($arr) - 1), array_keys($arr));
225 return (count($diff) > 0);