~loggerhead-team/loggerhead/trunk-rich

« back to all changes in this revision

Viewing changes to turbosimpletal/zptsupport.py

 * Make navigation links use revnos instead of revids
 * Don't always pass on start_revid. Why would we do that?

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
#
2
 
# This program is free software; you can redistribute it and/or modify
3
 
# it under the terms of the GNU General Public License as published by
4
 
# the Free Software Foundation; either version 2 of the License, or
5
 
# (at your option) any later version.
6
 
#
7
 
# This program is distributed in the hope that it will be useful,
8
 
# but WITHOUT ANY WARRANTY; without even the implied warranty of
9
 
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
10
 
# GNU General Public License for more details.
11
 
#
12
 
# You should have received a copy of the GNU General Public License
13
 
# along with this program; if not, write to the Free Software
14
 
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
15
 
#
16
 
"""Support for Zope Page Templates using the simpletal library."""
 
1
"TurboGears support for Zope Page Templates"
17
2
 
 
3
import StringIO
18
4
import logging
19
5
import os
20
6
import pkg_resources
21
 
import re
22
 
import StringIO
23
7
 
24
8
from simpletal import simpleTAL, simpleTALES
25
9
 
26
 
logging.getLogger("simpleTAL").setLevel(logging.INFO)
27
 
logging.getLogger("simpleTALES").setLevel(logging.INFO)
 
10
log = logging.getLogger("turbogears.zptsupport")
28
11
 
29
12
 
30
13
_zpt_cache = {}
31
 
 
32
 
 
33
14
def zpt(tfile):
34
15
    tinstance = _zpt_cache.get(tfile)
35
16
    stat = os.stat(tfile)
36
17
    if tinstance is None or tinstance.stat != stat:
37
 
        text = open(tfile).read()
38
 
        text = re.sub(r'\s*\n\s*', '\n', text)
39
 
        text = re.sub(r'[ \t]+', ' ', text)
40
18
        tinstance = _zpt_cache[tfile] = TemplateWrapper(
41
 
            simpleTAL.compileXMLTemplate(text), tfile, stat)
 
19
            simpleTAL.compileXMLTemplate(open(tfile)), tfile, stat)
42
20
    return tinstance
43
21
 
44
22
 
57
35
        self.template.expandInline(context, s)
58
36
        return s.getvalue()
59
37
 
60
 
    def expand_into(self, f, **info):
61
 
        context = simpleTALES.Context(allowPythonPath=1)
62
 
        for k, v in info.iteritems():
63
 
            context.addGlobal(k, v)
64
 
        self.template.expand(context, f, 'utf-8')
65
 
 
66
38
    @property
67
39
    def macros(self):
68
40
        return self.template.macros
69
41
 
70
42
 
71
 
def load_template(classname):
72
 
    """Searches for a template along the Python path.
73
 
 
74
 
    Template files must end in ".pt" and be in legitimate packages.
75
 
    Templates are automatically checked for changes and reloaded as
76
 
    neccessary.
77
 
    """
78
 
    divider = classname.rfind(".")
79
 
    if divider > -1:
80
 
        package = classname[0:divider]
81
 
        basename = classname[divider+1:]
82
 
    else:
83
 
        raise ValueError("All templates must be in a package")
84
 
 
85
 
    tfile = pkg_resources.resource_filename(
86
 
        package, "%s.%s" % (basename, "pt"))
87
 
    return zpt(tfile)
 
43
class TurboZpt(object):
 
44
    extension = "pt"
 
45
 
 
46
    def __init__(self, extra_vars_func=None):
 
47
        self.get_extra_vars = extra_vars_func
 
48
 
 
49
    def load_template(self, classname, loadingSite=False):
 
50
        """Searches for a template along the Python path.
 
51
 
 
52
        Template files must end in ".pt" and be in legitimate packages.
 
53
        Templates are automatically checked for changes and reloaded as
 
54
        neccessary.
 
55
        """
 
56
        divider = classname.rfind(".")
 
57
        if divider > -1:
 
58
            package = classname[0:divider]
 
59
            basename = classname[divider+1:]
 
60
        else:
 
61
            raise ValueError, "All templates must be in a package"
 
62
 
 
63
        tfile = pkg_resources.resource_filename(
 
64
            package, "%s.%s" % (basename, self.extension))
 
65
        return zpt(tfile)
 
66
 
 
67
    def render(self, info, format="html", fragment=False, template=None):
 
68
        """Renders data in the desired format.
 
69
 
 
70
        @param info: the data / context itself
 
71
        @type info: dict
 
72
        @para format: "html"
 
73
        @type format: "string"
 
74
        @para template: name of the template to use
 
75
        @type template: string
 
76
        """
 
77
        tinstance = self.load_template(template)
 
78
        log.debug("Applying template %s" % (tinstance.filename))
 
79
        data = dict()
 
80
        if self.get_extra_vars:
 
81
            data.update(self.get_extra_vars())
 
82
        data.update(info)
 
83
        return tinstance.expand(**data).encode('utf-8')