mirror of
				https://github.com/django/django.git
				synced 2025-10-25 22:56:12 +00:00 
			
		
		
		
	Fixed #911 -- Made template system scoped to the parser instead of the template module. Also changed the way tags/filters are registered and added support for multiple arguments to {% load %} tag. Thanks, rjwittams. This is a backwards-incompatible change for people who've created custom template tags or filters. See http://code.djangoproject.com/wiki/BackwardsIncompatibleChanges for upgrade instructions.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@1443 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -1,7 +1,5 @@ | |||||||
| {% extends "admin/base_site" %} | {% extends "admin/base_site" %} | ||||||
| {% load i18n %} | {% load i18n admin_modify adminmedia %} | ||||||
| {% load admin_modify %} |  | ||||||
| {% load adminmedia %} |  | ||||||
| {% block extrahead %} | {% block extrahead %} | ||||||
| {% for js in bound_manipulator.javascript_imports %}{% include_admin_script js %}{% endfor %} | {% for js in bound_manipulator.javascript_imports %}{% include_admin_script js %}{% endfor %} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|   | |||||||
| @@ -1,5 +1,4 @@ | |||||||
| {% load admin_list %} | {% load adminmedia admin_list i18n %} | ||||||
| {% load i18n %} |  | ||||||
| {% extends "admin/base_site" %} | {% extends "admin/base_site" %} | ||||||
| {% block bodyclass %}change-list{% endblock %} | {% block bodyclass %}change-list{% endblock %} | ||||||
| {% if not is_popup %}{% block breadcrumbs %}<div class="breadcrumbs"><a href="../../">{% trans "Home" %}</a> › {{ cl.opts.verbose_name_plural|capfirst }} </div>{% endblock %}{% endif %} | {% if not is_popup %}{% block breadcrumbs %}<div class="breadcrumbs"><a href="../../">{% trans "Home" %}</a> › {{ cl.opts.verbose_name_plural|capfirst }} </div>{% endblock %}{% endif %} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load admin_modify %} | ||||||
| <fieldset class="module aligned"> | <fieldset class="module aligned"> | ||||||
|    {% for fcw in bound_related_object.form_field_collection_wrappers %} |    {% for fcw in bound_related_object.form_field_collection_wrappers %} | ||||||
|       <h2>{{ bound_related_object.relation.opts.verbose_name|capfirst }} #{{ forloop.counter }}</h2> |       <h2>{{ bound_related_object.relation.opts.verbose_name|capfirst }} #{{ forloop.counter }}</h2> | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load admin_modify %} | ||||||
| <fieldset class="module"> | <fieldset class="module"> | ||||||
|    <h2>{{ bound_related_object.relation.opts.verbose_name_plural|capfirst }}</h2><table> |    <h2>{{ bound_related_object.relation.opts.verbose_name_plural|capfirst }}</h2><table> | ||||||
|    <thead><tr> |    <thead><tr> | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load admin_modify %} | ||||||
| <div class="{{ class_names }}" > | <div class="{{ class_names }}" > | ||||||
| {% for bound_field in bound_fields %}{{ bound_field.html_error_list }}{% endfor %} | {% for bound_field in bound_fields %}{{ bound_field.html_error_list }}{% endfor %} | ||||||
| {% for bound_field in bound_fields %} | {% for bound_field in bound_fields %} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load i18n %} | ||||||
| <h3>{% blocktrans %} By {{ title }} {% endblocktrans %}</h3> | <h3>{% blocktrans %} By {{ title }} {% endblocktrans %}</h3> | ||||||
| <ul> | <ul> | ||||||
| {% for choice in choices %} | {% for choice in choices %} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load admin_list %} | ||||||
| {% if cl.has_filters %}<div id="changelist-filter"> | {% if cl.has_filters %}<div id="changelist-filter"> | ||||||
| <h2>Filter</h2> | <h2>Filter</h2> | ||||||
| {% for spec in cl.filter_specs %} | {% for spec in cl.filter_specs %} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load admin_list %} | ||||||
| <p class="paginator"> | <p class="paginator"> | ||||||
| {% if pagination_required %} | {% if pagination_required %} | ||||||
| {% for i in page_range %} | {% for i in page_range %} | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | {% load adminmedia %} | ||||||
| {% if cl.lookup_opts.admin.search_fields %} | {% if cl.lookup_opts.admin.search_fields %} | ||||||
| <div id="toolbar"><form id="changelist-search" action="" method="get"> | <div id="toolbar"><form id="changelist-search" action="" method="get"> | ||||||
| <label><img src="{% admin_media_prefix %}img/admin/icon_searchbox.png" /></label> | <label><img src="{% admin_media_prefix %}img/admin/icon_searchbox.png" /></label> | ||||||
|   | |||||||
| @@ -1 +1 @@ | |||||||
| {% output_all bound_field.form_fields %} | {% load admin_modify %}{% output_all bound_field.form_fields %} | ||||||
|   | |||||||
| @@ -1,4 +1,4 @@ | |||||||
| {% if bound_field.original_value %} | {% load admin_modify %}{% if bound_field.original_value %} | ||||||
| Currently: <a href="{{ bound_field.original_url }}" > {{ bound_field.original_value }} </a><br /> | Currently: <a href="{{ bound_field.original_url }}" > {{ bound_field.original_value }} </a><br /> | ||||||
| Change: {% output_all bound_field.form_fields %} | Change: {% output_all bound_field.form_fields %} | ||||||
| {% else %} {% output_all bound_field.form_fields %} {% endif %} | {% else %} {% output_all bound_field.form_fields %} {% endif %} | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
|  | {% load admin_modify adminmedia %} | ||||||
| {% output_all bound_field.form_fields %} | {% output_all bound_field.form_fields %} | ||||||
| {% if bound_field.raw_id_admin %} | {% if bound_field.raw_id_admin %} | ||||||
|             <a href="../../../{{ bound_field.field.rel.to.app_label }}/{{ bound_field.field.rel.to.module_name }}/" class="related-lookup" id="lookup_{{bound_field.element_id}}" onclick="return showRelatedObjectLookupPopup(this);"> <img src="{% admin_media_prefix %}img/admin/selector-search.gif" width="16" height="16" alt="Lookup"></a> |             <a href="../../../{{ bound_field.field.rel.to.app_label }}/{{ bound_field.field.rel.to.module_name }}/" class="related-lookup" id="lookup_{{bound_field.element_id}}" onclick="return showRelatedObjectLookupPopup(this);"> <img src="{% admin_media_prefix %}img/admin/selector-search.gif" width="16" height="16" alt="Lookup"></a> | ||||||
| {% else %} | {% else %} | ||||||
| {% if bound_field.needs_add_label %} | {% if bound_field.needs_add_label %} | ||||||
|             <a href="../../../{{ bound_field.field.rel.to.app_label }}/{{ bound_field.field.rel.to.module_name }}/add/" class="add-another" id="add_{{ bound_field.element_id}}" onclick="return showAddAnotherPopup(this);"> <img src="{% admin_media_prefix %}img/admin/icon_addlink.gif" width="10" height="10" alt="Add Another"/></a> |             <a href="../../../{{ bound_field.field.rel.to.app_label }}/{{ bound_field.field.rel.to.module_name }}/add/" class="add-another" id="add_{{ bound_field.element_id}}" onclick="return showAddAnotherPopup(this);"> <img src="{% admin_media_prefix %}img/admin/icon_addlink.gif" width="10" height="10" alt="Add Another"/></a> | ||||||
| {% endif %} {% endif %} | {% endif %}{% endif %} | ||||||
|   | |||||||
| @@ -3,16 +3,18 @@ from django.contrib.admin.views.main import ORDER_VAR, ORDER_TYPE_VAR, PAGE_VAR, | |||||||
| from django.contrib.admin.views.main import IS_POPUP_VAR, EMPTY_CHANGELIST_VALUE, MONTHS | from django.contrib.admin.views.main import IS_POPUP_VAR, EMPTY_CHANGELIST_VALUE, MONTHS | ||||||
| from django.core import meta, template | from django.core import meta, template | ||||||
| from django.core.exceptions import ObjectDoesNotExist | from django.core.exceptions import ObjectDoesNotExist | ||||||
| from django.core.template.decorators import simple_tag, inclusion_tag |  | ||||||
| from django.utils import dateformat | from django.utils import dateformat | ||||||
| from django.utils.html import strip_tags, escape | from django.utils.html import strip_tags, escape | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.translation import get_date_formats | from django.utils.translation import get_date_formats | ||||||
| from django.conf.settings import ADMIN_MEDIA_PREFIX | from django.conf.settings import ADMIN_MEDIA_PREFIX | ||||||
|  | from django.core.template import Library | ||||||
|  |  | ||||||
|  | register = Library() | ||||||
|  |  | ||||||
| DOT = '.' | DOT = '.' | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def paginator_number(cl,i): | def paginator_number(cl,i): | ||||||
|     if i == DOT: |     if i == DOT: | ||||||
|        return '... ' |        return '... ' | ||||||
| @@ -20,9 +22,9 @@ def paginator_number(cl,i): | |||||||
|        return '<span class="this-page">%d</span> ' % (i+1) |        return '<span class="this-page">%d</span> ' % (i+1) | ||||||
|     else: |     else: | ||||||
|        return '<a href="%s"%s>%d</a> ' % (cl.get_query_string({PAGE_VAR: i}), (i == cl.paginator.pages-1 and ' class="end"' or ''), i+1) |        return '<a href="%s"%s>%d</a> ' % (cl.get_query_string({PAGE_VAR: i}), (i == cl.paginator.pages-1 and ' class="end"' or ''), i+1) | ||||||
| paginator_number = simple_tag(paginator_number) | paginator_number = register.simple_tag(paginator_number) | ||||||
|  |  | ||||||
| #@inclusion_tag('admin/pagination') | #@register.inclusion_tag('admin/pagination') | ||||||
| def pagination(cl): | def pagination(cl): | ||||||
|     paginator, page_num = cl.paginator, cl.page_num |     paginator, page_num = cl.paginator, cl.page_num | ||||||
|  |  | ||||||
| @@ -64,7 +66,7 @@ def pagination(cl): | |||||||
|         'ALL_VAR': ALL_VAR, |         'ALL_VAR': ALL_VAR, | ||||||
|         '1': 1, |         '1': 1, | ||||||
|     } |     } | ||||||
| pagination = inclusion_tag('admin/pagination')(pagination) | pagination = register.inclusion_tag('admin/pagination')(pagination) | ||||||
|  |  | ||||||
| def result_headers(cl): | def result_headers(cl): | ||||||
|     lookup_opts = cl.lookup_opts |     lookup_opts = cl.lookup_opts | ||||||
| @@ -177,15 +179,15 @@ def results(cl): | |||||||
|     for res in cl.result_list: |     for res in cl.result_list: | ||||||
|         yield list(items_for_result(cl,res)) |         yield list(items_for_result(cl,res)) | ||||||
|  |  | ||||||
| #@inclusion_tag("admin/change_list_results") | #@register.inclusion_tag("admin/change_list_results") | ||||||
| def result_list(cl): | def result_list(cl): | ||||||
|     res = list(results(cl)) |     res = list(results(cl)) | ||||||
|     return {'cl': cl, |     return {'cl': cl, | ||||||
|             'result_headers': list(result_headers(cl)), |             'result_headers': list(result_headers(cl)), | ||||||
|             'results': list(results(cl))} |             'results': list(results(cl))} | ||||||
| result_list = inclusion_tag("admin/change_list_results")(result_list) | result_list = register.inclusion_tag("admin/change_list_results")(result_list) | ||||||
|  |  | ||||||
| #@inclusion_tag("admin/date_hierarchy") | #@register.inclusion_tag("admin/date_hierarchy") | ||||||
| def date_hierarchy(cl): | def date_hierarchy(cl): | ||||||
|     lookup_opts, params, lookup_params, lookup_mod = \ |     lookup_opts, params, lookup_params, lookup_mod = \ | ||||||
|       cl.lookup_opts, cl.params, cl.lookup_params, cl.lookup_mod |       cl.lookup_opts, cl.params, cl.lookup_params, cl.lookup_mod | ||||||
| @@ -256,23 +258,23 @@ def date_hierarchy(cl): | |||||||
|                     'title': year.year |                     'title': year.year | ||||||
|                 } for year in years ] |                 } for year in years ] | ||||||
|             } |             } | ||||||
| date_hierarchy = inclusion_tag('admin/date_hierarchy')(date_hierarchy) | date_hierarchy = register.inclusion_tag('admin/date_hierarchy')(date_hierarchy) | ||||||
|  |  | ||||||
| #@inclusion_tag('admin/search_form') | #@register.inclusion_tag('admin/search_form') | ||||||
| def search_form(cl): | def search_form(cl): | ||||||
|     return { |     return { | ||||||
|         'cl': cl, |         'cl': cl, | ||||||
|         'show_result_count': cl.result_count != cl.full_result_count and not cl.opts.one_to_one_field, |         'show_result_count': cl.result_count != cl.full_result_count and not cl.opts.one_to_one_field, | ||||||
|         'search_var': SEARCH_VAR |         'search_var': SEARCH_VAR | ||||||
|     } |     } | ||||||
| search_form = inclusion_tag('admin/search_form')(search_form) | search_form = register.inclusion_tag('admin/search_form')(search_form) | ||||||
|  |  | ||||||
| #@inclusion_tag('admin/filter') | #@register.inclusion_tag('admin/filter') | ||||||
| def filter(cl, spec): | def filter(cl, spec): | ||||||
|     return {'title': spec.title(), 'choices' : list(spec.choices(cl))} |     return {'title': spec.title(), 'choices' : list(spec.choices(cl))} | ||||||
| filter = inclusion_tag('admin/filter')(filter) | filter = register.inclusion_tag('admin/filter')(filter) | ||||||
|  |  | ||||||
| #@inclusion_tag('admin/filters') | #@register.inclusion_tag('admin/filters') | ||||||
| def filters(cl): | def filters(cl): | ||||||
|     return {'cl': cl} |     return {'cl': cl} | ||||||
| filters = inclusion_tag('admin/filters')(filters) | filters = register.inclusion_tag('admin/filters')(filters) | ||||||
|   | |||||||
| @@ -2,24 +2,25 @@ from django.core import template, template_loader, meta | |||||||
| from django.utils.html import escape | from django.utils.html import escape | ||||||
| from django.utils.text import capfirst | from django.utils.text import capfirst | ||||||
| from django.utils.functional import curry | from django.utils.functional import curry | ||||||
| from django.core.template.decorators import simple_tag, inclusion_tag |  | ||||||
| from django.contrib.admin.views.main import AdminBoundField | from django.contrib.admin.views.main import AdminBoundField | ||||||
| from django.core.meta.fields import BoundField, Field | from django.core.meta.fields import BoundField, Field | ||||||
| from django.core.meta import BoundRelatedObject, TABULAR, STACKED | from django.core.meta import BoundRelatedObject, TABULAR, STACKED | ||||||
| from django.conf.settings import ADMIN_MEDIA_PREFIX | from django.conf.settings import ADMIN_MEDIA_PREFIX | ||||||
| import re | import re | ||||||
|  |  | ||||||
|  | register = template.Library() | ||||||
|  |  | ||||||
| word_re = re.compile('[A-Z][a-z]+') | word_re = re.compile('[A-Z][a-z]+') | ||||||
|  |  | ||||||
| def class_name_to_underscored(name): | def class_name_to_underscored(name): | ||||||
|     return '_'.join([s.lower() for s in word_re.findall(name)[:-1]]) |     return '_'.join([s.lower() for s in word_re.findall(name)[:-1]]) | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def include_admin_script(script_path): | def include_admin_script(script_path): | ||||||
|     return '<script type="text/javascript" src="%s%s"></script>' % (ADMIN_MEDIA_PREFIX, script_path) |     return '<script type="text/javascript" src="%s%s"></script>' % (ADMIN_MEDIA_PREFIX, script_path) | ||||||
| include_admin_script = simple_tag(include_admin_script) | include_admin_script = register.simple_tag(include_admin_script) | ||||||
|  |  | ||||||
| #@inclusion_tag('admin/submit_line', takes_context=True) | #@register.inclusion_tag('admin/submit_line', takes_context=True) | ||||||
| def submit_row(context, bound_manipulator): | def submit_row(context, bound_manipulator): | ||||||
|     change = context['change'] |     change = context['change'] | ||||||
|     add = context['add'] |     add = context['add'] | ||||||
| @@ -36,9 +37,9 @@ def submit_row(context, bound_manipulator): | |||||||
|         'show_save_and_continue': not is_popup, |         'show_save_and_continue': not is_popup, | ||||||
|         'show_save': True |         'show_save': True | ||||||
|     } |     } | ||||||
| submit_row = inclusion_tag('admin/submit_line', takes_context=True)(submit_row) | submit_row = register.inclusion_tag('admin/submit_line', takes_context=True)(submit_row) | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def field_label(bound_field): | def field_label(bound_field): | ||||||
|     class_names = [] |     class_names = [] | ||||||
|     if isinstance(bound_field.field, meta.BooleanField): |     if isinstance(bound_field.field, meta.BooleanField): | ||||||
| @@ -53,7 +54,7 @@ def field_label(bound_field): | |||||||
|     class_str = class_names and ' class="%s"' % ' '.join(class_names) or '' |     class_str = class_names and ' class="%s"' % ' '.join(class_names) or '' | ||||||
|     return '<label for="%s"%s>%s%s</label> ' % (bound_field.element_id, class_str, \ |     return '<label for="%s"%s>%s%s</label> ' % (bound_field.element_id, class_str, \ | ||||||
|         capfirst(bound_field.field.verbose_name), colon) |         capfirst(bound_field.field.verbose_name), colon) | ||||||
| field_label = simple_tag(field_label) | field_label = register.simple_tag(field_label) | ||||||
|  |  | ||||||
| class FieldWidgetNode(template.Node): | class FieldWidgetNode(template.Node): | ||||||
|     nodelists = {} |     nodelists = {} | ||||||
| @@ -170,12 +171,12 @@ class EditInlineNode(template.Node): | |||||||
|         context.pop() |         context.pop() | ||||||
|         return output |         return output | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def output_all(form_fields): | def output_all(form_fields): | ||||||
|     return ''.join([str(f) for f in form_fields]) |     return ''.join([str(f) for f in form_fields]) | ||||||
| output_all = simple_tag(output_all) | output_all = register.simple_tag(output_all) | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def auto_populated_field_script(auto_pop_fields, change = False): | def auto_populated_field_script(auto_pop_fields, change = False): | ||||||
|     for field in auto_pop_fields: |     for field in auto_pop_fields: | ||||||
|         t = [] |         t = [] | ||||||
| @@ -191,9 +192,9 @@ def auto_populated_field_script(auto_pop_fields, change = False): | |||||||
|                      ' if(!e._changed) { e.value = URLify(%s, %s);} }; ' % ( |                      ' if(!e._changed) { e.value = URLify(%s, %s);} }; ' % ( | ||||||
|                      f, field.name, add_values, field.maxlength)) |                      f, field.name, add_values, field.maxlength)) | ||||||
|     return ''.join(t) |     return ''.join(t) | ||||||
| auto_populated_field_script = simple_tag(auto_populated_field_script) | auto_populated_field_script = register.simple_tag(auto_populated_field_script) | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def filter_interface_script_maybe(bound_field): | def filter_interface_script_maybe(bound_field): | ||||||
|     f = bound_field.field |     f = bound_field.field | ||||||
|     if f.rel and isinstance(f.rel, meta.ManyToMany) and f.rel.filter_interface: |     if f.rel and isinstance(f.rel, meta.ManyToMany) and f.rel.filter_interface: | ||||||
| @@ -202,7 +203,7 @@ def filter_interface_script_maybe(bound_field): | |||||||
|               f.name, f.verbose_name, f.rel.filter_interface-1, ADMIN_MEDIA_PREFIX) |               f.name, f.verbose_name, f.rel.filter_interface-1, ADMIN_MEDIA_PREFIX) | ||||||
|     else: |     else: | ||||||
|         return '' |         return '' | ||||||
| filter_interface_script_maybe = simple_tag(filter_interface_script_maybe) | filter_interface_script_maybe = register.simple_tag(filter_interface_script_maybe) | ||||||
|  |  | ||||||
| def do_one_arg_tag(node_factory, parser,token): | def do_one_arg_tag(node_factory, parser,token): | ||||||
|     tokens = token.contents.split() |     tokens = token.contents.split() | ||||||
| @@ -213,7 +214,7 @@ def do_one_arg_tag(node_factory, parser,token): | |||||||
| def register_one_arg_tag(node): | def register_one_arg_tag(node): | ||||||
|     tag_name = class_name_to_underscored(node.__name__) |     tag_name = class_name_to_underscored(node.__name__) | ||||||
|     parse_func = curry(do_one_arg_tag, node) |     parse_func = curry(do_one_arg_tag, node) | ||||||
|     template.register_tag(tag_name, parse_func) |     register.tag(tag_name, parse_func) | ||||||
|  |  | ||||||
| one_arg_tag_nodes = ( | one_arg_tag_nodes = ( | ||||||
|     FieldWidgetNode, |     FieldWidgetNode, | ||||||
| @@ -223,7 +224,7 @@ one_arg_tag_nodes = ( | |||||||
| for node in one_arg_tag_nodes: | for node in one_arg_tag_nodes: | ||||||
|     register_one_arg_tag(node) |     register_one_arg_tag(node) | ||||||
|  |  | ||||||
| #@inclusion_tag('admin/field_line', takes_context=True) | #@register.inclusion_tag('admin/field_line', takes_context=True) | ||||||
| def admin_field_line(context, argument_val): | def admin_field_line(context, argument_val): | ||||||
|     if (isinstance(argument_val, BoundField)): |     if (isinstance(argument_val, BoundField)): | ||||||
|         bound_fields = [argument_val] |         bound_fields = [argument_val] | ||||||
| @@ -249,10 +250,10 @@ def admin_field_line(context, argument_val): | |||||||
|         'bound_fields':  bound_fields, |         'bound_fields':  bound_fields, | ||||||
|         'class_names': " ".join(class_names), |         'class_names': " ".join(class_names), | ||||||
|     } |     } | ||||||
| admin_field_line = inclusion_tag('admin/field_line', takes_context=True)(admin_field_line) | admin_field_line = register.inclusion_tag('admin/field_line', takes_context=True)(admin_field_line) | ||||||
|  |  | ||||||
| #@simple_tag | #@register.simple_tag | ||||||
| def object_pk(bound_manip, ordered_obj): | def object_pk(bound_manip, ordered_obj): | ||||||
|     return bound_manip.get_ordered_object_pk(ordered_obj) |     return bound_manip.get_ordered_object_pk(ordered_obj) | ||||||
|  |  | ||||||
| object_pk = simple_tag(object_pk) | object_pk = register.simple_tag(object_pk) | ||||||
|   | |||||||
| @@ -1,5 +1,7 @@ | |||||||
| from django.core import template | from django.core import template | ||||||
|  |  | ||||||
|  | register = template.Library() | ||||||
|  |  | ||||||
| class AdminApplistNode(template.Node): | class AdminApplistNode(template.Node): | ||||||
|     def __init__(self, varname): |     def __init__(self, varname): | ||||||
|         self.varname = varname |         self.varname = varname | ||||||
| @@ -54,4 +56,4 @@ def get_admin_app_list(parser, token): | |||||||
|         raise template.TemplateSyntaxError, "First argument to '%s' tag must be 'as'" % tokens[0] |         raise template.TemplateSyntaxError, "First argument to '%s' tag must be 'as'" % tokens[0] | ||||||
|     return AdminApplistNode(tokens[2]) |     return AdminApplistNode(tokens[2]) | ||||||
|  |  | ||||||
| template.register_tag('get_admin_app_list', get_admin_app_list) | register.tag('get_admin_app_list', get_admin_app_list) | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| from django.core.template.decorators import simple_tag | from django.core.template import Library | ||||||
|  | register = Library() | ||||||
|  |  | ||||||
| def admin_media_prefix(): | def admin_media_prefix(): | ||||||
|     try: |     try: | ||||||
| @@ -6,4 +7,4 @@ def admin_media_prefix(): | |||||||
|     except ImportError: |     except ImportError: | ||||||
|         return '' |         return '' | ||||||
|     return ADMIN_MEDIA_PREFIX |     return ADMIN_MEDIA_PREFIX | ||||||
| admin_media_prefix = simple_tag(admin_media_prefix) | admin_media_prefix = register.simple_tag(admin_media_prefix) | ||||||
| @@ -1,6 +1,8 @@ | |||||||
| from django.models.admin import log | from django.models.admin import log | ||||||
| from django.core import template | from django.core import template | ||||||
|  |  | ||||||
|  | register = template.Library() | ||||||
|  |  | ||||||
| class AdminLogNode(template.Node): | class AdminLogNode(template.Node): | ||||||
|     def __init__(self, limit, varname, user): |     def __init__(self, limit, varname, user): | ||||||
|         self.limit, self.varname, self.user = limit, varname, user |         self.limit, self.varname, self.user = limit, varname, user | ||||||
| @@ -48,4 +50,4 @@ class DoGetAdminLog: | |||||||
|                 raise template.TemplateSyntaxError, "Fourth argument in '%s' must be 'for_user'" % self.tag_name |                 raise template.TemplateSyntaxError, "Fourth argument in '%s' must be 'for_user'" % self.tag_name | ||||||
|         return AdminLogNode(limit=tokens[1], varname=tokens[3], user=(len(tokens) > 5 and tokens[5] or None)) |         return AdminLogNode(limit=tokens[1], varname=tokens[3], user=(len(tokens) > 5 and tokens[5] or None)) | ||||||
|  |  | ||||||
| template.register_tag('get_admin_log', DoGetAdminLog('get_admin_log')) | register.tag('get_admin_log', DoGetAdminLog('get_admin_log')) | ||||||
|   | |||||||
| @@ -50,21 +50,23 @@ class TemplateValidator(formfields.Manipulator): | |||||||
|             return |             return | ||||||
|  |  | ||||||
|         # so that inheritance works in the site's context, register a new function |         # so that inheritance works in the site's context, register a new function | ||||||
|         # for "extends" that uses the site's TEMPLATE_DIR instead |         # for "extends" that uses the site's TEMPLATE_DIRS instead. | ||||||
|         def new_do_extends(parser, token): |         def new_do_extends(parser, token): | ||||||
|             node = loader.do_extends(parser, token) |             node = loader.do_extends(parser, token) | ||||||
|             node.template_dirs = settings_module.TEMPLATE_DIRS |             node.template_dirs = settings_module.TEMPLATE_DIRS | ||||||
|             return node |             return node | ||||||
|         template.register_tag('extends', new_do_extends) |         register = template.Library() | ||||||
|  |         register.tag('extends', new_do_extends) | ||||||
|  |         template.builtins.append(register) | ||||||
|  |  | ||||||
|         # now validate the template using the new template dirs |         # Now validate the template using the new template dirs | ||||||
|         # making sure to reset the extends function in any case |         # making sure to reset the extends function in any case. | ||||||
|         error = None |         error = None | ||||||
|         try: |         try: | ||||||
|             tmpl = loader.get_template_from_string(field_data) |             tmpl = loader.get_template_from_string(field_data) | ||||||
|             tmpl.render(template.Context({})) |             tmpl.render(template.Context({})) | ||||||
|         except template.TemplateSyntaxError, e: |         except template.TemplateSyntaxError, e: | ||||||
|             error = e |             error = e | ||||||
|         template.register_tag('extends', loader.do_extends) |         template.builtins.remove(register) | ||||||
|         if error: |         if error: | ||||||
|             raise validators.ValidationError, e.args |             raise validators.ValidationError, e.args | ||||||
|   | |||||||
| @@ -6,6 +6,8 @@ from django.models.comments import comments, freecomments | |||||||
| from django.models.core import contenttypes | from django.models.core import contenttypes | ||||||
| import re | import re | ||||||
|  |  | ||||||
|  | register = template.Library() | ||||||
|  |  | ||||||
| COMMENT_FORM = ''' | COMMENT_FORM = ''' | ||||||
| {% load i18n %} | {% load i18n %} | ||||||
| {% if display_form %} | {% if display_form %} | ||||||
| @@ -360,10 +362,10 @@ class DoGetCommentList: | |||||||
|         return CommentListNode(package, module, var_name, obj_id, tokens[5], self.free, ordering) |         return CommentListNode(package, module, var_name, obj_id, tokens[5], self.free, ordering) | ||||||
|  |  | ||||||
| # registration comments | # registration comments | ||||||
| template.register_tag('get_comment_list', DoGetCommentList(False)) | register.tag('get_comment_list', DoGetCommentList(False)) | ||||||
| template.register_tag('comment_form', DoCommentForm(False)) | register.tag('comment_form', DoCommentForm(False)) | ||||||
| template.register_tag('get_comment_count', DoCommentCount(False)) | register.tag('get_comment_count', DoCommentCount(False)) | ||||||
| # free comments | # free comments | ||||||
| template.register_tag('get_free_comment_list', DoGetCommentList(True)) | register.tag('get_free_comment_list', DoGetCommentList(True)) | ||||||
| template.register_tag('free_comment_form', DoCommentForm(True)) | register.tag('free_comment_form', DoCommentForm(True)) | ||||||
| template.register_tag('get_free_comment_count', DoCommentCount(True)) | register.tag('get_free_comment_count', DoCommentCount(True)) | ||||||
|   | |||||||
| @@ -16,7 +16,9 @@ silently fail and return the un-marked-up text. | |||||||
|  |  | ||||||
| from django.core import template | from django.core import template | ||||||
|  |  | ||||||
| def textile(value, _): | register = template.Library() | ||||||
|  |  | ||||||
|  | def textile(value): | ||||||
|     try: |     try: | ||||||
|         import textile |         import textile | ||||||
|     except ImportError: |     except ImportError: | ||||||
| @@ -24,7 +26,7 @@ def textile(value, _): | |||||||
|     else: |     else: | ||||||
|         return textile.textile(value) |         return textile.textile(value) | ||||||
|  |  | ||||||
| def markdown(value, _): | def markdown(value): | ||||||
|     try: |     try: | ||||||
|         import markdown |         import markdown | ||||||
|     except ImportError: |     except ImportError: | ||||||
| @@ -32,7 +34,7 @@ def markdown(value, _): | |||||||
|     else: |     else: | ||||||
|         return markdown.markdown(value) |         return markdown.markdown(value) | ||||||
|  |  | ||||||
| def restructuredtext(value, _): | def restructuredtext(value): | ||||||
|     try: |     try: | ||||||
|         from docutils.core import publish_parts |         from docutils.core import publish_parts | ||||||
|     except ImportError: |     except ImportError: | ||||||
| @@ -41,6 +43,6 @@ def restructuredtext(value, _): | |||||||
|         parts = publish_parts(source=value, writer_name="html4css1") |         parts = publish_parts(source=value, writer_name="html4css1") | ||||||
|         return parts["fragment"] |         return parts["fragment"] | ||||||
|  |  | ||||||
| template.register_filter("textile", textile, False) | register.filter(textile) | ||||||
| template.register_filter("markdown", markdown, False) | register.filter(markdown) | ||||||
| template.register_filter("restructuredtext", restructuredtext, False) | register.filter(restructuredtext) | ||||||
|   | |||||||
| @@ -3,7 +3,7 @@ This is the Django template system. | |||||||
|  |  | ||||||
| How it works: | How it works: | ||||||
|  |  | ||||||
| The tokenize() function converts a template string (i.e., a string containing | The Lexer.tokenize() function converts a template string (i.e., a string containing | ||||||
| markup with custom template tags) to tokens, which can be either plain text | markup with custom template tags) to tokens, which can be either plain text | ||||||
| (TOKEN_TEXT), variables (TOKEN_VAR) or block statements (TOKEN_BLOCK). | (TOKEN_TEXT), variables (TOKEN_VAR) or block statements (TOKEN_BLOCK). | ||||||
|  |  | ||||||
| @@ -55,6 +55,8 @@ times with multiple contexts) | |||||||
| '\n<html>\n\n</html>\n' | '\n<html>\n\n</html>\n' | ||||||
| """ | """ | ||||||
| import re | import re | ||||||
|  | from inspect import getargspec | ||||||
|  | from django.utils.functional import curry | ||||||
| from django.conf.settings import DEFAULT_CHARSET, TEMPLATE_DEBUG | from django.conf.settings import DEFAULT_CHARSET, TEMPLATE_DEBUG | ||||||
|  |  | ||||||
| __all__ = ('Template','Context','compile_string') | __all__ = ('Template','Context','compile_string') | ||||||
| @@ -82,11 +84,10 @@ UNKNOWN_SOURCE="<unknown source>" | |||||||
| tag_re = re.compile('(%s.*?%s|%s.*?%s)' % (re.escape(BLOCK_TAG_START), re.escape(BLOCK_TAG_END), | tag_re = re.compile('(%s.*?%s|%s.*?%s)' % (re.escape(BLOCK_TAG_START), re.escape(BLOCK_TAG_END), | ||||||
|                                           re.escape(VARIABLE_TAG_START), re.escape(VARIABLE_TAG_END))) |                                           re.escape(VARIABLE_TAG_START), re.escape(VARIABLE_TAG_END))) | ||||||
|  |  | ||||||
| # global dict used by register_tag; maps custom tags to callback functions | # global dictionary of libraries that have been loaded using get_library | ||||||
| registered_tags = {} | libraries = {} | ||||||
|  | # global list of libraries to load by default for a new parser | ||||||
| # global dict used by register_filter; maps custom filters to callback functions | builtins = [] | ||||||
| registered_filters = {} |  | ||||||
|  |  | ||||||
| class TemplateSyntaxError(Exception): | class TemplateSyntaxError(Exception): | ||||||
|     pass |     pass | ||||||
| @@ -105,12 +106,15 @@ class SilentVariableFailure(Exception): | |||||||
|     "Any function raising this exception will be ignored by resolve_variable" |     "Any function raising this exception will be ignored by resolve_variable" | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  | class InvalidTemplateLibrary(Exception): | ||||||
|  |     pass | ||||||
|  |  | ||||||
| class Origin(object): | class Origin(object): | ||||||
|     def __init__(self, name): |     def __init__(self, name): | ||||||
|         self.name = name |         self.name = name | ||||||
|  |  | ||||||
|     def reload(self): |     def reload(self): | ||||||
|         raise NotImplementedException |         raise NotImplementedError | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return self.name |         return self.name | ||||||
| @@ -264,6 +268,10 @@ class DebugLexer(Lexer): | |||||||
| class Parser(object): | class Parser(object): | ||||||
|     def __init__(self, tokens): |     def __init__(self, tokens): | ||||||
|         self.tokens = tokens |         self.tokens = tokens | ||||||
|  |         self.tags = {} | ||||||
|  |         self.filters = {} | ||||||
|  |         for lib in builtins: | ||||||
|  |             self.add_library(lib) | ||||||
|  |  | ||||||
|     def parse(self, parse_until=[]): |     def parse(self, parse_until=[]): | ||||||
|         nodelist = self.create_nodelist() |         nodelist = self.create_nodelist() | ||||||
| @@ -274,7 +282,8 @@ class Parser(object): | |||||||
|             elif token.token_type == TOKEN_VAR: |             elif token.token_type == TOKEN_VAR: | ||||||
|                 if not token.contents: |                 if not token.contents: | ||||||
|                     self.empty_variable(token) |                     self.empty_variable(token) | ||||||
|                 var_node = self.create_variable_node(token.contents) |                 filter_expression = self.compile_filter(token.contents) | ||||||
|  |                 var_node = self.create_variable_node(filter_expression) | ||||||
|                 self.extend_nodelist(nodelist, var_node,token) |                 self.extend_nodelist(nodelist, var_node,token) | ||||||
|             elif token.token_type == TOKEN_BLOCK: |             elif token.token_type == TOKEN_BLOCK: | ||||||
|                 if token.contents in parse_until: |                 if token.contents in parse_until: | ||||||
| @@ -288,7 +297,7 @@ class Parser(object): | |||||||
|                 # execute callback function for this tag and append resulting node |                 # execute callback function for this tag and append resulting node | ||||||
|                 self.enter_command(command, token) |                 self.enter_command(command, token) | ||||||
|                 try: |                 try: | ||||||
|                     compile_func = registered_tags[command] |                     compile_func = self.tags[command] | ||||||
|                 except KeyError: |                 except KeyError: | ||||||
|                     self.invalid_block_tag(token, command) |                     self.invalid_block_tag(token, command) | ||||||
|                 try: |                 try: | ||||||
| @@ -302,8 +311,8 @@ class Parser(object): | |||||||
|             self.unclosed_block_tag(parse_until) |             self.unclosed_block_tag(parse_until) | ||||||
|         return nodelist |         return nodelist | ||||||
|  |  | ||||||
|     def create_variable_node(self, contents): |     def create_variable_node(self, filter_expression): | ||||||
|         return VariableNode(contents) |         return VariableNode(filter_expression) | ||||||
|  |  | ||||||
|     def create_nodelist(self): |     def create_nodelist(self): | ||||||
|         return NodeList() |         return NodeList() | ||||||
| @@ -344,6 +353,20 @@ class Parser(object): | |||||||
|     def delete_first_token(self): |     def delete_first_token(self): | ||||||
|         del self.tokens[0] |         del self.tokens[0] | ||||||
|  |  | ||||||
|  |     def add_library(self, lib): | ||||||
|  |         self.tags.update(lib.tags) | ||||||
|  |         self.filters.update(lib.filters) | ||||||
|  |  | ||||||
|  |     def compile_filter(self,token): | ||||||
|  |         "Convenient wrapper for FilterExpression" | ||||||
|  |         return FilterExpression(token, self) | ||||||
|  |  | ||||||
|  |     def find_filter(self, filter_name): | ||||||
|  |         if self.filters.has_key(filter_name): | ||||||
|  |             return self.filters[filter_name] | ||||||
|  |         else: | ||||||
|  |             raise TemplateSyntaxError, "Invalid filter: '%s'" % filter_name | ||||||
|  |  | ||||||
| class DebugParser(Parser): | class DebugParser(Parser): | ||||||
|     def __init__(self, lexer): |     def __init__(self, lexer): | ||||||
|         super(DebugParser, self).__init__(lexer) |         super(DebugParser, self).__init__(lexer) | ||||||
| @@ -483,7 +506,8 @@ filter_raw_string = r""" | |||||||
|          (?:%(arg_sep)s |          (?:%(arg_sep)s | ||||||
|              (?: |              (?: | ||||||
|               %(i18n_open)s"(?P<i18n_arg>%(str)s)"%(i18n_close)s| |               %(i18n_open)s"(?P<i18n_arg>%(str)s)"%(i18n_close)s| | ||||||
|               "(?P<arg>%(str)s)" |               "(?P<constant_arg>%(str)s)"| | ||||||
|  |               (?P<var_arg>[%(var_chars)s]+) | ||||||
|              ) |              ) | ||||||
|          )? |          )? | ||||||
|  )""" % { |  )""" % { | ||||||
| @@ -498,7 +522,7 @@ filter_raw_string = r""" | |||||||
| filter_raw_string = filter_raw_string.replace("\n", "").replace(" ", "") | filter_raw_string = filter_raw_string.replace("\n", "").replace(" ", "") | ||||||
| filter_re = re.compile(filter_raw_string) | filter_re = re.compile(filter_raw_string) | ||||||
|  |  | ||||||
| class FilterParser(object): | class FilterExpression(object): | ||||||
|     """ |     """ | ||||||
|     Parses a variable token and its optional filters (all as a single string), |     Parses a variable token and its optional filters (all as a single string), | ||||||
|     and return a list of tuples of the filter name and arguments. |     and return a list of tuples of the filter name and arguments. | ||||||
| @@ -513,7 +537,8 @@ class FilterParser(object): | |||||||
|     This class should never be instantiated outside of the |     This class should never be instantiated outside of the | ||||||
|     get_filters_from_token helper function. |     get_filters_from_token helper function. | ||||||
|     """ |     """ | ||||||
|     def __init__(self, token): |     def __init__(self, token, parser): | ||||||
|  |         self.token = token | ||||||
|         matches = filter_re.finditer(token) |         matches = filter_re.finditer(token) | ||||||
|         var = None |         var = None | ||||||
|         filters = [] |         filters = [] | ||||||
| @@ -536,27 +561,69 @@ class FilterParser(object): | |||||||
|                     raise TemplateSyntaxError, "Variables and attributes may not begin with underscores: '%s'" % var |                     raise TemplateSyntaxError, "Variables and attributes may not begin with underscores: '%s'" % var | ||||||
|             else: |             else: | ||||||
|                 filter_name = match.group("filter_name") |                 filter_name = match.group("filter_name") | ||||||
|                 arg, i18n_arg = match.group("arg","i18n_arg") |                 args = [] | ||||||
|  |                 constant_arg, i18n_arg, var_arg = match.group("constant_arg", "i18n_arg", "var_arg") | ||||||
|                 if i18n_arg: |                 if i18n_arg: | ||||||
|                     arg =_(i18n_arg.replace('\\', '')) |                     args.append((False, _(i18n_arg.replace('\\', '')))) | ||||||
|                 if arg: |                 elif constant_arg: | ||||||
|                     arg = arg.replace('\\', '') |                     args.append((False, constant_arg.replace('\\', ''))) | ||||||
|                 if not registered_filters.has_key(filter_name): |                 elif var_arg: | ||||||
|                     raise TemplateSyntaxError, "Invalid filter: '%s'" % filter_name |                     args.append((True, var_arg)) | ||||||
|                 if registered_filters[filter_name][1] == True and arg is None: |                 filter_func = parser.find_filter(filter_name) | ||||||
|                     raise TemplateSyntaxError, "Filter '%s' requires an argument" % filter_name |                 self.args_check(filter_name,filter_func, args) | ||||||
|                 if registered_filters[filter_name][1] == False and arg is not None: |                 filters.append( (filter_func,args)) | ||||||
|                     raise TemplateSyntaxError, "Filter '%s' should not have an argument (argument is %r)" % (filter_name, arg) |  | ||||||
|                 filters.append( (filter_name,arg) ) |  | ||||||
|                 upto = match.end() |                 upto = match.end() | ||||||
|         if upto != len(token): |         if upto != len(token): | ||||||
|             raise TemplateSyntaxError, "Could not parse the remainder: %s" % token[upto:] |             raise TemplateSyntaxError, "Could not parse the remainder: %s" % token[upto:] | ||||||
|         self.var , self.filters = var, filters |         self.var , self.filters = var, filters | ||||||
|  |  | ||||||
| def get_filters_from_token(token): |     def resolve(self, context): | ||||||
|     "Convenient wrapper for FilterParser" |         try: | ||||||
|     p = FilterParser(token) |             obj = resolve_variable(self.var, context) | ||||||
|     return (p.var, p.filters) |         except VariableDoesNotExist: | ||||||
|  |             obj = '' | ||||||
|  |         for func, args in self.filters: | ||||||
|  |             arg_vals = [] | ||||||
|  |             for lookup, arg in args: | ||||||
|  |                 if not lookup: | ||||||
|  |                     arg_vals.append(arg) | ||||||
|  |                 else: | ||||||
|  |                     arg_vals.append(resolve_variable(arg, context)) | ||||||
|  |             obj = func(obj, *arg_vals) | ||||||
|  |         return obj | ||||||
|  |  | ||||||
|  |     def args_check(name, func, provided): | ||||||
|  |         provided = list(provided) | ||||||
|  |         plen = len(provided) | ||||||
|  |         (args, varargs, varkw, defaults) = getargspec(func) | ||||||
|  |         # First argument is filter input. | ||||||
|  |         args.pop(0) | ||||||
|  |         if defaults: | ||||||
|  |             nondefs = args[:-len(defaults)] | ||||||
|  |         else: | ||||||
|  |             nondefs = args | ||||||
|  |         # Args without defaults must be provided. | ||||||
|  |         try: | ||||||
|  |             for arg in nondefs: | ||||||
|  |                 provided.pop(0) | ||||||
|  |         except IndexError: | ||||||
|  |             # Not enough | ||||||
|  |             raise TemplateSyntaxError, "%s requires %d arguments, %d provided" % (name, len(nondefs), plen) | ||||||
|  |  | ||||||
|  |         # Defaults can be overridden. | ||||||
|  |         defaults = defaults and list(defaults) or [] | ||||||
|  |         try: | ||||||
|  |             for parg in provided: | ||||||
|  |                 defaults.pop(0) | ||||||
|  |         except IndexError: | ||||||
|  |             # Too many. | ||||||
|  |             raise TemplateSyntaxError, "%s requires %d arguments, %d provided" % (name, len(nondefs), plen) | ||||||
|  |  | ||||||
|  |         return True | ||||||
|  |     args_check = staticmethod(args_check) | ||||||
|  |  | ||||||
|  |     def __str__(self): | ||||||
|  |         return self.token | ||||||
|  |  | ||||||
| def resolve_variable(path, context): | def resolve_variable(path, context): | ||||||
|     """ |     """ | ||||||
| @@ -607,22 +674,6 @@ def resolve_variable(path, context): | |||||||
|             del bits[0] |             del bits[0] | ||||||
|     return current |     return current | ||||||
|  |  | ||||||
| def resolve_variable_with_filters(var_string, context): |  | ||||||
|     """ |  | ||||||
|     var_string is a full variable expression with optional filters, like: |  | ||||||
|         a.b.c|lower|date:"y/m/d" |  | ||||||
|     This function resolves the variable in the context, applies all filters and |  | ||||||
|     returns the object. |  | ||||||
|     """ |  | ||||||
|     var, filters = get_filters_from_token(var_string) |  | ||||||
|     try: |  | ||||||
|         obj = resolve_variable(var, context) |  | ||||||
|     except VariableDoesNotExist: |  | ||||||
|         obj = '' |  | ||||||
|     for name, arg in filters: |  | ||||||
|         obj = registered_filters[name][0](obj, arg) |  | ||||||
|     return obj |  | ||||||
|  |  | ||||||
| class Node: | class Node: | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         "Return the node rendered as a string" |         "Return the node rendered as a string" | ||||||
| @@ -687,11 +738,11 @@ class TextNode(Node): | |||||||
|         return self.s |         return self.s | ||||||
|  |  | ||||||
| class VariableNode(Node): | class VariableNode(Node): | ||||||
|     def __init__(self, var_string): |     def __init__(self, filter_expression): | ||||||
|         self.var_string = var_string |         self.filter_expression = filter_expression | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return "<Variable Node: %s>" % self.var_string |         return "<Variable Node: %s>" % self.filter_expression | ||||||
|  |  | ||||||
|     def encode_output(self, output): |     def encode_output(self, output): | ||||||
|         # Check type so that we don't run str() on a Unicode object |         # Check type so that we don't run str() on a Unicode object | ||||||
| @@ -703,30 +754,153 @@ class VariableNode(Node): | |||||||
|             return output |             return output | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         output = resolve_variable_with_filters(self.var_string, context) |         output = self.filter_expression.resolve(context) | ||||||
|         return self.encode_output(output) |         return self.encode_output(output) | ||||||
|  |  | ||||||
| class DebugVariableNode(VariableNode): | class DebugVariableNode(VariableNode): | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         try: |         try: | ||||||
|              output = resolve_variable_with_filters(self.var_string, context) |              output = self.filter_expression.resolve(context) | ||||||
|         except TemplateSyntaxError, e: |         except TemplateSyntaxError, e: | ||||||
|             if not hasattr(e, 'source'): |             if not hasattr(e, 'source'): | ||||||
|                 e.source = self.source |                 e.source = self.source | ||||||
|             raise |             raise | ||||||
|         return self.encode_output(output) |         return self.encode_output(output) | ||||||
|  |  | ||||||
| def register_tag(token_command, callback_function): | def generic_tag_compiler(params, defaults, name, node_class, parser, token): | ||||||
|     registered_tags[token_command] = callback_function |     "Returns a template.Node subclass." | ||||||
|  |     bits = token.contents.split()[1:] | ||||||
|  |     bmax = len(params) | ||||||
|  |     def_len = defaults and len(defaults) or 0 | ||||||
|  |     bmin = bmax - def_len | ||||||
|  |     if(len(bits) < bmin or len(bits) > bmax): | ||||||
|  |         if bmin == bmax: | ||||||
|  |             message = "%s takes %s arguments" % (name, bmin) | ||||||
|  |         else: | ||||||
|  |             message = "%s takes between %s and %s arguments" % (name, bmin, bmax) | ||||||
|  |         raise TemplateSyntaxError, message | ||||||
|  |     return node_class(bits) | ||||||
|  |  | ||||||
| def unregister_tag(token_command): | class Library(object): | ||||||
|     del registered_tags[token_command] |     def __init__(self): | ||||||
|  |         self.filters = {} | ||||||
|  |         self.tags = {} | ||||||
|  |  | ||||||
| def register_filter(filter_name, callback_function, has_arg): |     def tag(self, name = None, compile_function = None): | ||||||
|     registered_filters[filter_name] = (callback_function, has_arg) |         if name == None and compile_function == None: | ||||||
|  |             # @register.tag() | ||||||
|  |             return self.tag_function | ||||||
|  |         elif name != None and compile_function == None: | ||||||
|  |             if(callable(name)): | ||||||
|  |                 # @register.tag | ||||||
|  |                 return self.tag_function(name) | ||||||
|  |             else: | ||||||
|  |                 # @register.tag('somename') or @register.tag(name='somename') | ||||||
|  |                 def dec(func): | ||||||
|  |                     return self.tag(name, func) | ||||||
|  |                 return dec | ||||||
|  |         elif name != None and compile_function != None: | ||||||
|  |             # register.tag('somename', somefunc) | ||||||
|  |             self.tags[name] = compile_function | ||||||
|  |             return compile_function | ||||||
|  |         else: | ||||||
|  |             raise InvalidTemplateLibrary, "Unsupported arguments to Library.tag: (%r, %r)", (name, compile_function) | ||||||
|  |  | ||||||
| def unregister_filter(filter_name): |     def tag_function(self,func): | ||||||
|     del registered_filters[filter_name] |         self.tags[func.__name__] = func | ||||||
|  |         return func | ||||||
|  |  | ||||||
| import defaulttags |     def filter(self, name = None, filter_func = None): | ||||||
| import defaultfilters |         if name == None and filter_func == None: | ||||||
|  |             # @register.filter() | ||||||
|  |             return self.filter_function | ||||||
|  |         elif filter_func == None: | ||||||
|  |             if(callable(name)): | ||||||
|  |                 # @register.filter | ||||||
|  |                 return self.filter_function(name) | ||||||
|  |             else: | ||||||
|  |                 # @register.filter('somename') or @register.filter(name='somename') | ||||||
|  |                 def dec(func): | ||||||
|  |                     return self.filter(name, func) | ||||||
|  |                 return dec | ||||||
|  |         elif name != None and filter_func != None: | ||||||
|  |             # register.filter('somename', somefunc) | ||||||
|  |             self.filters[name] = filter_func | ||||||
|  |         else: | ||||||
|  |             raise InvalidTemplateLibrary, "Unsupported arguments to Library.filter: (%r, %r, %r)", (name, compile_function, has_arg) | ||||||
|  |  | ||||||
|  |     def filter_function(self, func): | ||||||
|  |         self.filters[func.__name__] = func | ||||||
|  |         return func | ||||||
|  |  | ||||||
|  |     def simple_tag(self,func): | ||||||
|  |         (params, xx, xxx, defaults) = getargspec(func) | ||||||
|  |  | ||||||
|  |         class SimpleNode(Node): | ||||||
|  |             def __init__(self, vars_to_resolve): | ||||||
|  |                 self.vars_to_resolve = vars_to_resolve | ||||||
|  |  | ||||||
|  |             def render(self, context): | ||||||
|  |                 resolved_vars = [resolve_variable(var, context) for var in self.vars_to_resolve] | ||||||
|  |                 return func(*resolved_vars) | ||||||
|  |  | ||||||
|  |         compile_func = curry(generic_tag_compiler, params, defaults, func.__name__, SimpleNode) | ||||||
|  |         compile_func.__doc__ = func.__doc__ | ||||||
|  |         self.tag(func.__name__, compile_func) | ||||||
|  |         return func | ||||||
|  |  | ||||||
|  |     def inclusion_tag(self, file_name, context_class=Context, takes_context=False): | ||||||
|  |         def dec(func): | ||||||
|  |             (params, xx, xxx, defaults) = getargspec(func) | ||||||
|  |             if takes_context: | ||||||
|  |                 if params[0] == 'context': | ||||||
|  |                     params = params[1:] | ||||||
|  |                 else: | ||||||
|  |                     raise TemplateSyntaxError, "Any tag function decorated with takes_context=True must have a first argument of 'context'" | ||||||
|  |  | ||||||
|  |             class InclusionNode(Node): | ||||||
|  |                 def __init__(self, vars_to_resolve): | ||||||
|  |                     self.vars_to_resolve = vars_to_resolve | ||||||
|  |  | ||||||
|  |                 def render(self, context): | ||||||
|  |                     resolved_vars = [resolve_variable(var, context) for var in self.vars_to_resolve] | ||||||
|  |                     if takes_context: | ||||||
|  |                         args = [context] + resolved_vars | ||||||
|  |                     else: | ||||||
|  |                         args = resolved_vars | ||||||
|  |  | ||||||
|  |                     dict = func(*args) | ||||||
|  |  | ||||||
|  |                     if not getattr(self, 'nodelist', False): | ||||||
|  |                         from django.core.template_loader import get_template | ||||||
|  |                         t = get_template(file_name) | ||||||
|  |                         self.nodelist = t.nodelist | ||||||
|  |                     return self.nodelist.render(context_class(dict)) | ||||||
|  |  | ||||||
|  |             compile_func = curry(generic_tag_compiler, params, defaults, func.__name__, InclusionNode) | ||||||
|  |             compile_func.__doc__ = func.__doc__ | ||||||
|  |             self.tag(func.__name__, compile_func) | ||||||
|  |             return func | ||||||
|  |         return dec | ||||||
|  |  | ||||||
|  | def get_library(module_name): | ||||||
|  |     lib = libraries.get(module_name, None) | ||||||
|  |     if not lib: | ||||||
|  |         try: | ||||||
|  |             mod = __import__(module_name, '', '', ['']) | ||||||
|  |         except ImportError, e: | ||||||
|  |             raise InvalidTemplateLibrary, "Could not load template library from %s, %s" % (module_name, e) | ||||||
|  |         for k, v in mod.__dict__.items(): | ||||||
|  |             if isinstance(v, Library): | ||||||
|  |                 lib = v | ||||||
|  |                 libraries[module_name] = lib | ||||||
|  |                 break | ||||||
|  |     if not lib: | ||||||
|  |         raise InvalidTemplateLibrary, "Template library %s does not have a Library member" % module_name | ||||||
|  |     return lib | ||||||
|  |  | ||||||
|  | def add_to_builtins(module_name): | ||||||
|  |     builtins.append(get_library(module_name)) | ||||||
|  |  | ||||||
|  | add_to_builtins('django.core.template.defaulttags') | ||||||
|  | add_to_builtins('django.core.template.defaultfilters') | ||||||
|   | |||||||
| @@ -1,67 +0,0 @@ | |||||||
| from django.core.template import Context, Node, TemplateSyntaxError, register_tag, resolve_variable |  | ||||||
| from django.core.template_loader import get_template |  | ||||||
| from django.utils.functional import curry |  | ||||||
| from inspect import getargspec |  | ||||||
|  |  | ||||||
| def generic_tag_compiler(params, defaults, name, node_class, parser, token): |  | ||||||
|     "Returns a template.Node subclass." |  | ||||||
|     bits = token.contents.split()[1:] |  | ||||||
|     bmax = len(params) |  | ||||||
|     def_len = defaults and len(defaults) or 0 |  | ||||||
|     bmin = bmax - def_len |  | ||||||
|     if(len(bits) < bmin or len(bits) > bmax): |  | ||||||
|         if bmin == bmax: |  | ||||||
|             message = "%s takes %s arguments" % (name, bmin) |  | ||||||
|         else: |  | ||||||
|             message = "%s takes between %s and %s arguments" % (name, bmin, bmax) |  | ||||||
|         raise TemplateSyntaxError, message |  | ||||||
|     return node_class(bits) |  | ||||||
|  |  | ||||||
| def simple_tag(func): |  | ||||||
|     (params, xx, xxx, defaults) = getargspec(func) |  | ||||||
|  |  | ||||||
|     class SimpleNode(Node): |  | ||||||
|         def __init__(self, vars_to_resolve): |  | ||||||
|             self.vars_to_resolve = vars_to_resolve |  | ||||||
|  |  | ||||||
|         def render(self, context): |  | ||||||
|             resolved_vars = [resolve_variable(var, context) for var in self.vars_to_resolve] |  | ||||||
|             return func(*resolved_vars) |  | ||||||
|  |  | ||||||
|     compile_func = curry(generic_tag_compiler, params, defaults, func.__name__, SimpleNode) |  | ||||||
|     compile_func.__doc__ = func.__doc__ |  | ||||||
|     register_tag(func.__name__, compile_func) |  | ||||||
|     return func |  | ||||||
|  |  | ||||||
| def inclusion_tag(file_name, context_class=Context, takes_context=False): |  | ||||||
|     def dec(func): |  | ||||||
|         (params, xx, xxx, defaults) = getargspec(func) |  | ||||||
|         if takes_context: |  | ||||||
|             if params[0] == 'context': |  | ||||||
|                 params = params[1:] |  | ||||||
|             else: |  | ||||||
|                 raise TemplateSyntaxError, "Any tag function decorated with takes_context=True must have a first argument of 'context'" |  | ||||||
|  |  | ||||||
|         class InclusionNode(Node): |  | ||||||
|             def __init__(self, vars_to_resolve): |  | ||||||
|                 self.vars_to_resolve = vars_to_resolve |  | ||||||
|  |  | ||||||
|             def render(self, context): |  | ||||||
|                 resolved_vars = [resolve_variable(var, context) for var in self.vars_to_resolve] |  | ||||||
|                 if takes_context: |  | ||||||
|                     args = [context] + resolved_vars |  | ||||||
|                 else: |  | ||||||
|                     args = resolved_vars |  | ||||||
|  |  | ||||||
|                 dict = func(*args) |  | ||||||
|  |  | ||||||
|                 if not getattr(self, 'nodelist', False): |  | ||||||
|                     t = get_template(file_name) |  | ||||||
|                     self.nodelist = t.nodelist |  | ||||||
|                 return self.nodelist.render(context_class(dict)) |  | ||||||
|  |  | ||||||
|         compile_func = curry(generic_tag_compiler, params, defaults, func.__name__, InclusionNode) |  | ||||||
|         compile_func.__doc__ = func.__doc__ |  | ||||||
|         register_tag(func.__name__, compile_func) |  | ||||||
|         return func |  | ||||||
|     return dec |  | ||||||
| @@ -1,28 +1,32 @@ | |||||||
| "Default variable filters" | "Default variable filters" | ||||||
|  |  | ||||||
| from django.core.template import register_filter, resolve_variable | from django.core.template import resolve_variable, Library | ||||||
|  | from django.conf.settings import DATE_FORMAT, TIME_FORMAT | ||||||
| import re | import re | ||||||
| import random as random_module | import random as random_module | ||||||
|  |  | ||||||
|  | register = Library() | ||||||
|  |  | ||||||
| ################### | ################### | ||||||
| # STRINGS         # | # STRINGS         # | ||||||
| ################### | ################### | ||||||
|  |  | ||||||
| def addslashes(value, _): |  | ||||||
|  | def addslashes(value): | ||||||
|     "Adds slashes - useful for passing strings to JavaScript, for example." |     "Adds slashes - useful for passing strings to JavaScript, for example." | ||||||
|     return value.replace('"', '\\"').replace("'", "\\'") |     return value.replace('"', '\\"').replace("'", "\\'") | ||||||
|  |  | ||||||
| def capfirst(value, _): | def capfirst(value): | ||||||
|     "Capitalizes the first character of the value" |     "Capitalizes the first character of the value" | ||||||
|     value = str(value) |     value = str(value) | ||||||
|     return value and value[0].upper() + value[1:] |     return value and value[0].upper() + value[1:] | ||||||
|  |  | ||||||
| def fix_ampersands(value, _): | def fix_ampersands(value): | ||||||
|     "Replaces ampersands with ``&`` entities" |     "Replaces ampersands with ``&`` entities" | ||||||
|     from django.utils.html import fix_ampersands |     from django.utils.html import fix_ampersands | ||||||
|     return fix_ampersands(value) |     return fix_ampersands(value) | ||||||
|  |  | ||||||
| def floatformat(text, _): | def floatformat(text): | ||||||
|     """ |     """ | ||||||
|     Displays a floating point number as 34.2 (with one decimal place) -- but |     Displays a floating point number as 34.2 (with one decimal place) -- but | ||||||
|     only if there's a point to be displayed |     only if there's a point to be displayed | ||||||
| @@ -37,7 +41,7 @@ def floatformat(text, _): | |||||||
|     else: |     else: | ||||||
|         return '%d' % int(f) |         return '%d' % int(f) | ||||||
|  |  | ||||||
| def linenumbers(value, _): | def linenumbers(value): | ||||||
|     "Displays text with line numbers" |     "Displays text with line numbers" | ||||||
|     from django.utils.html import escape |     from django.utils.html import escape | ||||||
|     lines = value.split('\n') |     lines = value.split('\n') | ||||||
| @@ -47,18 +51,18 @@ def linenumbers(value, _): | |||||||
|         lines[i] = ("%0" + width  + "d. %s") % (i + 1, escape(line)) |         lines[i] = ("%0" + width  + "d. %s") % (i + 1, escape(line)) | ||||||
|     return '\n'.join(lines) |     return '\n'.join(lines) | ||||||
|  |  | ||||||
| def lower(value, _): | def lower(value): | ||||||
|     "Converts a string into all lowercase" |     "Converts a string into all lowercase" | ||||||
|     return value.lower() |     return value.lower() | ||||||
|  |  | ||||||
| def make_list(value, _): | def make_list(value): | ||||||
|     """ |     """ | ||||||
|     Returns the value turned into a list. For an integer, it's a list of |     Returns the value turned into a list. For an integer, it's a list of | ||||||
|     digits. For a string, it's a list of characters. |     digits. For a string, it's a list of characters. | ||||||
|     """ |     """ | ||||||
|     return list(str(value)) |     return list(str(value)) | ||||||
|  |  | ||||||
| def slugify(value, _): | def slugify(value): | ||||||
|     "Converts to lowercase, removes non-alpha chars and converts spaces to hyphens" |     "Converts to lowercase, removes non-alpha chars and converts spaces to hyphens" | ||||||
|     value = re.sub('[^\w\s-]', '', value).strip().lower() |     value = re.sub('[^\w\s-]', '', value).strip().lower() | ||||||
|     return re.sub('\s+', '-', value) |     return re.sub('\s+', '-', value) | ||||||
| @@ -77,7 +81,7 @@ def stringformat(value, arg): | |||||||
|     except (ValueError, TypeError): |     except (ValueError, TypeError): | ||||||
|         return "" |         return "" | ||||||
|  |  | ||||||
| def title(value, _): | def title(value): | ||||||
|     "Converts a string into titlecase" |     "Converts a string into titlecase" | ||||||
|     return re.sub("([a-z])'([A-Z])", lambda m: m.group(0).lower(), value.title()) |     return re.sub("([a-z])'([A-Z])", lambda m: m.group(0).lower(), value.title()) | ||||||
|  |  | ||||||
| @@ -96,16 +100,16 @@ def truncatewords(value, arg): | |||||||
|         value = str(value) |         value = str(value) | ||||||
|     return truncate_words(value, length) |     return truncate_words(value, length) | ||||||
|  |  | ||||||
| def upper(value, _): | def upper(value): | ||||||
|     "Converts a string into all uppercase" |     "Converts a string into all uppercase" | ||||||
|     return value.upper() |     return value.upper() | ||||||
|  |  | ||||||
| def urlencode(value, _): | def urlencode(value): | ||||||
|     "Escapes a value for use in a URL" |     "Escapes a value for use in a URL" | ||||||
|     import urllib |     import urllib | ||||||
|     return urllib.quote(value) |     return urllib.quote(value) | ||||||
|  |  | ||||||
| def urlize(value, _): | def urlize(value): | ||||||
|     "Converts URLs in plain text into clickable links" |     "Converts URLs in plain text into clickable links" | ||||||
|     from django.utils.html import urlize |     from django.utils.html import urlize | ||||||
|     return urlize(value, nofollow=True) |     return urlize(value, nofollow=True) | ||||||
| @@ -119,7 +123,7 @@ def urlizetrunc(value, limit): | |||||||
|     from django.utils.html import urlize |     from django.utils.html import urlize | ||||||
|     return urlize(value, trim_url_limit=int(limit), nofollow=True) |     return urlize(value, trim_url_limit=int(limit), nofollow=True) | ||||||
|  |  | ||||||
| def wordcount(value, _): | def wordcount(value): | ||||||
|     "Returns the number of words" |     "Returns the number of words" | ||||||
|     return len(value.split()) |     return len(value.split()) | ||||||
|  |  | ||||||
| @@ -160,17 +164,17 @@ def cut(value, arg): | |||||||
| # HTML STRINGS    # | # HTML STRINGS    # | ||||||
| ################### | ################### | ||||||
|  |  | ||||||
| def escape(value, _): | def escape(value): | ||||||
|     "Escapes a string's HTML" |     "Escapes a string's HTML" | ||||||
|     from django.utils.html import escape |     from django.utils.html import escape | ||||||
|     return escape(value) |     return escape(value) | ||||||
|  |  | ||||||
| def linebreaks(value, _): | def linebreaks(value): | ||||||
|     "Converts newlines into <p> and <br />s" |     "Converts newlines into <p> and <br />s" | ||||||
|     from django.utils.html import linebreaks |     from django.utils.html import linebreaks | ||||||
|     return linebreaks(value) |     return linebreaks(value) | ||||||
|  |  | ||||||
| def linebreaksbr(value, _): | def linebreaksbr(value): | ||||||
|     "Converts newlines into <br />s" |     "Converts newlines into <br />s" | ||||||
|     return value.replace('\n', '<br />') |     return value.replace('\n', '<br />') | ||||||
|  |  | ||||||
| @@ -184,7 +188,7 @@ def removetags(value, tags): | |||||||
|     value = endtag_re.sub('', value) |     value = endtag_re.sub('', value) | ||||||
|     return value |     return value | ||||||
|  |  | ||||||
| def striptags(value, _): | def striptags(value): | ||||||
|     "Strips all [X]HTML tags" |     "Strips all [X]HTML tags" | ||||||
|     from django.utils.html import strip_tags |     from django.utils.html import strip_tags | ||||||
|     if not isinstance(value, basestring): |     if not isinstance(value, basestring): | ||||||
| @@ -214,7 +218,7 @@ def dictsortreversed(value, arg): | |||||||
|     decorated.reverse() |     decorated.reverse() | ||||||
|     return [item[1] for item in decorated] |     return [item[1] for item in decorated] | ||||||
|  |  | ||||||
| def first(value, _): | def first(value): | ||||||
|     "Returns the first item in a list" |     "Returns the first item in a list" | ||||||
|     try: |     try: | ||||||
|         return value[0] |         return value[0] | ||||||
| @@ -228,7 +232,7 @@ def join(value, arg): | |||||||
|     except AttributeError: # fail silently but nicely |     except AttributeError: # fail silently but nicely | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
| def length(value, _): | def length(value): | ||||||
|     "Returns the length of the value - useful for lists" |     "Returns the length of the value - useful for lists" | ||||||
|     return len(value) |     return len(value) | ||||||
|  |  | ||||||
| @@ -236,7 +240,7 @@ def length_is(value, arg): | |||||||
|     "Returns a boolean of whether the value's length is the argument" |     "Returns a boolean of whether the value's length is the argument" | ||||||
|     return len(value) == int(arg) |     return len(value) == int(arg) | ||||||
|  |  | ||||||
| def random(value, _): | def random(value): | ||||||
|     "Returns a random item from the list" |     "Returns a random item from the list" | ||||||
|     return random_module.choice(value) |     return random_module.choice(value) | ||||||
|  |  | ||||||
| @@ -253,7 +257,7 @@ def slice_(value, arg): | |||||||
|     except (ValueError, TypeError): |     except (ValueError, TypeError): | ||||||
|         return value # Fail silently. |         return value # Fail silently. | ||||||
|  |  | ||||||
| def unordered_list(value, _): | def unordered_list(value): | ||||||
|     """ |     """ | ||||||
|     Recursively takes a self-nested list and returns an HTML unordered list -- |     Recursively takes a self-nested list and returns an HTML unordered list -- | ||||||
|     WITHOUT opening and closing <ul> tags. |     WITHOUT opening and closing <ul> tags. | ||||||
| @@ -314,17 +318,17 @@ def get_digit(value, arg): | |||||||
| # DATES           # | # DATES           # | ||||||
| ################### | ################### | ||||||
|  |  | ||||||
| def date(value, arg): | def date(value, arg=DATE_FORMAT): | ||||||
|     "Formats a date according to the given format" |     "Formats a date according to the given format" | ||||||
|     from django.utils.dateformat import format |     from django.utils.dateformat import format | ||||||
|     return format(value, arg) |     return format(value, arg) | ||||||
|  |  | ||||||
| def time(value, arg): | def time(value, arg=TIME_FORMAT): | ||||||
|     "Formats a time according to the given format" |     "Formats a time according to the given format" | ||||||
|     from django.utils.dateformat import time_format |     from django.utils.dateformat import time_format | ||||||
|     return time_format(value, arg) |     return time_format(value, arg) | ||||||
|  |  | ||||||
| def timesince(value, _): | def timesince(value): | ||||||
|     'Formats a date as the time since that date (i.e. "4 days, 6 hours")' |     'Formats a date as the time since that date (i.e. "4 days, 6 hours")' | ||||||
|     from django.utils.timesince import timesince |     from django.utils.timesince import timesince | ||||||
|     return timesince(value) |     return timesince(value) | ||||||
| @@ -347,7 +351,7 @@ def divisibleby(value, arg): | |||||||
|     "Returns true if the value is devisible by the argument" |     "Returns true if the value is devisible by the argument" | ||||||
|     return int(value) % int(arg) == 0 |     return int(value) % int(arg) == 0 | ||||||
|  |  | ||||||
| def yesno(value, arg): | def yesno(value, arg=_("yes,no,maybe")): | ||||||
|     """ |     """ | ||||||
|     Given a string mapping values for true, false and (optionally) None, |     Given a string mapping values for true, false and (optionally) None, | ||||||
|     returns one of those strings accoding to the value: |     returns one of those strings accoding to the value: | ||||||
| @@ -379,7 +383,7 @@ def yesno(value, arg): | |||||||
| # MISC            # | # MISC            # | ||||||
| ################### | ################### | ||||||
|  |  | ||||||
| def filesizeformat(bytes, _): | def filesizeformat(bytes): | ||||||
|     """ |     """ | ||||||
|     Format the value like a 'human-readable' file size (i.e. 13 KB, 4.1 MB, 102 |     Format the value like a 'human-readable' file size (i.e. 13 KB, 4.1 MB, 102 | ||||||
|     bytes, etc). |     bytes, etc). | ||||||
| @@ -393,7 +397,7 @@ def filesizeformat(bytes, _): | |||||||
|         return "%.1f MB" % (bytes / (1024 * 1024)) |         return "%.1f MB" % (bytes / (1024 * 1024)) | ||||||
|     return "%.1f GB" % (bytes / (1024 * 1024 * 1024)) |     return "%.1f GB" % (bytes / (1024 * 1024 * 1024)) | ||||||
|  |  | ||||||
| def pluralize(value, _): | def pluralize(value): | ||||||
|     "Returns 's' if the value is not 1, for '1 vote' vs. '2 votes'" |     "Returns 's' if the value is not 1, for '1 vote' vs. '2 votes'" | ||||||
|     try: |     try: | ||||||
|         if int(value) != 1: |         if int(value) != 1: | ||||||
| @@ -408,62 +412,62 @@ def pluralize(value, _): | |||||||
|             pass |             pass | ||||||
|     return '' |     return '' | ||||||
|  |  | ||||||
| def phone2numeric(value, _): | def phone2numeric(value): | ||||||
|     "Takes a phone number and converts it in to its numerical equivalent" |     "Takes a phone number and converts it in to its numerical equivalent" | ||||||
|     from django.utils.text import phone2numeric |     from django.utils.text import phone2numeric | ||||||
|     return phone2numeric(value) |     return phone2numeric(value) | ||||||
|  |  | ||||||
| def pprint(value, _): | def pprint(value): | ||||||
|     "A wrapper around pprint.pprint -- for debugging, really" |     "A wrapper around pprint.pprint -- for debugging, really" | ||||||
|     from pprint import pformat |     from pprint import pformat | ||||||
|     return pformat(value) |     return pformat(value) | ||||||
|  |  | ||||||
| # Syntax: register_filter(name of filter, callback, has_argument) | # Syntax: register.filter(name of filter, callback) | ||||||
| register_filter('add', add, True) | register.filter(add) | ||||||
| register_filter('addslashes', addslashes, False) | register.filter(addslashes) | ||||||
| register_filter('capfirst', capfirst, False) | register.filter(capfirst) | ||||||
| register_filter('center', center, True) | register.filter(center) | ||||||
| register_filter('cut', cut, True) | register.filter(cut) | ||||||
| register_filter('date', date, True) | register.filter(date) | ||||||
| register_filter('default', default, True) | register.filter(default) | ||||||
| register_filter('default_if_none', default_if_none, True) | register.filter(default_if_none) | ||||||
| register_filter('dictsort', dictsort, True) | register.filter(dictsort) | ||||||
| register_filter('dictsortreversed', dictsortreversed, True) | register.filter(dictsortreversed) | ||||||
| register_filter('divisibleby', divisibleby, True) | register.filter(divisibleby) | ||||||
| register_filter('escape', escape, False) | register.filter(escape) | ||||||
| register_filter('filesizeformat', filesizeformat, False) | register.filter(filesizeformat) | ||||||
| register_filter('first', first, False) | register.filter(first) | ||||||
| register_filter('fix_ampersands', fix_ampersands, False) | register.filter(fix_ampersands) | ||||||
| register_filter('floatformat', floatformat, False) | register.filter(floatformat) | ||||||
| register_filter('get_digit', get_digit, True) | register.filter(get_digit) | ||||||
| register_filter('join', join, True) | register.filter(join) | ||||||
| register_filter('length', length, False) | register.filter(length) | ||||||
| register_filter('length_is', length_is, True) | register.filter(length_is) | ||||||
| register_filter('linebreaks', linebreaks, False) | register.filter(linebreaks) | ||||||
| register_filter('linebreaksbr', linebreaksbr, False) | register.filter(linebreaksbr) | ||||||
| register_filter('linenumbers', linenumbers, False) | register.filter(linenumbers) | ||||||
| register_filter('ljust', ljust, True) | register.filter(ljust) | ||||||
| register_filter('lower', lower, False) | register.filter(lower) | ||||||
| register_filter('make_list', make_list, False) | register.filter(make_list) | ||||||
| register_filter('phone2numeric', phone2numeric, False) | register.filter(phone2numeric) | ||||||
| register_filter('pluralize', pluralize, False) | register.filter(pluralize) | ||||||
| register_filter('pprint', pprint, False) | register.filter(pprint) | ||||||
| register_filter('removetags', removetags, True) | register.filter(removetags) | ||||||
| register_filter('random', random, False) | register.filter(random) | ||||||
| register_filter('rjust', rjust, True) | register.filter(rjust) | ||||||
| register_filter('slice', slice_, True) | register.filter(slice_) | ||||||
| register_filter('slugify', slugify, False) | register.filter(slugify) | ||||||
| register_filter('stringformat', stringformat, True) | register.filter(stringformat) | ||||||
| register_filter('striptags', striptags, False) | register.filter(striptags) | ||||||
| register_filter('time', time, True) | register.filter(time) | ||||||
| register_filter('timesince', timesince, False) | register.filter(timesince) | ||||||
| register_filter('title', title, False) | register.filter(title) | ||||||
| register_filter('truncatewords', truncatewords, True) | register.filter(truncatewords) | ||||||
| register_filter('unordered_list', unordered_list, False) | register.filter(unordered_list) | ||||||
| register_filter('upper', upper, False) | register.filter(upper) | ||||||
| register_filter('urlencode', urlencode, False) | register.filter(urlencode) | ||||||
| register_filter('urlize', urlize, False) | register.filter(urlize) | ||||||
| register_filter('urlizetrunc', urlizetrunc, True) | register.filter(urlizetrunc) | ||||||
| register_filter('wordcount', wordcount, False) | register.filter(wordcount) | ||||||
| register_filter('wordwrap', wordwrap, True) | register.filter(wordwrap) | ||||||
| register_filter('yesno', yesno, True) | register.filter(yesno) | ||||||
| @@ -1,9 +1,12 @@ | |||||||
| "Default tags used by the template system, available to all templates." | "Default tags used by the template system, available to all templates." | ||||||
|  |  | ||||||
| from django.core.template import Node, NodeList, Template, Context, resolve_variable, resolve_variable_with_filters, get_filters_from_token, registered_filters | from django.core.template import Node, NodeList, Template, Context, resolve_variable | ||||||
| from django.core.template import TemplateSyntaxError, VariableDoesNotExist, BLOCK_TAG_START, BLOCK_TAG_END, VARIABLE_TAG_START, VARIABLE_TAG_END, register_tag | from django.core.template import TemplateSyntaxError, VariableDoesNotExist, BLOCK_TAG_START, BLOCK_TAG_END, VARIABLE_TAG_START, VARIABLE_TAG_END | ||||||
|  | from django.core.template import get_library, Library, InvalidTemplateLibrary | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
|  | register = Library() | ||||||
|  |  | ||||||
| class CommentNode(Node): | class CommentNode(Node): | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         return '' |         return '' | ||||||
| @@ -27,15 +30,13 @@ class DebugNode(Node): | |||||||
|         return ''.join(output) |         return ''.join(output) | ||||||
|  |  | ||||||
| class FilterNode(Node): | class FilterNode(Node): | ||||||
|     def __init__(self, filters, nodelist): |     def __init__(self, filter_expr, nodelist): | ||||||
|         self.filters, self.nodelist = filters, nodelist |         self.filter_expr, self.nodelist = filter_expr, nodelist | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         output = self.nodelist.render(context) |         output = self.nodelist.render(context) | ||||||
|         # apply filters |         # apply filters | ||||||
|         for f in self.filters: |         return self.filter_expr.resolve(Context({'var': output})) | ||||||
|             output = registered_filters[f[0]][0](output, f[1]) |  | ||||||
|         return output |  | ||||||
|  |  | ||||||
| class FirstOfNode(Node): | class FirstOfNode(Node): | ||||||
|     def __init__(self, vars): |     def __init__(self, vars): | ||||||
| @@ -81,7 +82,7 @@ class ForNode(Node): | |||||||
|             parentloop = {} |             parentloop = {} | ||||||
|         context.push() |         context.push() | ||||||
|         try: |         try: | ||||||
|             values = resolve_variable_with_filters(self.sequence, context) |             values = self.sequence.resolve(context) | ||||||
|         except VariableDoesNotExist: |         except VariableDoesNotExist: | ||||||
|             values = [] |             values = [] | ||||||
|         if values is None: |         if values is None: | ||||||
| @@ -147,8 +148,8 @@ class IfEqualNode(Node): | |||||||
|         return self.nodelist_false.render(context) |         return self.nodelist_false.render(context) | ||||||
|  |  | ||||||
| class IfNode(Node): | class IfNode(Node): | ||||||
|     def __init__(self, boolvars, nodelist_true, nodelist_false): |     def __init__(self, bool_exprs, nodelist_true, nodelist_false): | ||||||
|         self.boolvars = boolvars |         self.bool_exprs = bool_exprs | ||||||
|         self.nodelist_true, self.nodelist_false = nodelist_true, nodelist_false |         self.nodelist_true, self.nodelist_false = nodelist_true, nodelist_false | ||||||
|  |  | ||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
| @@ -169,9 +170,9 @@ class IfNode(Node): | |||||||
|         return nodes |         return nodes | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         for ifnot, boolvar in self.boolvars: |         for ifnot, bool_expr in self.bool_exprs: | ||||||
|             try: |             try: | ||||||
|                 value = resolve_variable_with_filters(boolvar, context) |                 value = bool_expr.resolve(context) | ||||||
|             except VariableDoesNotExist: |             except VariableDoesNotExist: | ||||||
|                 value = None |                 value = None | ||||||
|             if (value and not ifnot) or (ifnot and not value): |             if (value and not ifnot) or (ifnot and not value): | ||||||
| @@ -179,19 +180,18 @@ class IfNode(Node): | |||||||
|         return self.nodelist_false.render(context) |         return self.nodelist_false.render(context) | ||||||
|  |  | ||||||
| class RegroupNode(Node): | class RegroupNode(Node): | ||||||
|     def __init__(self, target_var, expression, var_name): |     def __init__(self, target, expression, var_name): | ||||||
|         self.target_var, self.expression = target_var, expression |         self.target, self.expression = target, expression | ||||||
|         self.var_name = var_name |         self.var_name = var_name | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         obj_list = resolve_variable_with_filters(self.target_var, context) |         obj_list = self.target.resolve(context) | ||||||
|         if obj_list == '': # target_var wasn't found in context; fail silently |         if obj_list == '': # target_var wasn't found in context; fail silently | ||||||
|             context[self.var_name] = [] |             context[self.var_name] = [] | ||||||
|             return '' |             return '' | ||||||
|         output = [] # list of dictionaries in the format {'grouper': 'key', 'list': [list of contents]} |         output = [] # list of dictionaries in the format {'grouper': 'key', 'list': [list of contents]} | ||||||
|         for obj in obj_list: |         for obj in obj_list: | ||||||
|             grouper = resolve_variable_with_filters('var.%s' % self.expression, \ |             grouper = self.expression.resolve(Context({'var': obj})) | ||||||
|                 Context({'var': obj})) |  | ||||||
|             # TODO: Is this a sensible way to determine equality? |             # TODO: Is this a sensible way to determine equality? | ||||||
|             if output and repr(output[-1]['grouper']) == repr(grouper): |             if output and repr(output[-1]['grouper']) == repr(grouper): | ||||||
|                 output[-1]['list'].append(obj) |                 output[-1]['list'].append(obj) | ||||||
| @@ -236,21 +236,7 @@ class SsiNode(Node): | |||||||
|         return output |         return output | ||||||
|  |  | ||||||
| class LoadNode(Node): | class LoadNode(Node): | ||||||
|     def __init__(self, taglib): |  | ||||||
|         self.taglib = taglib |  | ||||||
|  |  | ||||||
|     def load_taglib(taglib): |  | ||||||
|         mod = __import__("django.templatetags.%s" % taglib.split('.')[-1], '', '', ['']) |  | ||||||
|         reload(mod) |  | ||||||
|         return mod |  | ||||||
|     load_taglib = staticmethod(load_taglib) |  | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         "Import the relevant module" |  | ||||||
|         try: |  | ||||||
|             self.__class__.load_taglib(self.taglib) |  | ||||||
|         except ImportError: |  | ||||||
|             pass # Fail silently for invalid loads. |  | ||||||
|         return '' |         return '' | ||||||
|  |  | ||||||
| class NowNode(Node): | class NowNode(Node): | ||||||
| @@ -276,15 +262,15 @@ class TemplateTagNode(Node): | |||||||
|         return self.mapping.get(self.tagtype, '') |         return self.mapping.get(self.tagtype, '') | ||||||
|  |  | ||||||
| class WidthRatioNode(Node): | class WidthRatioNode(Node): | ||||||
|     def __init__(self, val_var, max_var, max_width): |     def __init__(self, val_expr, max_expr, max_width): | ||||||
|         self.val_var = val_var |         self.val_expr = val_expr | ||||||
|         self.max_var = max_var |         self.max_expr = max_expr | ||||||
|         self.max_width = max_width |         self.max_width = max_width | ||||||
|  |  | ||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         try: |         try: | ||||||
|             value = resolve_variable_with_filters(self.val_var, context) |             value = self.val_expr.resolve(context) | ||||||
|             maxvalue = resolve_variable_with_filters(self.max_var, context) |             maxvalue = self.max_expr.resolve(context) | ||||||
|         except VariableDoesNotExist: |         except VariableDoesNotExist: | ||||||
|             return '' |             return '' | ||||||
|         try: |         try: | ||||||
| @@ -295,15 +281,18 @@ class WidthRatioNode(Node): | |||||||
|             return '' |             return '' | ||||||
|         return str(int(round(ratio))) |         return str(int(round(ratio))) | ||||||
|  |  | ||||||
| def do_comment(parser, token): | #@register.tag | ||||||
|  | def comment(parser, token): | ||||||
|     """ |     """ | ||||||
|     Ignore everything between ``{% comment %}`` and ``{% endcomment %}`` |     Ignore everything between ``{% comment %}`` and ``{% endcomment %}`` | ||||||
|     """ |     """ | ||||||
|     nodelist = parser.parse(('endcomment',)) |     nodelist = parser.parse(('endcomment',)) | ||||||
|     parser.delete_first_token() |     parser.delete_first_token() | ||||||
|     return CommentNode() |     return CommentNode() | ||||||
|  | comment = register.tag(comment) | ||||||
|  |  | ||||||
| def do_cycle(parser, token): | #@register.tag | ||||||
|  | def cycle(parser, token): | ||||||
|     """ |     """ | ||||||
|     Cycle among the given strings each time this tag is encountered |     Cycle among the given strings each time this tag is encountered | ||||||
|  |  | ||||||
| @@ -369,11 +358,9 @@ def do_cycle(parser, token): | |||||||
|  |  | ||||||
|     else: |     else: | ||||||
|         raise TemplateSyntaxError("Invalid arguments to 'cycle': %s" % args) |         raise TemplateSyntaxError("Invalid arguments to 'cycle': %s" % args) | ||||||
|  | cycle = register.tag(cycle) | ||||||
|  |  | ||||||
| def do_debug(parser, token): | #@register.tag(name="filter") | ||||||
|     "Print a whole load of debugging information, including the context and imported modules" |  | ||||||
|     return DebugNode() |  | ||||||
|  |  | ||||||
| def do_filter(parser, token): | def do_filter(parser, token): | ||||||
|     """ |     """ | ||||||
|     Filter the contents of the blog through variable filters. |     Filter the contents of the blog through variable filters. | ||||||
| @@ -388,12 +375,14 @@ def do_filter(parser, token): | |||||||
|         {% endfilter %} |         {% endfilter %} | ||||||
|     """ |     """ | ||||||
|     _, rest = token.contents.split(None, 1) |     _, rest = token.contents.split(None, 1) | ||||||
|     _, filters = get_filters_from_token('var|%s' % rest) |     filter_expr = parser.compile_filter("var|%s" % (rest)) | ||||||
|     nodelist = parser.parse(('endfilter',)) |     nodelist = parser.parse(('endfilter',)) | ||||||
|     parser.delete_first_token() |     parser.delete_first_token() | ||||||
|     return FilterNode(filters, nodelist) |     return FilterNode(filter_expr, nodelist) | ||||||
|  | filter = register.tag("filter", do_filter) | ||||||
|  |  | ||||||
| def do_firstof(parser, token): | #@register.tag | ||||||
|  | def firstof(parser, token): | ||||||
|     """ |     """ | ||||||
|     Outputs the first variable passed that is not False. |     Outputs the first variable passed that is not False. | ||||||
|  |  | ||||||
| @@ -419,8 +408,9 @@ def do_firstof(parser, token): | |||||||
|     if len(bits) < 1: |     if len(bits) < 1: | ||||||
|         raise TemplateSyntaxError, "'firstof' statement requires at least one argument" |         raise TemplateSyntaxError, "'firstof' statement requires at least one argument" | ||||||
|     return FirstOfNode(bits) |     return FirstOfNode(bits) | ||||||
|  | firstof = register.tag(firstof) | ||||||
|  |  | ||||||
|  | #@register.tag(name="for") | ||||||
| def do_for(parser, token): | def do_for(parser, token): | ||||||
|     """ |     """ | ||||||
|     Loop over each item in an array. |     Loop over each item in an array. | ||||||
| @@ -462,11 +452,12 @@ def do_for(parser, token): | |||||||
|     if bits[2] != 'in': |     if bits[2] != 'in': | ||||||
|         raise TemplateSyntaxError, "'for' statement must contain 'in' as the second word: %s" % token.contents |         raise TemplateSyntaxError, "'for' statement must contain 'in' as the second word: %s" % token.contents | ||||||
|     loopvar = bits[1] |     loopvar = bits[1] | ||||||
|     sequence = bits[3] |     sequence = parser.compile_filter(bits[3]) | ||||||
|     reversed = (len(bits) == 5) |     reversed = (len(bits) == 5) | ||||||
|     nodelist_loop = parser.parse(('endfor',)) |     nodelist_loop = parser.parse(('endfor',)) | ||||||
|     parser.delete_first_token() |     parser.delete_first_token() | ||||||
|     return ForNode(loopvar, sequence, reversed, nodelist_loop) |     return ForNode(loopvar, sequence, reversed, nodelist_loop) | ||||||
|  | do_for = register.tag("for", do_for) | ||||||
|  |  | ||||||
| def do_ifequal(parser, token, negate): | def do_ifequal(parser, token, negate): | ||||||
|     """ |     """ | ||||||
| @@ -497,6 +488,17 @@ def do_ifequal(parser, token, negate): | |||||||
|         nodelist_false = NodeList() |         nodelist_false = NodeList() | ||||||
|     return IfEqualNode(bits[1], bits[2], nodelist_true, nodelist_false, negate) |     return IfEqualNode(bits[1], bits[2], nodelist_true, nodelist_false, negate) | ||||||
|  |  | ||||||
|  | #@register.tag | ||||||
|  | def ifequal(parser, token): | ||||||
|  |     return do_ifequal(parser, token, False) | ||||||
|  | ifequal = register.tag(ifequal) | ||||||
|  |  | ||||||
|  | #@register.tag | ||||||
|  | def ifnotequal(parser, token): | ||||||
|  |     return do_ifequal(parser, token, True) | ||||||
|  | ifnotequal = register.tag(ifnotequal) | ||||||
|  |  | ||||||
|  | #@register.tag(name="if") | ||||||
| def do_if(parser, token): | def do_if(parser, token): | ||||||
|     """ |     """ | ||||||
|     The ``{% if %}`` tag evaluates a variable, and if that variable is "true" |     The ``{% if %}`` tag evaluates a variable, and if that variable is "true" | ||||||
| @@ -554,9 +556,9 @@ def do_if(parser, token): | |||||||
|             not_, boolvar = boolpair.split() |             not_, boolvar = boolpair.split() | ||||||
|             if not_ != 'not': |             if not_ != 'not': | ||||||
|                 raise TemplateSyntaxError, "Expected 'not' in if statement" |                 raise TemplateSyntaxError, "Expected 'not' in if statement" | ||||||
|             boolvars.append((True, boolvar)) |             boolvars.append((True, parser.compile_filter(boolvar))) | ||||||
|         else: |         else: | ||||||
|             boolvars.append((False, boolpair)) |             boolvars.append((False, parser.compile_filter(boolpair))) | ||||||
|     nodelist_true = parser.parse(('else', 'endif')) |     nodelist_true = parser.parse(('else', 'endif')) | ||||||
|     token = parser.next_token() |     token = parser.next_token() | ||||||
|     if token.contents == 'else': |     if token.contents == 'else': | ||||||
| @@ -565,8 +567,10 @@ def do_if(parser, token): | |||||||
|     else: |     else: | ||||||
|         nodelist_false = NodeList() |         nodelist_false = NodeList() | ||||||
|     return IfNode(boolvars, nodelist_true, nodelist_false) |     return IfNode(boolvars, nodelist_true, nodelist_false) | ||||||
|  | do_if = register.tag("if", do_if) | ||||||
|  |  | ||||||
| def do_ifchanged(parser, token): | #@register.tag | ||||||
|  | def ifchanged(parser, token): | ||||||
|     """ |     """ | ||||||
|     Check if a value has changed from the last iteration of a loop. |     Check if a value has changed from the last iteration of a loop. | ||||||
|  |  | ||||||
| @@ -587,8 +591,10 @@ def do_ifchanged(parser, token): | |||||||
|     nodelist = parser.parse(('endifchanged',)) |     nodelist = parser.parse(('endifchanged',)) | ||||||
|     parser.delete_first_token() |     parser.delete_first_token() | ||||||
|     return IfChangedNode(nodelist) |     return IfChangedNode(nodelist) | ||||||
|  | ifchanged = register.tag(ifchanged) | ||||||
|  |  | ||||||
| def do_ssi(parser, token): | #@register.tag | ||||||
|  | def ssi(parser, token): | ||||||
|     """ |     """ | ||||||
|     Output the contents of a given file into the page. |     Output the contents of a given file into the page. | ||||||
|  |  | ||||||
| @@ -613,8 +619,10 @@ def do_ssi(parser, token): | |||||||
|         else: |         else: | ||||||
|             raise TemplateSyntaxError, "Second (optional) argument to %s tag must be 'parsed'" % bits[0] |             raise TemplateSyntaxError, "Second (optional) argument to %s tag must be 'parsed'" % bits[0] | ||||||
|     return SsiNode(bits[1], parsed) |     return SsiNode(bits[1], parsed) | ||||||
|  | ssi = register.tag(ssi) | ||||||
|  |  | ||||||
| def do_load(parser, token): | #@register.tag | ||||||
|  | def load(parser, token): | ||||||
|     """ |     """ | ||||||
|     Load a custom template tag set. |     Load a custom template tag set. | ||||||
|  |  | ||||||
| @@ -623,17 +631,18 @@ def do_load(parser, token): | |||||||
|         {% load news.photos %} |         {% load news.photos %} | ||||||
|     """ |     """ | ||||||
|     bits = token.contents.split() |     bits = token.contents.split() | ||||||
|     if len(bits) != 2: |     for taglib in bits[1:]: | ||||||
|         raise TemplateSyntaxError, "'load' statement takes one argument" |         # add the library to the parser | ||||||
|     taglib = bits[1] |  | ||||||
|     # check at compile time that the module can be imported |  | ||||||
|         try: |         try: | ||||||
|         LoadNode.load_taglib(taglib) |             lib = get_library("django.templatetags.%s" % taglib.split('.')[-1]) | ||||||
|     except ImportError, e: |             parser.add_library(lib) | ||||||
|  |         except InvalidTemplateLibrary, e: | ||||||
|             raise TemplateSyntaxError, "'%s' is not a valid tag library: %s" % (taglib, e) |             raise TemplateSyntaxError, "'%s' is not a valid tag library: %s" % (taglib, e) | ||||||
|     return LoadNode(taglib) |     return LoadNode() | ||||||
|  | load = register.tag(load) | ||||||
|  |  | ||||||
| def do_now(parser, token): | #@register.tag | ||||||
|  | def now(parser, token): | ||||||
|     """ |     """ | ||||||
|     Display the date, formatted according to the given string. |     Display the date, formatted according to the given string. | ||||||
|  |  | ||||||
| @@ -649,8 +658,10 @@ def do_now(parser, token): | |||||||
|         raise TemplateSyntaxError, "'now' statement takes one argument" |         raise TemplateSyntaxError, "'now' statement takes one argument" | ||||||
|     format_string = bits[1] |     format_string = bits[1] | ||||||
|     return NowNode(format_string) |     return NowNode(format_string) | ||||||
|  | now = register.tag(now) | ||||||
|  |  | ||||||
| def do_regroup(parser, token): | #@register.tag | ||||||
|  | def regroup(parser, token): | ||||||
|     """ |     """ | ||||||
|     Regroup a list of alike objects by a common attribute. |     Regroup a list of alike objects by a common attribute. | ||||||
|  |  | ||||||
| @@ -699,17 +710,21 @@ def do_regroup(parser, token): | |||||||
|     firstbits = token.contents.split(None, 3) |     firstbits = token.contents.split(None, 3) | ||||||
|     if len(firstbits) != 4: |     if len(firstbits) != 4: | ||||||
|         raise TemplateSyntaxError, "'regroup' tag takes five arguments" |         raise TemplateSyntaxError, "'regroup' tag takes five arguments" | ||||||
|     target_var = firstbits[1] |     target = parser.compile_filter(firstbits[1]) | ||||||
|     if firstbits[2] != 'by': |     if firstbits[2] != 'by': | ||||||
|         raise TemplateSyntaxError, "second argument to 'regroup' tag must be 'by'" |         raise TemplateSyntaxError, "second argument to 'regroup' tag must be 'by'" | ||||||
|     lastbits_reversed = firstbits[3][::-1].split(None, 2) |     lastbits_reversed = firstbits[3][::-1].split(None, 2) | ||||||
|     if lastbits_reversed[1][::-1] != 'as': |     if lastbits_reversed[1][::-1] != 'as': | ||||||
|         raise TemplateSyntaxError, "next-to-last argument to 'regroup' tag must be 'as'" |         raise TemplateSyntaxError, "next-to-last argument to 'regroup' tag must be 'as'" | ||||||
|     expression = lastbits_reversed[2][::-1] |  | ||||||
|     var_name = lastbits_reversed[0][::-1] |  | ||||||
|     return RegroupNode(target_var, expression, var_name) |  | ||||||
|  |  | ||||||
| def do_templatetag(parser, token): |     expression = parser.compile_filters('var.%s' % lastbits_reversed[2][::-1]) | ||||||
|  |  | ||||||
|  |     var_name = lastbits_reversed[0][::-1] | ||||||
|  |     return RegroupNode(target, expression, var_name) | ||||||
|  | regroup = register.tag(regroup) | ||||||
|  |  | ||||||
|  | #@register.tag | ||||||
|  | def templatetag(parser, token): | ||||||
|     """ |     """ | ||||||
|     Output one of the bits used to compose template tags. |     Output one of the bits used to compose template tags. | ||||||
|  |  | ||||||
| @@ -735,8 +750,10 @@ def do_templatetag(parser, token): | |||||||
|         raise TemplateSyntaxError, "Invalid templatetag argument: '%s'. Must be one of: %s" % \ |         raise TemplateSyntaxError, "Invalid templatetag argument: '%s'. Must be one of: %s" % \ | ||||||
|             (tag, TemplateTagNode.mapping.keys()) |             (tag, TemplateTagNode.mapping.keys()) | ||||||
|     return TemplateTagNode(tag) |     return TemplateTagNode(tag) | ||||||
|  | templatetag = register.tag(templatetag) | ||||||
|  |  | ||||||
| def do_widthratio(parser, token): | @register.tag | ||||||
|  | def widthratio(parser, token): | ||||||
|     """ |     """ | ||||||
|     For creating bar charts and such, this tag calculates the ratio of a given |     For creating bar charts and such, this tag calculates the ratio of a given | ||||||
|     value to a maximum value, and then applies that ratio to a constant. |     value to a maximum value, and then applies that ratio to a constant. | ||||||
| @@ -752,26 +769,11 @@ def do_widthratio(parser, token): | |||||||
|     bits = token.contents.split() |     bits = token.contents.split() | ||||||
|     if len(bits) != 4: |     if len(bits) != 4: | ||||||
|         raise TemplateSyntaxError("widthratio takes three arguments") |         raise TemplateSyntaxError("widthratio takes three arguments") | ||||||
|     tag, this_value_var, max_value_var, max_width = bits |     tag, this_value_expr, max_value_expr, max_width = bits | ||||||
|     try: |     try: | ||||||
|         max_width = int(max_width) |         max_width = int(max_width) | ||||||
|     except ValueError: |     except ValueError: | ||||||
|         raise TemplateSyntaxError("widthratio final argument must be an integer") |         raise TemplateSyntaxError("widthratio final argument must be an integer") | ||||||
|     return WidthRatioNode(this_value_var, max_value_var, max_width) |     return WidthRatioNode(parser.compile_filter(this_value_expr), | ||||||
|  |                           parser.compile_filter(max_value_expr), max_width) | ||||||
| register_tag('comment', do_comment) | widthratio = register.tag(widthratio) | ||||||
| register_tag('cycle', do_cycle) |  | ||||||
| register_tag('debug', do_debug) |  | ||||||
| register_tag('filter', do_filter) |  | ||||||
| register_tag('firstof', do_firstof) |  | ||||||
| register_tag('for', do_for) |  | ||||||
| register_tag('ifequal', lambda parser, token: do_ifequal(parser, token, False)) |  | ||||||
| register_tag('ifnotequal', lambda parser, token: do_ifequal(parser, token, True)) |  | ||||||
| register_tag('if', do_if) |  | ||||||
| register_tag('ifchanged', do_ifchanged) |  | ||||||
| register_tag('regroup', do_regroup) |  | ||||||
| register_tag('ssi', do_ssi) |  | ||||||
| register_tag('load', do_load) |  | ||||||
| register_tag('now', do_now) |  | ||||||
| register_tag('templatetag', do_templatetag) |  | ||||||
| register_tag('widthratio', do_widthratio) |  | ||||||
|   | |||||||
| @@ -21,7 +21,7 @@ | |||||||
| # installed, because pkg_resources is necessary to read eggs. | # installed, because pkg_resources is necessary to read eggs. | ||||||
|  |  | ||||||
| from django.core.exceptions import ImproperlyConfigured | from django.core.exceptions import ImproperlyConfigured | ||||||
| from django.core.template import Origin, StringOrigin, Template, Context, Node, TemplateDoesNotExist, TemplateSyntaxError, resolve_variable_with_filters, resolve_variable, register_tag | from django.core.template import Origin, StringOrigin, Template,  TemplateDoesNotExist, add_to_builtins | ||||||
| from django.conf.settings import TEMPLATE_LOADERS, TEMPLATE_DEBUG | from django.conf.settings import TEMPLATE_LOADERS, TEMPLATE_DEBUG | ||||||
|  |  | ||||||
| template_source_loaders = [] | template_source_loaders = [] | ||||||
| @@ -68,9 +68,6 @@ def find_template_source(name, dirs=None): | |||||||
| def load_template_source(name, dirs=None): | def load_template_source(name, dirs=None): | ||||||
|     find_template_source(name, dirs)[0] |     find_template_source(name, dirs)[0] | ||||||
|  |  | ||||||
| class ExtendsError(Exception): |  | ||||||
|     pass |  | ||||||
|  |  | ||||||
| def get_template(template_name): | def get_template(template_name): | ||||||
|     """ |     """ | ||||||
|     Returns a compiled Template object for the given template name, |     Returns a compiled Template object for the given template name, | ||||||
| @@ -113,166 +110,4 @@ def select_template(template_name_list): | |||||||
|     # If we get here, none of the templates could be loaded |     # If we get here, none of the templates could be loaded | ||||||
|     raise TemplateDoesNotExist, ', '.join(template_name_list) |     raise TemplateDoesNotExist, ', '.join(template_name_list) | ||||||
|  |  | ||||||
| class BlockNode(Node): | add_to_builtins('django.core.template.loader_tags') | ||||||
|     def __init__(self, name, nodelist, parent=None): |  | ||||||
|         self.name, self.nodelist, self.parent = name, nodelist, parent |  | ||||||
|  |  | ||||||
|     def __repr__(self): |  | ||||||
|         return "<Block Node: %s. Contents: %r>" % (self.name, self.nodelist) |  | ||||||
|  |  | ||||||
|     def render(self, context): |  | ||||||
|         context.push() |  | ||||||
|         # Save context in case of block.super(). |  | ||||||
|         self.context = context |  | ||||||
|         context['block'] = self |  | ||||||
|         result = self.nodelist.render(context) |  | ||||||
|         context.pop() |  | ||||||
|         return result |  | ||||||
|  |  | ||||||
|     def super(self): |  | ||||||
|         if self.parent: |  | ||||||
|             return self.parent.render(self.context) |  | ||||||
|         return '' |  | ||||||
|  |  | ||||||
|     def add_parent(self, nodelist): |  | ||||||
|         if self.parent: |  | ||||||
|             self.parent.add_parent(nodelist) |  | ||||||
|         else: |  | ||||||
|             self.parent = BlockNode(self.name, nodelist) |  | ||||||
|  |  | ||||||
| class ExtendsNode(Node): |  | ||||||
|     def __init__(self, nodelist, parent_name, parent_name_var, template_dirs=None): |  | ||||||
|         self.nodelist = nodelist |  | ||||||
|         self.parent_name, self.parent_name_var = parent_name, parent_name_var |  | ||||||
|         self.template_dirs = template_dirs |  | ||||||
|  |  | ||||||
|     def get_parent(self, context): |  | ||||||
|         if self.parent_name_var: |  | ||||||
|             self.parent_name = resolve_variable_with_filters(self.parent_name_var, context) |  | ||||||
|         parent = self.parent_name |  | ||||||
|         if not parent: |  | ||||||
|             error_msg = "Invalid template name in 'extends' tag: %r." % parent |  | ||||||
|             if self.parent_name_var: |  | ||||||
|                 error_msg += " Got this from the %r variable." % self.parent_name_var |  | ||||||
|             raise TemplateSyntaxError, error_msg |  | ||||||
|         try: |  | ||||||
|             return get_template_from_string(*find_template_source(parent, self.template_dirs)) |  | ||||||
|         except TemplateDoesNotExist: |  | ||||||
|             raise TemplateSyntaxError, "Template %r cannot be extended, because it doesn't exist" % parent |  | ||||||
|  |  | ||||||
|     def render(self, context): |  | ||||||
|         compiled_parent = self.get_parent(context) |  | ||||||
|         parent_is_child = isinstance(compiled_parent.nodelist[0], ExtendsNode) |  | ||||||
|         parent_blocks = dict([(n.name, n) for n in compiled_parent.nodelist.get_nodes_by_type(BlockNode)]) |  | ||||||
|         for block_node in self.nodelist.get_nodes_by_type(BlockNode): |  | ||||||
|             # Check for a BlockNode with this node's name, and replace it if found. |  | ||||||
|             try: |  | ||||||
|                 parent_block = parent_blocks[block_node.name] |  | ||||||
|             except KeyError: |  | ||||||
|                 # This BlockNode wasn't found in the parent template, but the |  | ||||||
|                 # parent block might be defined in the parent's *parent*, so we |  | ||||||
|                 # add this BlockNode to the parent's ExtendsNode nodelist, so |  | ||||||
|                 # it'll be checked when the parent node's render() is called. |  | ||||||
|                 if parent_is_child: |  | ||||||
|                     compiled_parent.nodelist[0].nodelist.append(block_node) |  | ||||||
|             else: |  | ||||||
|                 # Keep any existing parents and add a new one. Used by BlockNode. |  | ||||||
|                 parent_block.parent = block_node.parent |  | ||||||
|                 parent_block.add_parent(parent_block.nodelist) |  | ||||||
|                 parent_block.nodelist = block_node.nodelist |  | ||||||
|         return compiled_parent.render(context) |  | ||||||
|  |  | ||||||
| class ConstantIncludeNode(Node): |  | ||||||
|     def __init__(self, template_path): |  | ||||||
|         try: |  | ||||||
|             t = get_template(template_path) |  | ||||||
|             self.template = t |  | ||||||
|         except Exception, e: |  | ||||||
|             if TEMPLATE_DEBUG: |  | ||||||
|                 raise |  | ||||||
|             self.template = None |  | ||||||
|  |  | ||||||
|     def render(self, context): |  | ||||||
|         if self.template: |  | ||||||
|             return self.template.render(context) |  | ||||||
|         else: |  | ||||||
|             return '' |  | ||||||
|  |  | ||||||
| class IncludeNode(Node): |  | ||||||
|     def __init__(self, template_name): |  | ||||||
|         self.template_name = template_name |  | ||||||
|  |  | ||||||
|     def render(self, context): |  | ||||||
|          try: |  | ||||||
|              template_name = resolve_variable(self.template_name, context) |  | ||||||
|              t = get_template(template_name) |  | ||||||
|              return t.render(context) |  | ||||||
|          except TemplateSyntaxError, e: |  | ||||||
|              if TEMPLATE_DEBUG: |  | ||||||
|                  raise |  | ||||||
|              return '' |  | ||||||
|          except: |  | ||||||
|              return '' # Fail silently for invalid included templates. |  | ||||||
|  |  | ||||||
| def do_block(parser, token): |  | ||||||
|     """ |  | ||||||
|     Define a block that can be overridden by child templates. |  | ||||||
|     """ |  | ||||||
|     bits = token.contents.split() |  | ||||||
|     if len(bits) != 2: |  | ||||||
|         raise TemplateSyntaxError, "'%s' tag takes only one argument" % bits[0] |  | ||||||
|     block_name = bits[1] |  | ||||||
|     # Keep track of the names of BlockNodes found in this template, so we can |  | ||||||
|     # check for duplication. |  | ||||||
|     try: |  | ||||||
|         if block_name in parser.__loaded_blocks: |  | ||||||
|             raise TemplateSyntaxError, "'%s' tag with name '%s' appears more than once" % (bits[0], block_name) |  | ||||||
|         parser.__loaded_blocks.append(block_name) |  | ||||||
|     except AttributeError: # parser._loaded_blocks isn't a list yet |  | ||||||
|         parser.__loaded_blocks = [block_name] |  | ||||||
|     nodelist = parser.parse(('endblock',)) |  | ||||||
|     parser.delete_first_token() |  | ||||||
|     return BlockNode(block_name, nodelist) |  | ||||||
|  |  | ||||||
| def do_extends(parser, token): |  | ||||||
|     """ |  | ||||||
|     Signal that this template extends a parent template. |  | ||||||
|  |  | ||||||
|     This tag may be used in two ways: ``{% extends "base" %}`` (with quotes) |  | ||||||
|     uses the literal value "base" as the name of the parent template to extend, |  | ||||||
|     or ``{% extends variable %}`` uses the value of ``variable`` as the name |  | ||||||
|     of the parent template to extend. |  | ||||||
|     """ |  | ||||||
|     bits = token.contents.split() |  | ||||||
|     if len(bits) != 2: |  | ||||||
|         raise TemplateSyntaxError, "'%s' takes one argument" % bits[0] |  | ||||||
|     parent_name, parent_name_var = None, None |  | ||||||
|     if bits[1][0] in ('"', "'") and bits[1][-1] == bits[1][0]: |  | ||||||
|         parent_name = bits[1][1:-1] |  | ||||||
|     else: |  | ||||||
|         parent_name_var = bits[1] |  | ||||||
|     nodelist = parser.parse() |  | ||||||
|     if nodelist.get_nodes_by_type(ExtendsNode): |  | ||||||
|         raise TemplateSyntaxError, "'%s' cannot appear more than once in the same template" % bits[0] |  | ||||||
|     return ExtendsNode(nodelist, parent_name, parent_name_var) |  | ||||||
|  |  | ||||||
| def do_include(parser, token): |  | ||||||
|     """ |  | ||||||
|     Loads a template and renders it with the current context. |  | ||||||
|  |  | ||||||
|     Example:: |  | ||||||
|  |  | ||||||
|         {% include "foo/some_include" %} |  | ||||||
|     """ |  | ||||||
|  |  | ||||||
|     bits = token.contents.split() |  | ||||||
|     if len(bits) != 2: |  | ||||||
|         raise TemplateSyntaxError, "%r tag takes one argument: the name of the template to be included" % bits[0] |  | ||||||
|     path = bits[1] |  | ||||||
|     if path[0] in ('"', "'") and path[-1] == path[0]: |  | ||||||
|         return ConstantIncludeNode(path[1:-1]) |  | ||||||
|     return IncludeNode(bits[1]) |  | ||||||
|  |  | ||||||
| register_tag('block', do_block) |  | ||||||
| register_tag('extends', do_extends) |  | ||||||
| register_tag('include', do_include) |  | ||||||
|   | |||||||
							
								
								
									
										172
									
								
								django/core/template/loader_tags.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								django/core/template/loader_tags.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | |||||||
|  | from django.core.template import TemplateSyntaxError, TemplateDoesNotExist, resolve_variable | ||||||
|  | from django.core.template import Library, Context, Node | ||||||
|  | from django.core.template.loader import get_template, get_template_from_string, find_template_source | ||||||
|  | from django.conf.settings import TEMPLATE_DEBUG | ||||||
|  | register = Library() | ||||||
|  |  | ||||||
|  | class ExtendsError(Exception): | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  | class BlockNode(Node): | ||||||
|  |     def __init__(self, name, nodelist, parent=None): | ||||||
|  |         self.name, self.nodelist, self.parent = name, nodelist, parent | ||||||
|  |  | ||||||
|  |     def __repr__(self): | ||||||
|  |         return "<Block Node: %s. Contents: %r>" % (self.name, self.nodelist) | ||||||
|  |  | ||||||
|  |     def render(self, context): | ||||||
|  |         context.push() | ||||||
|  |         # Save context in case of block.super(). | ||||||
|  |         self.context = context | ||||||
|  |         context['block'] = self | ||||||
|  |         result = self.nodelist.render(context) | ||||||
|  |         context.pop() | ||||||
|  |         return result | ||||||
|  |  | ||||||
|  |     def super(self): | ||||||
|  |         if self.parent: | ||||||
|  |             return self.parent.render(self.context) | ||||||
|  |         return '' | ||||||
|  |  | ||||||
|  |     def add_parent(self, nodelist): | ||||||
|  |         if self.parent: | ||||||
|  |             self.parent.add_parent(nodelist) | ||||||
|  |         else: | ||||||
|  |             self.parent = BlockNode(self.name, nodelist) | ||||||
|  |  | ||||||
|  | class ExtendsNode(Node): | ||||||
|  |     def __init__(self, nodelist, parent_name, parent_name_expr, template_dirs=None): | ||||||
|  |         self.nodelist = nodelist | ||||||
|  |         self.parent_name, self.parent_name_expr = parent_name, parent_name_expr | ||||||
|  |         self.template_dirs = template_dirs | ||||||
|  |  | ||||||
|  |     def get_parent(self, context): | ||||||
|  |         if self.parent_name_expr: | ||||||
|  |             self.parent_name = self.parent_name_expr.resolve(context) | ||||||
|  |         parent = self.parent_name | ||||||
|  |         if not parent: | ||||||
|  |             error_msg = "Invalid template name in 'extends' tag: %r." % parent | ||||||
|  |             if self.parent_name_expr: | ||||||
|  |                 error_msg += " Got this from the %r variable." % self.parent_name_expr #TODO nice repr. | ||||||
|  |             raise TemplateSyntaxError, error_msg | ||||||
|  |         try: | ||||||
|  |             return get_template_from_string(*find_template_source(parent, self.template_dirs)) | ||||||
|  |         except TemplateDoesNotExist: | ||||||
|  |             raise TemplateSyntaxError, "Template %r cannot be extended, because it doesn't exist" % parent | ||||||
|  |  | ||||||
|  |     def render(self, context): | ||||||
|  |         compiled_parent = self.get_parent(context) | ||||||
|  |         parent_is_child = isinstance(compiled_parent.nodelist[0], ExtendsNode) | ||||||
|  |         parent_blocks = dict([(n.name, n) for n in compiled_parent.nodelist.get_nodes_by_type(BlockNode)]) | ||||||
|  |         for block_node in self.nodelist.get_nodes_by_type(BlockNode): | ||||||
|  |             # Check for a BlockNode with this node's name, and replace it if found. | ||||||
|  |             try: | ||||||
|  |                 parent_block = parent_blocks[block_node.name] | ||||||
|  |             except KeyError: | ||||||
|  |                 # This BlockNode wasn't found in the parent template, but the | ||||||
|  |                 # parent block might be defined in the parent's *parent*, so we | ||||||
|  |                 # add this BlockNode to the parent's ExtendsNode nodelist, so | ||||||
|  |                 # it'll be checked when the parent node's render() is called. | ||||||
|  |                 if parent_is_child: | ||||||
|  |                     compiled_parent.nodelist[0].nodelist.append(block_node) | ||||||
|  |             else: | ||||||
|  |                 # Keep any existing parents and add a new one. Used by BlockNode. | ||||||
|  |                 parent_block.parent = block_node.parent | ||||||
|  |                 parent_block.add_parent(parent_block.nodelist) | ||||||
|  |                 parent_block.nodelist = block_node.nodelist | ||||||
|  |         return compiled_parent.render(context) | ||||||
|  |  | ||||||
|  | class ConstantIncludeNode(Node): | ||||||
|  |     def __init__(self, template_path): | ||||||
|  |         try: | ||||||
|  |             t = get_template(template_path) | ||||||
|  |             self.template = t | ||||||
|  |         except: | ||||||
|  |             if TEMPLATE_DEBUG: | ||||||
|  |                 pass | ||||||
|  | #                 raise | ||||||
|  |             self.template = None | ||||||
|  |  | ||||||
|  |     def render(self, context): | ||||||
|  |         if self.template: | ||||||
|  |             return self.template.render(context) | ||||||
|  |         else: | ||||||
|  |             return '' | ||||||
|  |  | ||||||
|  | class IncludeNode(Node): | ||||||
|  |     def __init__(self, template_name): | ||||||
|  |         self.template_name = template_name | ||||||
|  |  | ||||||
|  |     def render(self, context): | ||||||
|  |          try: | ||||||
|  |              template_name = resolve_variable(self.template_name, context) | ||||||
|  |              t = get_template(template_name) | ||||||
|  |              return t.render(context) | ||||||
|  |          except TemplateSyntaxError, e: | ||||||
|  |              if TEMPLATE_DEBUG: | ||||||
|  |                 raise | ||||||
|  |              return '' | ||||||
|  |          except: | ||||||
|  |              return '' # Fail silently for invalid included templates. | ||||||
|  |  | ||||||
|  | def do_block(parser, token): | ||||||
|  |     """ | ||||||
|  |     Define a block that can be overridden by child templates. | ||||||
|  |     """ | ||||||
|  |     bits = token.contents.split() | ||||||
|  |     if len(bits) != 2: | ||||||
|  |         raise TemplateSyntaxError, "'%s' tag takes only one argument" % bits[0] | ||||||
|  |     block_name = bits[1] | ||||||
|  |     # Keep track of the names of BlockNodes found in this template, so we can | ||||||
|  |     # check for duplication. | ||||||
|  |     try: | ||||||
|  |         if block_name in parser.__loaded_blocks: | ||||||
|  |             raise TemplateSyntaxError, "'%s' tag with name '%s' appears more than once" % (bits[0], block_name) | ||||||
|  |         parser.__loaded_blocks.append(block_name) | ||||||
|  |     except AttributeError: # parser._loaded_blocks isn't a list yet | ||||||
|  |         parser.__loaded_blocks = [block_name] | ||||||
|  |     nodelist = parser.parse(('endblock',)) | ||||||
|  |     parser.delete_first_token() | ||||||
|  |     return BlockNode(block_name, nodelist) | ||||||
|  |  | ||||||
|  | def do_extends(parser, token): | ||||||
|  |     """ | ||||||
|  |     Signal that this template extends a parent template. | ||||||
|  |  | ||||||
|  |     This tag may be used in two ways: ``{% extends "base" %}`` (with quotes) | ||||||
|  |     uses the literal value "base" as the name of the parent template to extend, | ||||||
|  |     or ``{% extends variable %}`` uses the value of ``variable`` as the name | ||||||
|  |     of the parent template to extend. | ||||||
|  |     """ | ||||||
|  |     bits = token.contents.split() | ||||||
|  |     if len(bits) != 2: | ||||||
|  |         raise TemplateSyntaxError, "'%s' takes one argument" % bits[0] | ||||||
|  |     parent_name, parent_name_expr = None, None | ||||||
|  |     if bits[1][0] in ('"', "'") and bits[1][-1] == bits[1][0]: | ||||||
|  |         parent_name = bits[1][1:-1] | ||||||
|  |     else: | ||||||
|  |         parent_name_expr = parser.compile_filter(bits[1]) | ||||||
|  |     nodelist = parser.parse() | ||||||
|  |     if nodelist.get_nodes_by_type(ExtendsNode): | ||||||
|  |         raise TemplateSyntaxError, "'%s' cannot appear more than once in the same template" % bits[0] | ||||||
|  |     return ExtendsNode(nodelist, parent_name, parent_name_expr) | ||||||
|  |  | ||||||
|  | def do_include(parser, token): | ||||||
|  |     """ | ||||||
|  |     Loads a template and renders it with the current context. | ||||||
|  |  | ||||||
|  |     Example:: | ||||||
|  |  | ||||||
|  |         {% include "foo/some_include" %} | ||||||
|  |     """ | ||||||
|  |     bits = token.contents.split() | ||||||
|  |     if len(bits) != 2: | ||||||
|  |         raise TemplateSyntaxError, "%r tag takes one argument: the name of the template to be included" % bits[0] | ||||||
|  |     path = bits[1] | ||||||
|  |     if path[0] in ('"', "'") and path[-1] == path[0]: | ||||||
|  |         return ConstantIncludeNode(path[1:-1]) | ||||||
|  |     return IncludeNode(bits[1]) | ||||||
|  |  | ||||||
|  | register.tag('block', do_block) | ||||||
|  | register.tag('extends', do_extends) | ||||||
|  | register.tag('include', do_include) | ||||||
| @@ -1,9 +1,11 @@ | |||||||
| from django.core.template import Node, NodeList, Template, Context, resolve_variable, resolve_variable_with_filters, registered_filters | from django.core.template import Node, NodeList, Template, Context, resolve_variable | ||||||
| from django.core.template import TemplateSyntaxError, register_tag, TokenParser | from django.core.template import TemplateSyntaxError, TokenParser, Library | ||||||
| from django.core.template import TOKEN_BLOCK, TOKEN_TEXT, TOKEN_VAR | from django.core.template import TOKEN_BLOCK, TOKEN_TEXT, TOKEN_VAR | ||||||
| from django.utils import translation | from django.utils import translation | ||||||
| import re, sys | import re, sys | ||||||
|  |  | ||||||
|  | register = Library() | ||||||
|  |  | ||||||
| class GetAvailableLanguagesNode(Node): | class GetAvailableLanguagesNode(Node): | ||||||
|     def __init__(self, variable): |     def __init__(self, variable): | ||||||
|         self.variable = variable |         self.variable = variable | ||||||
| @@ -53,10 +55,10 @@ class BlockTranslateNode(Node): | |||||||
|     def render(self, context): |     def render(self, context): | ||||||
|         context.push() |         context.push() | ||||||
|         for var,val in self.extra_context.items(): |         for var,val in self.extra_context.items(): | ||||||
|             context[var] = resolve_variable_with_filters(val, context) |             context[var] = val.resolve(context) | ||||||
|         singular = self.render_token_list(self.singular) |         singular = self.render_token_list(self.singular) | ||||||
|         if self.plural and self.countervar and self.counter: |         if self.plural and self.countervar and self.counter: | ||||||
|             count = resolve_variable_with_filters(self.counter, context) |             count = self.counter.resolve(context) | ||||||
|             context[self.countervar] = count |             context[self.countervar] = count | ||||||
|             plural = self.render_token_list(self.plural) |             plural = self.render_token_list(self.plural) | ||||||
|             result = translation.ngettext(singular, plural, count) % context |             result = translation.ngettext(singular, plural, count) % context | ||||||
| @@ -179,9 +181,9 @@ def do_block_translate(parser, token): | |||||||
|                     value = self.value() |                     value = self.value() | ||||||
|                     if self.tag() != 'as': |                     if self.tag() != 'as': | ||||||
|                         raise TemplateSyntaxError, "variable bindings in 'blocktrans' must be 'with value as variable'" |                         raise TemplateSyntaxError, "variable bindings in 'blocktrans' must be 'with value as variable'" | ||||||
|                     extra_context[self.tag()] = value |                     extra_context[self.tag()] = parser.compile_filter(value) | ||||||
|                 elif tag == 'count': |                 elif tag == 'count': | ||||||
|                     counter = self.value() |                     counter = parser.compile_filter(self.value()) | ||||||
|                     if self.tag() != 'as': |                     if self.tag() != 'as': | ||||||
|                         raise TemplateSyntaxError, "counter specification in 'blocktrans' must be 'count value as variable'" |                         raise TemplateSyntaxError, "counter specification in 'blocktrans' must be 'count value as variable'" | ||||||
|                     countervar = self.tag() |                     countervar = self.tag() | ||||||
| @@ -213,7 +215,7 @@ def do_block_translate(parser, token): | |||||||
|  |  | ||||||
|     return BlockTranslateNode(extra_context, singular, plural, countervar, counter) |     return BlockTranslateNode(extra_context, singular, plural, countervar, counter) | ||||||
|  |  | ||||||
| register_tag('get_available_languages', do_get_available_languages) | register.tag('get_available_languages', do_get_available_languages) | ||||||
| register_tag('get_current_language', do_get_current_language) | register.tag('get_current_language', do_get_current_language) | ||||||
| register_tag('trans', do_translate) | register.tag('trans', do_translate) | ||||||
| register_tag('blocktrans', do_block_translate) | register.tag('blocktrans', do_block_translate) | ||||||
|   | |||||||
| @@ -278,6 +278,11 @@ In the above, the ``load`` tag loads the ``comments`` tag library, which then | |||||||
| makes the ``comment_form`` tag available for use. Consult the documentation | makes the ``comment_form`` tag available for use. Consult the documentation | ||||||
| area in your admin to find the list of custom libraries in your installation. | area in your admin to find the list of custom libraries in your installation. | ||||||
|  |  | ||||||
|  | **New in Django development version:** The ``{% load %}`` tag can take multiple | ||||||
|  | library names, separated by spaces. Example:: | ||||||
|  |  | ||||||
|  |     {% load comments i18n %} | ||||||
|  |  | ||||||
| Built-in tag and filter reference | Built-in tag and filter reference | ||||||
| ================================= | ================================= | ||||||
|  |  | ||||||
|   | |||||||
| @@ -444,6 +444,15 @@ the given Python module name, not the name of the app. | |||||||
| Once you've created that Python module, you'll just have to write a bit of | Once you've created that Python module, you'll just have to write a bit of | ||||||
| Python code, depending on whether you're writing filters or tags. | Python code, depending on whether you're writing filters or tags. | ||||||
|  |  | ||||||
|  | To be a valid tag library, the module contain a module-level variable that is a | ||||||
|  | ``template.Library`` instance, in which all the tags and filters are | ||||||
|  | registered. So, near the top of your module, put the following:: | ||||||
|  |  | ||||||
|  |     from django.core import template | ||||||
|  |     register = template.Library() | ||||||
|  |  | ||||||
|  | Convention is to call this instance ``register``. | ||||||
|  |  | ||||||
| .. admonition:: Behind the scenes | .. admonition:: Behind the scenes | ||||||
|  |  | ||||||
|     For a ton of examples, read the source code for Django's default filters |     For a ton of examples, read the source code for Django's default filters | ||||||
| @@ -453,10 +462,16 @@ Python code, depending on whether you're writing filters or tags. | |||||||
| Writing custom template filters | Writing custom template filters | ||||||
| ------------------------------- | ------------------------------- | ||||||
|  |  | ||||||
| Custom filters are just Python functions that take two arguments: | **This section applies to the Django development version.** | ||||||
|  |  | ||||||
|     * The value of the variable (input) -- not necessarily a string | Custom filters are just Python functions that take one or two arguments: | ||||||
|     * The value of the argument -- always a string |  | ||||||
|  |     * The value of the variable (input) -- not necessarily a string. | ||||||
|  |     * The value of the argument -- this can have a default value, or be left | ||||||
|  |       out altogether. | ||||||
|  |  | ||||||
|  | For example, in the filter ``{{ var|foo:"bar" }}``, the filter ``foo`` would be | ||||||
|  | passed the variable ``var`` and the argument ``"bar"``. | ||||||
|  |  | ||||||
| Filter functions should always return something. They shouldn't raise | Filter functions should always return something. They shouldn't raise | ||||||
| exceptions. They should fail silently. In case of error, they should return | exceptions. They should fail silently. In case of error, they should return | ||||||
| @@ -468,36 +483,48 @@ Here's an example filter definition:: | |||||||
|         "Removes all values of arg from the given string" |         "Removes all values of arg from the given string" | ||||||
|         return value.replace(arg, '') |         return value.replace(arg, '') | ||||||
|  |  | ||||||
| Most filters don't take arguments. For filters that don't take arguments, the | And here's an example of how that filter would be used:: | ||||||
| convention is to use a single underscore as the second argument to the filter |  | ||||||
| definition. Example:: |  | ||||||
|  |  | ||||||
|     def lower(value, _): |     {{ somevariable|cut:"0" }} | ||||||
|  |  | ||||||
|  | Most filters don't take arguments. In this case, just leave the argument out of | ||||||
|  | your function. Example:: | ||||||
|  |  | ||||||
|  |     def lower(value): # Only one argument. | ||||||
|         "Converts a string into all lowercase" |         "Converts a string into all lowercase" | ||||||
|         return value.lower() |         return value.lower() | ||||||
|  |  | ||||||
| When you've written your filter definition, you need to register it, to make it | When you've written your filter definition, you need to register it with | ||||||
| available to Django's template language:: | your ``Library`` instance, to make it available to Django's template language:: | ||||||
|  |  | ||||||
|     from django.core import template |     register.filter('cut', cut) | ||||||
|     template.register_filter('cut', cut, True) |     register.filter('lower', lower) | ||||||
|     template.register_filter('lower', lower, False) |  | ||||||
|  |  | ||||||
| ``register_filter`` takes three arguments: | The ``Library.filter()`` method takes two arguments: | ||||||
|  |  | ||||||
|     1. The name of the filter -- a string. |     1. The name of the filter -- a string. | ||||||
|     2. The compilation function -- a Python function (not the name of the |     2. The compilation function -- a Python function (not the name of the | ||||||
|        function as a string). |        function as a string). | ||||||
|     3. A boolean, designating whether the filter requires an argument. This |  | ||||||
|        tells Django's template parser whether to throw ``TemplateSyntaxError`` |  | ||||||
|        when filter arguments are given (or missing). |  | ||||||
|  |  | ||||||
| The convention is to put all ``register_filter`` calls at the bottom of your | If you're using Python 2.4 or above, you can use ``register.filter()`` as a | ||||||
| template-library module. | decorator instead:: | ||||||
|  |  | ||||||
|  |     @register.filter(name='cut') | ||||||
|  |     def cut(value, arg): | ||||||
|  |         return value.replace(arg, '') | ||||||
|  |  | ||||||
|  |     @register.filter | ||||||
|  |     def lower(value): | ||||||
|  |         return value.lower() | ||||||
|  |  | ||||||
|  | If you leave off the ``name`` argument, as in the second example above, Django | ||||||
|  | will use the function's name as the filter name. | ||||||
|  |  | ||||||
| Writing custom template tags | Writing custom template tags | ||||||
| ---------------------------- | ---------------------------- | ||||||
|  |  | ||||||
|  | **This section applies to the Django development version.** | ||||||
|  |  | ||||||
| Tags are more complex than filters, because tags can do anything. | Tags are more complex than filters, because tags can do anything. | ||||||
|  |  | ||||||
| A quick overview | A quick overview | ||||||
| @@ -525,8 +552,6 @@ For each template tag the template parser encounters, it calls a Python | |||||||
| function with the tag contents and the parser object itself. This function is | function with the tag contents and the parser object itself. This function is | ||||||
| responsible for returning a ``Node`` instance based on the contents of the tag. | responsible for returning a ``Node`` instance based on the contents of the tag. | ||||||
|  |  | ||||||
| By convention, the name of each compilation function should start with ``do_``. |  | ||||||
|  |  | ||||||
| For example, let's write a template tag, ``{% current_time %}``, that displays | For example, let's write a template tag, ``{% current_time %}``, that displays | ||||||
| the current date/time, formatted according to a parameter given in the tag, in | the current date/time, formatted according to a parameter given in the tag, in | ||||||
| `strftime syntax`_. It's a good idea to decide the tag syntax before anything | `strftime syntax`_. It's a good idea to decide the tag syntax before anything | ||||||
| @@ -612,17 +637,32 @@ without having to be parsed multiple times. | |||||||
| Registering the tag | Registering the tag | ||||||
| ~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
| Finally, use a ``register_tag`` call, as in ``register_filter`` above. Example:: | Finally, register the tag with your module's ``Library`` instance, as explained | ||||||
|  | in "Writing custom template filters" above. Example:: | ||||||
|  |  | ||||||
|     from django.core import template |     register.tag('current_time', do_current_time) | ||||||
|     template.register_tag('current_time', do_current_time) |  | ||||||
|  |  | ||||||
| ``register_tag`` takes two arguments: | The ``tag()`` method takes two arguments: | ||||||
|  |  | ||||||
|     1. The name of the template tag -- a string. |     1. The name of the template tag -- a string. If this is left out, the | ||||||
|  |        name of the compilation function will be used. | ||||||
|     2. The compilation function -- a Python function (not the name of the |     2. The compilation function -- a Python function (not the name of the | ||||||
|        function as a string). |        function as a string). | ||||||
|  |  | ||||||
|  | As with filter registration, it is also possible to use this as a decorator, in | ||||||
|  | Python 2.4 and above: | ||||||
|  |  | ||||||
|  |     @register.tag(name="current_time") | ||||||
|  |     def do_current_time(parser, token): | ||||||
|  |         # ... | ||||||
|  |  | ||||||
|  |     @register.tag | ||||||
|  |     def shout(parser, token): | ||||||
|  |         # ... | ||||||
|  |  | ||||||
|  | If you leave off the ``name`` argument, as in the second example above, Django | ||||||
|  | will use the function's name as the tag name. | ||||||
|  |  | ||||||
| Setting a variable in the context | Setting a variable in the context | ||||||
| ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,15 +1,15 @@ | |||||||
| """ | """ | ||||||
| >>> floatformat(7.7, None) | >>> floatformat(7.7) | ||||||
| '7.7' | '7.7' | ||||||
| >>> floatformat(7.0, None) | >>> floatformat(7.0) | ||||||
| '7' | '7' | ||||||
| >>> floatformat(0.7, None) | >>> floatformat(0.7) | ||||||
| '0.7' | '0.7' | ||||||
| >>> floatformat(0.07, None) | >>> floatformat(0.07) | ||||||
| '0.1' | '0.1' | ||||||
| >>> floatformat(0.007, None) | >>> floatformat(0.007) | ||||||
| '0.0' | '0.0' | ||||||
| >>> floatformat(0.0, None) | >>> floatformat(0.0) | ||||||
| '0' | '0' | ||||||
| """ | """ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,7 +1,8 @@ | |||||||
| # Quick tests for the markup templatetags (django.contrib.markup) | # Quick tests for the markup templatetags (django.contrib.markup) | ||||||
|  |  | ||||||
| from django.core.template import Template, Context | from django.core.template import Template, Context, add_to_builtins | ||||||
| import django.contrib.markup.templatetags.markup # this registers the filters |  | ||||||
|  | add_to_builtins('django.contrib.markup.templatetags.markup') | ||||||
|  |  | ||||||
| # find out if markup modules are installed and tailor the test appropriately | # find out if markup modules are installed and tailor the test appropriately | ||||||
| try: | try: | ||||||
|   | |||||||
| @@ -1,8 +1,12 @@ | |||||||
| import traceback | from django.conf import settings | ||||||
|  |  | ||||||
|  | # Turn TEMPLATE_DEBUG off, because tests assume that. | ||||||
|  | settings.TEMPLATE_DEBUG = False | ||||||
|  |  | ||||||
| from django.core import template | from django.core import template | ||||||
| from django.core.template import loader | from django.core.template import loader | ||||||
| from django.utils.translation import activate, deactivate, install | from django.utils.translation import activate, deactivate, install | ||||||
|  | import traceback | ||||||
|  |  | ||||||
| # Helper objects for template tests | # Helper objects for template tests | ||||||
| class SomeClass: | class SomeClass: | ||||||
| @@ -99,8 +103,14 @@ TEMPLATE_TESTS = { | |||||||
|     # Chained filters, with an argument to the first one |     # Chained filters, with an argument to the first one | ||||||
|     'basic-syntax29': ('{{ var|removetags:"b i"|upper|lower }}', {"var": "<b><i>Yes</i></b>"}, "yes"), |     'basic-syntax29': ('{{ var|removetags:"b i"|upper|lower }}', {"var": "<b><i>Yes</i></b>"}, "yes"), | ||||||
|  |  | ||||||
|     #Escaped string as argument |     # Escaped string as argument | ||||||
|     'basic-syntax30': (r"""{{ var|default_if_none:" endquote\" hah" }}""", {"var": None}, ' endquote" hah'), |     'basic-syntax30': (r'{{ var|default_if_none:" endquote\" hah" }}', {"var": None}, ' endquote" hah'), | ||||||
|  |  | ||||||
|  |     # Variable as argument | ||||||
|  |     'basic-syntax31': (r'{{ var|default_if_none:var2 }}', {"var": None, "var2": "happy"}, 'happy'), | ||||||
|  |  | ||||||
|  |     # Default argument testing | ||||||
|  |     'basic-syntax32' : (r'{{ var|yesno:"yup,nup,mup" }} {{ var|yesno }}', {"var": True}, 'yup yes'), | ||||||
|  |  | ||||||
|     ### IF TAG ################################################################ |     ### IF TAG ################################################################ | ||||||
|     'if-tag01': ("{% if foo %}yes{% else %}no{% endif %}", {"foo": True}, "yes"), |     'if-tag01': ("{% if foo %}yes{% else %}no{% endif %}", {"foo": True}, "yes"), | ||||||
| @@ -289,7 +299,7 @@ TEMPLATE_TESTS = { | |||||||
| def test_template_loader(template_name, template_dirs=None): | def test_template_loader(template_name, template_dirs=None): | ||||||
|     "A custom template loader that loads the unit-test templates." |     "A custom template loader that loads the unit-test templates." | ||||||
|     try: |     try: | ||||||
|         return ( TEMPLATE_TESTS[template_name][0] , "test:%s" % template_name ) |         return (TEMPLATE_TESTS[template_name][0] , "test:%s" % template_name) | ||||||
|     except KeyError: |     except KeyError: | ||||||
|         raise template.TemplateDoesNotExist, template_name |         raise template.TemplateDoesNotExist, template_name | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,6 +2,8 @@ | |||||||
|  |  | ||||||
| from django.core import template | from django.core import template | ||||||
|  |  | ||||||
|  | register = template.Library() | ||||||
|  |  | ||||||
| class EchoNode(template.Node): | class EchoNode(template.Node): | ||||||
|     def __init__(self, contents): |     def __init__(self, contents): | ||||||
|         self.contents = contents |         self.contents = contents | ||||||
| @@ -12,4 +14,4 @@ class EchoNode(template.Node): | |||||||
| def do_echo(parser, token): | def do_echo(parser, token): | ||||||
|     return EchoNode(token.contents.split()[1:]) |     return EchoNode(token.contents.split()[1:]) | ||||||
|  |  | ||||||
| template.register_tag("echo", do_echo) | register.tag("echo", do_echo) | ||||||
		Reference in New Issue
	
	Block a user