我之前使用过add
但这次出现了这个奇怪的错误:
django.template.exceptions.TemplateSyntaxError: add requires 2 arguments, 1 provided
使用以下代码:
{% render_field form.full_name type="hidden" class="form-control" \
value=user.first_name|add:"something" %}
但如果我将add
移至with
,我就不会收到错误:
{% with fullname=user.first_name|add:"something"%}
{% render_field form.full_name type="hidden" class="form-control" \
value=fullname %}
{% endwith %}
答案 0 :(得分:1)
所以看起来您正在使用https://github.com/jazzband/django-widget-tweaks/
中的render_field
模板标记
render_field
实现为由包维护者实现的Advanced Custom Tag,遗憾的是它不支持其参数列表中的过滤器。
另一方面,simple_tag
或inclusion_tag
;由Django定义;确实支持其参数列表中的其他过滤器。
我的意思是,如果是simple_tag
或inclusion_tag
,这将有效
{% my_tag 123 "abcd" book.title warning=message|lower profile=user.profile %}
但是,如果my_tag
被注册为高级自定义标记;在参数中支持过滤器取决于实现。
正如我从render_field
代码的实现中看到的那样;它是一个高级自定义标记,过滤器支持似乎已被破坏:
@register.tag
def render_field(parser, token):
"""
Render a form field using given attribute-value pairs
Takes form field as first argument and list of attribute-value pairs for
all other arguments. Attribute-value pairs should be in the form of
attribute=value or attribute="a value" for assignment and attribute+=value
or attribute+="value" for appending.
"""
error_msg = '%r tag requires a form field followed by a list of attributes and values in the form attr="value"' % token.split_contents()[0]
try:
bits = token.split_contents()
tag_name = bits[0]
form_field = bits[1]
attr_list = bits[2:]
except ValueError:
raise TemplateSyntaxError(error_msg)
form_field = parser.compile_filter(form_field)
set_attrs = []
append_attrs = []
for pair in attr_list:
match = ATTRIBUTE_RE.match(pair)
if not match:
raise TemplateSyntaxError(error_msg + ": %s" % pair)
dct = match.groupdict()
attr, sign, value = \
dct['attr'], dct['sign'], parser.compile_filter(dct['value'])
if sign == "=":
set_attrs.append((attr, value))
else:
append_attrs.append((attr, value))
return FieldAttributeNode(form_field, set_attrs, append_attrs)
具体而言;成对的key=value
存储在代码中的attr_list
中,并针对ATTRIBUTE_RE
正则表达式匹配
然后使用parser.compile_filter(dct['value'])
运行过滤器,但dct['value']
为'user.first_name|add:"'
- >文本"something"
在这里丢失了。我的猜测是需要改进ATTRIBUTE_RE
正则表达式以支持这一点。
>>> ATTRIBUTE_RE.match('value=user.first_name|add:"something"')
<_sre.SRE_Match object at 0x7f8617f5a160>
>>> match.groupdict()
{'attr': 'value', 'value': 'user.first_name|add:"', 'sign': '='}
正如您所看到的,值键被破坏并且缺少您提供的全文;这打破了下游的添加过滤器。
这正是with
标记要解决的用例;因为它将过滤过程抽象到上面的水平;并且您可以很好地将新变量传递给自定义render_field
标记。
参考文献:
How to use a template filter on a custom template tag?
https://docs.djangoproject.com/en/2.0/howto/custom-template-tags/#advanced-custom-template-tags