5 from django.template import Library, Node, Variable, VariableDoesNotExist, TemplateSyntaxError
6 from django.conf import settings
7 from django.utils.encoding import force_unicode
10 from filebrowser.fb_settings import MEDIA_ROOT, MEDIA_URL, VERSIONS
11 from filebrowser.functions import _url_to_path, _path_to_url, _get_version_path, _version_generator
12 from filebrowser.base import FileObject
15 logger = logging.getLogger("django.filebrowser")
20 class VersionNode(Node):
21 def __init__(self, src, version_prefix):
22 self.src = Variable(src)
23 if (version_prefix[0] == version_prefix[-1] and version_prefix[0] in ('"', "'")):
24 self.version_prefix = version_prefix[1:-1]
26 self.version_prefix = None
27 self.version_prefix_var = Variable(version_prefix)
29 def render(self, context):
31 source = self.src.resolve(context)
32 except VariableDoesNotExist:
35 if self.version_prefix:
36 version_prefix = self.version_prefix
39 version_prefix = self.version_prefix_var.resolve(context)
40 except VariableDoesNotExist:
44 version_path = _get_version_path(_url_to_path(str(source)), version_prefix)
45 if not os.path.isfile(os.path.join(MEDIA_ROOT, version_path)):
47 version_path = _version_generator(_url_to_path(str(source)), version_prefix)
48 elif os.path.getmtime(os.path.join(MEDIA_ROOT, _url_to_path(str(source)))) > os.path.getmtime(os.path.join(MEDIA_ROOT, version_path)):
49 # recreate version if original image was updated
50 version_path = _version_generator(_url_to_path(str(source)), version_prefix, force=True)
51 return _path_to_url(version_path)
53 logger.exception("Version error")
57 def version(parser, token):
59 Displaying a version of an existing Image according to the predefined VERSIONS settings (see fb_settings).
60 {% version field_name version_prefix %}
62 Use {% version my_image 'medium' %} in order to display the medium-size
63 version of an Image stored in a field name my_image.
65 version_prefix can be a string or a variable. if version_prefix is a string, use quotes.
69 tag, src, version_prefix = token.split_contents()
71 raise TemplateSyntaxError("%s tag requires 2 arguments" % token.contents.split()[0])
72 if (version_prefix[0] == version_prefix[-1] and version_prefix[0] in ('"', "'")) and version_prefix.lower()[1:-1] not in VERSIONS:
73 raise TemplateSyntaxError("%s tag received bad version_prefix %s" % (tag, version_prefix))
74 return VersionNode(src, version_prefix)
77 class VersionObjectNode(Node):
79 def __init__(self, src, version_prefix, var_name):
80 self.var_name = var_name
81 self.src = Variable(src)
82 if (version_prefix[0] == version_prefix[-1] and version_prefix[0] in ('"', "'")):
83 self.version_prefix = version_prefix[1:-1]
85 self.version_prefix = None
86 self.version_prefix_var = Variable(version_prefix)
88 def render(self, context):
90 source = self.src.resolve(context)
91 except VariableDoesNotExist:
93 if self.version_prefix:
94 version_prefix = self.version_prefix
97 version_prefix = self.version_prefix_var.resolve(context)
98 except VariableDoesNotExist:
101 version_path = _get_version_path(_url_to_path(str(source)), version_prefix)
102 if not os.path.isfile(os.path.join(MEDIA_ROOT, version_path)):
104 version_path = _version_generator(_url_to_path(str(source)), version_prefix)
105 elif os.path.getmtime(os.path.join(MEDIA_ROOT, _url_to_path(str(source)))) > os.path.getmtime(os.path.join(MEDIA_ROOT, version_path)):
106 # recreate version if original image was updated
107 version_path = _version_generator(_url_to_path(str(source)), version_prefix, force=True)
108 context[self.var_name] = FileObject(version_path)
110 context[self.var_name] = ""
114 def version_object(parser, token):
116 Returns a context variable 'version_object'.
117 {% version_object field_name version_prefix %}
119 Use {% version_object my_image 'medium' %} in order to retrieve the medium
120 version of an Image stored in a field name my_image.
121 Use {% version_object my_image 'medium' as var %} in order to use 'var' as
122 your context variable.
124 version_prefix can be a string or a variable. if version_prefix is a string, use quotes.
128 #tag, src, version_prefix = token.split_contents()
129 tag, arg = token.contents.split(None, 1)
131 raise TemplateSyntaxError("%s tag requires arguments" % token.contents.split()[0])
132 m = re.search(r'(.*?) (.*?) as (\w+)', arg)
134 raise TemplateSyntaxError("%r tag had invalid arguments" % tag)
135 src, version_prefix, var_name = m.groups()
136 if (version_prefix[0] == version_prefix[-1] and version_prefix[0] in ('"', "'")) and version_prefix.lower()[1:-1] not in VERSIONS:
137 raise TemplateSyntaxError("%s tag received bad version_prefix %s" % (tag, version_prefix))
138 return VersionObjectNode(src, version_prefix, var_name)
141 class VersionSettingNode(Node):
143 def __init__(self, version_prefix):
144 if (version_prefix[0] == version_prefix[-1] and version_prefix[0] in ('"', "'")):
145 self.version_prefix = version_prefix[1:-1]
147 self.version_prefix = None
148 self.version_prefix_var = Variable(version_prefix)
150 def render(self, context):
151 if self.version_prefix:
152 version_prefix = self.version_prefix
155 version_prefix = self.version_prefix_var.resolve(context)
156 except VariableDoesNotExist:
158 context['version_setting'] = VERSIONS[version_prefix]
162 def version_setting(parser, token):
164 Get Information about a version setting.
168 tag, version_prefix = token.split_contents()
170 raise TemplateSyntaxError("%s tag requires 1 argument" % token.contents.split()[0])
171 if (version_prefix[0] == version_prefix[-1] and version_prefix[0] in ('"', "'")) and version_prefix.lower()[1:-1] not in VERSIONS:
172 raise TemplateSyntaxError("%s tag received bad version_prefix %s" % (tag, version_prefix))
173 return VersionSettingNode(version_prefix)
176 register.tag(version)
177 register.tag(version_object)
178 register.tag(version_setting)