假设我有一个模板
<html>
<div>Hello {{name}}!</div>
</html>
在测试它时,在不触及调用此模板的 python 代码的情况下定义变量的值会很有用。所以我正在寻找这样的东西
{% set name="World" %}
<html>
<div>Hello {{name}}!</div>
</html>
Django中是否存在类似的东西?
您可以使用 with
模板标记。
{% with name="World" %}
<html>
<div>Hello {{name}}!</div>
</html>
{% endwith %}
创建模板标签:
应用程序应包含与 models.py
、views.py
等处于同一级别的 templatetags
目录。如果该目录尚不存在,请创建它 - 不要忘记 __init__.py
文件以确保该目录是被视为 Python 包。
使用以下代码在 templatetags 目录中创建一个名为 define_action.py 的文件:
from django import template
register = template.Library()
@register.simple_tag
def define(val=None):
return val
注意:开发服务器不会自动重启。添加 templatetags
模块后,您需要重新启动服务器,然后才能在模板中使用标签或过滤器。
然后在您的模板中,您可以像这样为上下文分配值:
{% load define_action %}
{% if item %}
{% define "Edit" as action %}
{% else %}
{% define "Create" as action %}
{% endif %}
Would you like to {{action}} this item?
add
: {% define counter|add:1 as counter %}
。其他操作类似。
另一种不需要将所有内容都放在“with”块中的方法是创建一个自定义标签,将新变量添加到上下文中。如:
class SetVarNode(template.Node):
def __init__(self, new_val, var_name):
self.new_val = new_val
self.var_name = var_name
def render(self, context):
context[self.var_name] = self.new_val
return ''
import re
@register.tag
def setvar(parser,token):
# This version uses a regular expression to parse tag contents.
try:
# Splitting by None == splitting by spaces.
tag_name, arg = token.contents.split(None, 1)
except ValueError:
raise template.TemplateSyntaxError, "%r tag requires arguments" % token.contents.split()[0]
m = re.search(r'(.*?) as (\w+)', arg)
if not m:
raise template.TemplateSyntaxError, "%r tag had invalid arguments" % tag_name
new_val, var_name = m.groups()
if not (new_val[0] == new_val[-1] and new_val[0] in ('"', "'")):
raise template.TemplateSyntaxError, "%r tag's argument should be in quotes" % tag_name
return SetVarNode(new_val[1:-1], var_name)
这将允许您在模板中编写类似这样的内容:
{% setvar "a string" as new_template_var %}
请注意,其中大部分是 taken from here
有一些像约翰描述的技巧;但是,Django 的模板语言在设计上不支持设置变量(请参阅 Django documentation for templates 中的“哲学”框)。
因此,更改任何变量的推荐方法是通过触摸Python 代码。
最好的解决方案是编写自定义 assignment_tag
。此解决方案比使用 with
标记更简洁,因为它在逻辑和样式之间实现了非常清晰的分离。
首先创建一个模板标记文件(例如 appname/templatetags/hello_world.py
):
from django import template
register = template.Library()
@register.simple_tag
def get_addressee():
return "World"
现在您可以在模板中使用 get_addressee
模板标签:
{% load hello_world %}
{% get_addressee as addressee %}
<html>
<body>
<h1>hello {{addressee}}</h1>
</body>
</html>
也许 default
template filter 在 2009 年不是一个选项...
<html>
<div>Hello {{name|default:"World"}}!</div>
</html>
{% with state=form.state.value|default:other_context_variable %}
而不是 other_context_variable
我们也可以使用任何 'string_value'
一般来说,这不是一个好主意。在python中执行所有逻辑并将数据传递给模板进行显示。模板应该尽可能简单,以确保从事设计工作的人可以专注于设计而不是担心逻辑。
举个例子,如果您需要模板中的一些派生信息,最好将其放入 python 代码中的变量中,然后将其传递给模板。
在您的模板中,您可以这样做:
{% jump_link as name %}
{% for obj in name %}
<div>{{obj.helo}} - {{obj.how}}</div>
{% endfor %}
在您的模板标签中,您可以添加这样的标签:
@register.assignment_tag
def jump_link():
listArr = []
for i in range(5):
listArr.append({"helo" : i,"how" : i})
return listArr
使用 with statement。
{% with total=business.employees.count %}
{{ total }} employee{{ total|pluralize }}
{% endwith %}
我不能暗示 this answer 中第一段中的代码。也许模板语言已经弃用了旧格式。