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
|
########################################################################
#
# File Name: CallTemplateElement.py
#
#
"""
Implementation of the XSLT Spec call-template stylesheet element.
WWW: http://4suite.com/4XSLT e-mail: support@4suite.com
Copyright (c) 1999-2000 Fourthought Inc, USA. All Rights Reserved.
See http://4suite.com/COPYRIGHT for license and copyright information
"""
from xml.dom import EMPTY_NAMESPACE
from xml.dom import Node
import xml.dom.ext
import xml.xslt
from xml.xslt import XsltElement, XsltException, Error, XSL_NAMESPACE
from xml.xpath import Util
class CallTemplateElement(XsltElement):
legalAttrs = ('name', )
def __init__(self, doc, uri=xml.xslt.XSL_NAMESPACE,
localName='call-template', prefix='xsl', baseUri=''):
XsltElement.__init__(self, doc, uri, localName, prefix, baseUri)
def setup(self):
self.__dict__['_nss'] = xml.dom.ext.GetAllNs(self)
split_name = Util.ExpandQName(
self.getAttributeNS(EMPTY_NAMESPACE, 'name'),
namespaces=self._nss
)
self.__dict__['_name'] = split_name
self.__dict__['_tailRecursive'] = "unknown"
self.__dict__['_params'] = filter(lambda node:
node.nodeType == Node.ELEMENT_NODE,
self.childNodes)
for child in self.__dict__['_params']:
if child.nodeType == Node.ELEMENT_NODE:
if child.namespaceURI == XSL_NAMESPACE:
if child.localName != 'with-param':
raise XsltException(Error.ILLEGAL_CALLTEMPLATE_CHILD)
else:
raise XsltException(Error.ILLEGAL_CALLTEMPLATE_CHILD)
return
def instantiate(self, context, processor, new_level=1):
if self._tailRecursive == 'unknown':
self.__dict__['_tailRecursive'] = CheckTailRecursion(self, self._name)
origState = context.copy()
context.setNamespaces(self._nss)
params = {}
for child in self._params:
param = child.instantiate(context, processor)[1]
params[param[0]] = param[1]
if self._tailRecursive:
if not new_level:
context.set(origState)
return (context, params)
while params is not None:
params = processor.callTemplate(self._name, context, params, 0)
if params:
context.varBindings.update(params)
else:
processor.callTemplate(self._name, context, params, 1)
context.set(origState)
return (context, None)
def __getinitargs__(self):
return (None, self.namespaceURI, self.localName, self.prefix,
self.baseUri)
def __getstate__(self):
base_state = XsltElement.__getstate__(self)
new_state = (base_state,
self._nss,
self._name,
self._tailRecursive,
self._params)
return new_state
def __setstate__(self, state):
XsltElement.__setstate__(self, state[0])
self._nss = state[1]
self._name = state[2]
self._tailRecursive = state[3]
self._params = state[4]
return
def CheckTailRecursion(node, name):
if node.nextSibling and node.nodeType == Node.ELEMENT_NODE:
return 0
p = node.nodeType == Node.ATTRIBUTE_NODE and node.ownerElement or node.parentNode
if p and p.nodeType == Node.ELEMENT_NODE and p.namespaceURI == XSL_NAMESPACE:
if p.localName in ['if', 'choose']:
return CheckTailRecursion(p, name)
elif p.localName == 'template' and name == p._name:
return 1
return 0
|