如何在模板代码中设置变量的值?


216

说我有一个模板

<html>
<div>Hello {{name}}!</div>
</html>

在测试时,定义变量的值而不接触调用此模板的python代码将很有用。所以我正在寻找这样的东西

{% set name="World" %}     
<html>
<div>Hello {{name}}!</div>
</html>

Django中是否存在类似的内容?

Answers:


327

您可以使用with模板标签。

{% with name="World" %}     
<html>
<div>Hello {{name}}!</div>
</html>
{% endwith %}

35
但是您可以在with中更改变量的值吗?
大卫·天宇Wong

2
似乎您无法在with子句中声明容器(我已经尝试过列表和元组)
Vladislav Ivanishin

如果需要声明列表,请使用make_list。 docs.djangoproject.com/en/1.9/ref/templates/builtins/#make-list
MrValdez '16

3
Jinja说是{%set myvar = value%}为什么在Django中不起作用?
霍姆斯

3
@holms因为Django不使用Jinja :-) docs.djangoproject.com/en/1.7/topics/templates
elimisteve

50

创建一个模板标签:

该应用程序应该包含一个templatetags目录,在相同的水平models.pyviews.py等等。如果不存在,创建它-不要忘了__init__.py文件,以确保该目录作为一个Python的包装处理。

使用以下代码define_action.pytemplatetags目录内部创建一个文件:

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?

2
在我的情况下,循环后返回旧值:(
holms

7
在最新版本中,您似乎可以使用simple_tag代替Assignment_tag(它对我有用)。
凯瑟琳·奥斯本

我对这种解决方案的了解是,它似乎无法覆盖值。
雅加布(JakubJabłoński)

如果要使用此技术设置列表而不是仅设置值,请检查以下内容:stackoverflow.com/a/34407158/2193235
msb

如果要将变量设置为整数,并且想要增加它的值(例如),则需要使用add{% define counter|add:1 as counter %}。对于其他操作类似。
msb

35

不需要将所有内容都放入“ 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 %}

请注意,其中大部分是从此处获取的


如何将变量分配给上下文中存在的其他变量?另外要注意的是:允许模板在不检查上下文变量是否存在的情况下随意分配上下文变量可能会带来安全隐患。在我看来,一个更明智的做法是试图将其分配之前检查该变量的情况下:
soze

如果context.get(self.var_name):提高SuspiciousOperation(“尝试从上下文中已经存在的模板中分配变量”)
播出

27

有一些像约翰描述的一样的把戏。但是,Django的模板语言在设计上不支持设置变量(有关模板,请参见Django文档中的“哲学”框)。
因此,建议更改任何变量的方法触摸Python代码。


7
感谢您的指导。从设计人员的角度来看,有时在设计时快速设置变量以测试页面的各种状态有时会更容易。不建议在正在运行的代码中使用此做法。
亚历克西斯

2
django1.0接受“ with”标签。看来他们终于在修改自己的哲学:)。
Evgeny

2
实际上,“ with”标记仅用于别名。这可能会对性能(以及可读性!)产生巨大影响,但实际上并没有在传统编程术语中设置变量。
罗布

12

最好的解决方案是编写一个自定义assignment_tag。该解决方案比使用with标签更干净,因为它在逻辑和样式之间实现了非常清晰的分离。

首先创建模板标签文件(例如appname/templatetags/hello_world.py):

from django import template

register = template.Library()

@register.assignment_tag
def get_addressee():
    return "World"

现在,您可以在get_addressee模板中使用template标签:

{% load hello_world %}

{% get_addressee as addressee %}

<html>
    <body>
        <h1>hello {{addressee}}</h1>
    </body>
</html>

3
对于使用较新Django版本的人们,现在称为simple_tag!节省时间弄清楚为什么代码中无法识别“ register ..” ...
kaya

11

早在2009年,default模板过滤器就不是一个选择。

<html>
<div>Hello {{name|default:"World"}}!</div>
</html>

我必须说这就是我一直在寻找的东西!它也可以一起使用{% with state=form.state.value|default:other_context_variable %}不是other_context_variable,我们也可以使用任何'string_value'以及
Saurav库马尔

但是它将打印出来,我需要将其保存以备后用
holms

4

一般来说,这不是一个好主意。执行python中的所有逻辑,然后将数据传递到模板进行显示。模板应尽可能简单,以确保从事设计的人员可以专注于设计,而不必担心逻辑。

举个例子,如果您需要模板中的一些派生信息,最好将其放入python代码中的变量中,然后将其传递给模板。


3

使用with语句

{% with total=business.employees.count %}
    {{ total }} employee{{ total|pluralize }}
{% endwith %}

我不能在这个答案的第一段中暗示代码。模板语言可能已弃用旧格式。


2

您可以在模板中执行以下操作:

{% 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
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.