1 from __future__ import absolute_import, division, unicode_literals
4 from collections import MutableMapping
5 from xml.dom import minidom, Node
9 from .. import constants
10 from ..constants import namespaces
11 from .._utils import moduleFactoryFactory
14 def getDomBuilder(DomImplementation):
15 Dom = DomImplementation
17 class AttrList(MutableMapping):
18 def __init__(self, element):
19 self.element = element
22 return iter(self.element.attributes.keys())
24 def __setitem__(self, name, value):
25 if isinstance(name, tuple):
26 raise NotImplementedError
28 attr = self.element.ownerDocument.createAttribute(name)
30 self.element.attributes[name] = attr
33 return len(self.element.attributes)
36 return list(self.element.attributes.items())
39 return list(self.element.attributes.values())
41 def __getitem__(self, name):
42 if isinstance(name, tuple):
43 raise NotImplementedError
45 return self.element.attributes[name].value
47 def __delitem__(self, name):
48 if isinstance(name, tuple):
49 raise NotImplementedError
51 del self.element.attributes[name]
53 class NodeBuilder(base.Node):
54 def __init__(self, element):
55 base.Node.__init__(self, element.nodeName)
56 self.element = element
58 namespace = property(lambda self: hasattr(self.element, "namespaceURI") and
59 self.element.namespaceURI or None)
61 def appendChild(self, node):
63 self.element.appendChild(node.element)
65 def insertText(self, data, insertBefore=None):
66 text = self.element.ownerDocument.createTextNode(data)
68 self.element.insertBefore(text, insertBefore.element)
70 self.element.appendChild(text)
72 def insertBefore(self, node, refNode):
73 self.element.insertBefore(node.element, refNode.element)
76 def removeChild(self, node):
77 if node.element.parentNode == self.element:
78 self.element.removeChild(node.element)
81 def reparentChildren(self, newParent):
82 while self.element.hasChildNodes():
83 child = self.element.firstChild
84 self.element.removeChild(child)
85 newParent.element.appendChild(child)
88 def getAttributes(self):
89 return AttrList(self.element)
91 def setAttributes(self, attributes):
93 for name, value in list(attributes.items()):
94 if isinstance(name, tuple):
95 if name[0] is not None:
96 qualifiedName = (name[0] + ":" + name[1])
98 qualifiedName = name[1]
99 self.element.setAttributeNS(name[2], qualifiedName,
102 self.element.setAttribute(
104 attributes = property(getAttributes, setAttributes)
107 return NodeBuilder(self.element.cloneNode(False))
109 def hasContent(self):
110 return self.element.hasChildNodes()
112 def getNameTuple(self):
113 if self.namespace is None:
114 return namespaces["html"], self.name
116 return self.namespace, self.name
118 nameTuple = property(getNameTuple)
120 class TreeBuilder(base.TreeBuilder): # pylint:disable=unused-variable
121 def documentClass(self):
122 self.dom = Dom.getDOMImplementation().createDocument(None, None, None)
123 return weakref.proxy(self)
125 def insertDoctype(self, token):
127 publicId = token["publicId"]
128 systemId = token["systemId"]
130 domimpl = Dom.getDOMImplementation()
131 doctype = domimpl.createDocumentType(name, publicId, systemId)
132 self.document.appendChild(NodeBuilder(doctype))
134 doctype.ownerDocument = self.dom
136 def elementClass(self, name, namespace=None):
137 if namespace is None and self.defaultNamespace is None:
138 node = self.dom.createElement(name)
140 node = self.dom.createElementNS(namespace, name)
142 return NodeBuilder(node)
144 def commentClass(self, data):
145 return NodeBuilder(self.dom.createComment(data))
147 def fragmentClass(self):
148 return NodeBuilder(self.dom.createDocumentFragment())
150 def appendChild(self, node):
151 self.dom.appendChild(node.element)
153 def testSerializer(self, element):
154 return testSerializer(element)
156 def getDocument(self):
159 def getFragment(self):
160 return base.TreeBuilder.getFragment(self).element
162 def insertText(self, data, parent=None):
165 base.TreeBuilder.insertText(self, data, parent)
167 # HACK: allow text nodes as children of the document node
168 if hasattr(self.dom, '_child_node_types'):
169 # pylint:disable=protected-access
170 if Node.TEXT_NODE not in self.dom._child_node_types:
171 self.dom._child_node_types = list(self.dom._child_node_types)
172 self.dom._child_node_types.append(Node.TEXT_NODE)
173 self.dom.appendChild(self.dom.createTextNode(data))
175 implementation = DomImplementation
178 def testSerializer(element):
182 def serializeElement(element, indent=0):
183 if element.nodeType == Node.DOCUMENT_TYPE_NODE:
185 if element.publicId or element.systemId:
186 publicId = element.publicId or ""
187 systemId = element.systemId or ""
188 rv.append("""|%s<!DOCTYPE %s "%s" "%s">""" %
189 (' ' * indent, element.name, publicId, systemId))
191 rv.append("|%s<!DOCTYPE %s>" % (' ' * indent, element.name))
193 rv.append("|%s<!DOCTYPE >" % (' ' * indent,))
194 elif element.nodeType == Node.DOCUMENT_NODE:
195 rv.append("#document")
196 elif element.nodeType == Node.DOCUMENT_FRAGMENT_NODE:
197 rv.append("#document-fragment")
198 elif element.nodeType == Node.COMMENT_NODE:
199 rv.append("|%s<!-- %s -->" % (' ' * indent, element.nodeValue))
200 elif element.nodeType == Node.TEXT_NODE:
201 rv.append("|%s\"%s\"" % (' ' * indent, element.nodeValue))
203 if (hasattr(element, "namespaceURI") and
204 element.namespaceURI is not None):
205 name = "%s %s" % (constants.prefixes[element.namespaceURI],
208 name = element.nodeName
209 rv.append("|%s<%s>" % (' ' * indent, name))
210 if element.hasAttributes():
212 for i in range(len(element.attributes)):
213 attr = element.attributes.item(i)
216 ns = attr.namespaceURI
218 name = "%s %s" % (constants.prefixes[ns], attr.localName)
221 attributes.append((name, value))
223 for name, value in sorted(attributes):
224 rv.append('|%s%s="%s"' % (' ' * (indent + 2), name, value))
226 for child in element.childNodes:
227 serializeElement(child, indent)
228 serializeElement(element, 0)
235 # The actual means to get a module!
236 getDomModule = moduleFactoryFactory(getDomBuilder)