Commit 1dff9e98 authored by Guido van Rossum's avatar Guido van Rossum

Add opcode implementation for "mode".

Properly implement tal:replace="structure ..." combined with
tal:attributes="...", both in HTML and in XML mode.  This is pretty
tricky, and requires instantiating a parser with a modified code
generator which suppresses any TAL/METAL interpretation but does the
attribute replacement on the first encountered start tag as required
by the TAL 1.0 specs.
parent 08e7613c
...@@ -96,8 +96,8 @@ try: ...@@ -96,8 +96,8 @@ try:
except ImportError: except ImportError:
from StringIO import StringIO from StringIO import StringIO
from XMLParser import XMLParser
from TALDefs import TALError, TALESError, quote, TAL_VERSION from TALDefs import TALError, TALESError, quote, TAL_VERSION
from TALGenerator import TALGenerator
BOOLEAN_HTML_ATTRS = [ BOOLEAN_HTML_ATTRS = [
# List of Boolean attributes in HTML that should be rendered in # List of Boolean attributes in HTML that should be rendered in
...@@ -118,6 +118,35 @@ EMPTY_HTML_TAGS = [ ...@@ -118,6 +118,35 @@ EMPTY_HTML_TAGS = [
"input", "col", "basefont", "isindex", "frame", "input", "col", "basefont", "isindex", "frame",
] ]
class AltTALGenerator(TALGenerator):
def __init__(self, repldict, expressionCompiler=None, xml=0):
self.repldict = repldict
self.enabled = 1
TALGenerator.__init__(self, expressionCompiler, xml)
def enable(self, enabled):
self.enabled = enabled
def emit(self, *args):
if self.enabled:
apply(TALGenerator.emit, (self,) + args)
def emitStartElement(self, name, attrlist, taldict, metaldict,
position=(None, None), isend=0):
metaldict = {}
taldict = {}
if self.enabled and self.repldict:
taldict["attributes"] = ""
TALGenerator.emitStartElement(self, name, attrlist,
taldict, metaldict, position, isend)
def replaceAttrs(self, attrlist, repldict):
if self.enabled and self.repldict:
repldict = self.repldict
self.repldict = None
return TALGenerator.replaceAttrs(self, attrlist, repldict)
class TALInterpreter: class TALInterpreter:
def __init__(self, program, macros, engine, stream=None, def __init__(self, program, macros, engine, stream=None,
...@@ -180,6 +209,10 @@ class TALInterpreter: ...@@ -180,6 +209,10 @@ class TALInterpreter:
def do_version(self, version): def do_version(self, version):
assert version == TAL_VERSION assert version == TAL_VERSION
def do_mode(self, mode):
assert mode in ["html", "xml"]
self.html = (mode == "html")
def do_setPosition(self, position): def do_setPosition(self, position):
self.position = position self.position = position
...@@ -262,19 +295,32 @@ class TALInterpreter: ...@@ -262,19 +295,32 @@ class TALInterpreter:
structure = self.engine.evaluateStructure(expr) structure = self.engine.evaluateStructure(expr)
if structure is None: if structure is None:
return return
if repldict:
raise TALError("replace structure with attribute replacements "
"not yet implemented", self.position)
text = str(structure) text = str(structure)
self.checkXMLSyntax(text) if self.html:
self.stream_write(text) # No quoting -- this is intentional self.insertHTMLStructure(text, repldict)
else:
self.insertXMLStructure(text, repldict)
def checkXMLSyntax(self, text): def insertHTMLStructure(self, text, repldict):
# XXX This is a bit of a hack! from HTMLTALParser import HTMLTALParser
text = '<!DOCTYPE foo PUBLIC "foo" "bar"><foo>%s</foo>' % text gen = AltTALGenerator(repldict, self.engine, 0)
p = XMLParser() p = HTMLTALParser(gen) # Raises an exception if text is invalid
p.parseString(text) p.parseString(text)
# If this succeeds without errors, we're fine program, macros = p.getCode()
self.interpret(program)
def insertXMLStructure(self, text, repldict):
from TALParser import TALParser
gen = AltTALGenerator(repldict, self.engine, 0)
p = TALParser(gen)
gen.enable(0)
p.parseFragment('<!DOCTYPE foo PUBLIC "foo" "bar"><foo>')
gen.enable(1)
p.parseFragment(text) # Raises an exception if text is invalid
gen.enable(0)
p.parseFragment('</foo>', 1)
program, macros = gen.getCode()
self.interpret(program)
def do_loop(self, name, expr, block): def do_loop(self, name, expr, block):
if not self.tal: if not self.tal:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment