1
"""Support for Zope Page Templates using the simpletal library."""
1
"TurboGears support for Zope Page Templates"
5
from template import PageTemplate
8
from simpletal import simpleTAL, simpleTALES
10
logging.getLogger("simpleTAL").setLevel(logging.INFO)
11
logging.getLogger("simpleTALES").setLevel(logging.INFO)
16
tinstance = _zpt_cache.get(tfile)
18
if tinstance is None or tinstance.stat != stat:
19
tinstance = _zpt_cache[tfile] = TemplateWrapper(
20
simpleTAL.compileXMLTemplate(open(tfile)), tfile, stat)
24
class TemplateWrapper(object):
26
def __init__(self, template, filename, stat):
27
self.template = template
28
self.filename = filename
31
def expand(self, **info):
32
context = simpleTALES.Context(allowPythonPath=1)
33
for k, v in info.iteritems():
34
context.addGlobal(k, v)
35
s = StringIO.StringIO()
36
self.template.expandInline(context, s)
39
def expand_into(self, f, **info):
40
context = simpleTALES.Context(allowPythonPath=1)
41
for k, v in info.iteritems():
42
context.addGlobal(k, v)
43
self.template.expand(context, f, 'utf-8')
47
return self.template.macros
50
def load_template(classname):
51
"""Searches for a template along the Python path.
53
Template files must end in ".pt" and be in legitimate packages.
54
Templates are automatically checked for changes and reloaded as
57
divider = classname.rfind(".")
59
package = classname[0:divider]
60
basename = classname[divider+1:]
62
raise ValueError, "All templates must be in a package"
64
tfile = pkg_resources.resource_filename(
65
package, "%s.%s" % (basename, "pt"))
9
log = logging.getLogger("turbogears.zptsupport")
11
def _recompile_template(package, basename, tfile, classname):
12
log.debug("Recompiling template for %s" % classname)
13
mod = PageTemplate(tfile)
14
mtime = os.stat(tfile).st_mtime
21
def __init__(self, extra_vars_func=None, options={}):
22
self.options = options
23
self.get_extra_vars = extra_vars_func
24
self.compiledTemplates = {}
26
def load_template(self, classname, loadingSite=False):
27
"""Searches for a template along the Python path.
29
Template files must end in ".pt" and be in legitimate packages.
30
U can set "zpt.cache_templates" option to cache a loaded template
31
class and only check for updates. Templates are automatically
32
checked for changes and reloaded as neccessary.
34
ct = self.compiledTemplates
36
divider = classname.rfind(".")
38
package = classname[0:divider]
39
basename = classname[divider+1:]
41
raise ValueError, "All templates must be in a package"
43
cache_templates = self.options.get("zpt.cache_templates", True)
44
tfile = pkg_resources.resource_filename(package,
50
if ct.has_key(classname):
51
mtime = os.stat(tfile).st_mtime
53
if mod.__mtime__ != mtime:
55
mod = _recompile_template(package, basename, tfile, classname)
58
# use template from cache
61
# First time compile template
62
mod = PageTemplate(tfile)
63
mod.__mtime__ = os.stat(tfile).st_mtime
66
mod = PageTemplate(tfile)
70
def render(self, info, format="html", fragment=False, template=None):
71
"""Renders data in the desired format.
73
@param info: the data / context itself
76
@type format: "string"
77
@para template: name of the template to use
78
@type template: string
80
#if isinstance(template, type):
84
tinstance = self.load_template(template)
85
log.debug("Applying template %s" % (tinstance.filename))
87
if self.get_extra_vars:
88
data.update(self.get_extra_vars())
90
return tinstance(**data).encode('utf-8')
92
def transform(self, info, template):
93
"Render the output to Elements"