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

Ale*_*xis 194 django django-templates

说我有一个模板

<html>
<div>Hello {{name}}!</div>
</html>
Run Code Online (Sandbox Code Playgroud)

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

{% set name="World" %}     
<html>
<div>Hello {{name}}!</div>
</html>
Run Code Online (Sandbox Code Playgroud)

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

Joh*_*ohn 298

您可能需要with template标签.

{% with name="World" %}     
<html>
<div>Hello {{name}}!</div>
</html>
{% endwith %}
Run Code Online (Sandbox Code Playgroud)

编辑:

现在使用with标签的正确方法是:(仍然支持旧格式)

{% with name="World" %}     
<html>
<div>Hello {{name}}!</div>
</html>
{% endwith %}
Run Code Online (Sandbox Code Playgroud)

  • 但是你可以在with中更改变量的值吗? (28认同)
  • @holms 因为 Django 不使用 Jinja :-) https://docs.djangoproject.com/en/1.7/topics/templates/ (8认同)
  • 似乎你不能在 with 子句中声明一个容器(我已经尝试过列表和元组) (2认同)
  • Jinja说它是{%set myvar = value%}为什么它在django中不起作用? (2认同)

小智 37

创建模板标记:

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

使用以下代码创建一个名为templatetags目录define_action.py内部的文件:

from django import template
register = template.Library()

@register.simple_tag
def define(val=None):
  return val
Run Code Online (Sandbox Code Playgroud)

注意:开发服务器不会自动重启.添加templatetags模块后,您需要重新启动服务器,然后才能在模板中使用标记或过滤器.


然后在您的模板中,您可以将值分配给上下文,如下所示:

{% load define_action %}
{% if item %}

   {% define "Edit" as action %}

{% else %}

   {% define "Create" as action %}

{% endif %}


Would you like to {{action}} this item?
Run Code Online (Sandbox Code Playgroud)

  • 在最新版本中,您似乎可以使用simple_tag而不是assignment_tag(它对我有用). (6认同)
  • 在我的情况下,循环后返回旧值:( (2认同)
  • 如果您将变量设置为整数并且想要递增它(例如),则需要使用“add”:“{% Define counter|add:1 as counter %}”。对于其他操作也是如此。 (2认同)

Kar*_*rim 31

另一种不要求将所有内容放在"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)
Run Code Online (Sandbox Code Playgroud)

这将允许您在模板中编写类似的内容:

{% setvar "a string" as new_template_var %}
Run Code Online (Sandbox Code Playgroud)

请注意,大部分内容都来自此处


rob*_*rob 24

有像约翰描述的那样的技巧; 但是,Django的设计模板语言不支持设置变量(请参阅Django文档中的"Philosophy"框中的模板).
因此,更改任何变量的推荐方法触摸Python代码.

  • 谢谢你的指针.从设计者的角度来看,有时更容易快速设置变量来测试页面的各种状态.不建议在运行的代码中使用此实践. (7认同)
  • django1.0中接受"with"标记.所以看起来他们终于修改了他们的哲学:). (2认同)
  • 事实上,"with"标签仅用于别名.这可能会对性能(以及可读性)产生巨大影响,但它并没有真正在传统的编程术语中设置变量. (2认同)

uni*_*ity 10

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

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

from django import template

register = template.Library()

@register.assignment_tag
def get_addressee():
    return "World"
Run Code Online (Sandbox Code Playgroud)

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

{% load hello_world %}

{% get_addressee as addressee %}

<html>
    <body>
        <h1>hello {{addressee}}</h1>
    </body>
</html>
Run Code Online (Sandbox Code Playgroud)

  • 对于使用较新Django版本的用户,现在称为simple_tag!节省时间弄清楚为什么您的代码中无法识别“ register ..” ... (3认同)

Sar*_*ang 5

一般来说,这不是一个好主意。在python中完成所有逻辑并将数据传递给模板进行显示。模板应该尽可能简单,以确保设计人员可以专注于设计而不是担心逻辑。

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


Joh*_*Mee 5

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

<html>
<div>Hello {{name|default:"World"}}!</div>
</html>
Run Code Online (Sandbox Code Playgroud)