如何在django模板中重复"块"

Dav*_*cos 122 django dry django-templates

我想在同一个django模板中使用相同的{%block%}两次.我希望此块在我的基本模板中出现多次:

# base.html
<html>
    <head>
        <title>{% block title %}My Cool Website{% endblock %}</title>
    </head>
    <body>
        <h1>{% block title %}My Cool Website{% endblock %}</h1>
    </body>
</html>
Run Code Online (Sandbox Code Playgroud)

然后扩展它:

# blog.html
{% extends 'base.html' %}
{% block title %}My Blog{% endblock %}

# pictures.html
{% extends 'base.html' %}
{% block title %}My Pictures{% endblock %}

# cats.html
{% extends 'base.html' %}
{% block title %}My Cats{% endblock %}
Run Code Online (Sandbox Code Playgroud)

我会得到一个例外,因为Django希望块只出现一次:

在/的TemplateSyntaxError

名称为"title"的'block'标记不止一次出现

一个快速而肮脏的解决方案是将块标题复制到title1title2:

# blog.html
{% extends 'base.html' %}
{% block title1 %}My Blog{% endblock %}
{% block title2 %}My Blog{% endblock %}
Run Code Online (Sandbox Code Playgroud)

但这违反了DRY原则.这将是非常困难的,因为我有很多继承模板,也因为我不想下地狱;-)

这个问题有什么技巧或解决方法吗?如何在模板中重复相同的块,而不重复所有代码?

小智 81

使用Django模板宏插件:

http://www.djangosnippets.org/snippets/363/(django <1.4)

要么

https://gist.github.com/1715202 (django> = 1.4)

然后,

# base.html
{% kwacro title %}
    {% block title %}My Cool Website{% endblock %}
{% endkwacro %}

<html>
    <head>
        <title>{% usekwacro title %}</title>
    </head>
    <body>
        <h1>{% usekwacro title %}</h1>
    </body>
</html>
Run Code Online (Sandbox Code Playgroud)

# blog.html
{% extends 'base.html' %}
{% block title %}My Blog{% endblock %}
Run Code Online (Sandbox Code Playgroud)

  • 这个解决方案非常旧,kwacros不再有效(我使用的是Python 3.9,Django 3.2.8) (5认同)
  • 这是太棒了!这可以真正清理我通过django循环和ajax数据循环共享模板所获得的问题. (2认同)

dqd*_*dqd 65

我认为在这种情况下使用上下文处理器是一种过度杀伤力.你可以轻松地做到这一点:

#base.html
<html>
    <head>
        <title>{% block title %}My Cool Website{% endblock %}</title>
    </head>
    <body>
        {% block content %}{% endblock %}
    </body>
</html>
Run Code Online (Sandbox Code Playgroud)

然后:

# blog.html
{% extends 'base.html' %}
{% block content %}
    <h1>{% block title %}My Blog{% endblock %}</h1>
    Lorem ipsum here...
{% endblock %}
Run Code Online (Sandbox Code Playgroud)

等等......看起来像DRY兼容.

  • 这不允许多次使用文本,对吗? (2认同)

Aar*_*paa 18

您可能实际上并不想使用块而是只使用变量:

# base.html
<html>
    <head>
        <title>{{ title|default:"My Cool Website" }}</title>
    </head>
    <body>
        <h1>{{ title|default:"My Cool Website" }}</h1>
    </body>
</html>
Run Code Online (Sandbox Code Playgroud)

然后,您可以通过上下文设置标题.

  • 可能干.但是你不想在视图中设置标题; 但在模板中. (17认同)
  • 标题应该在模板中设置,而不是由上下文提供,你需要有一种方法来定义这个"标题"变量,否则这不是一个好的解决方案. (6认同)

Jav*_*ier 12

你可以使用{% include subtemplate.html %}不止一次.它与块不同,但可以解决问题.


Rom*_*kov 11

这是我在尝试自己做同样事情时发现的一种方式:

# base_helper.html
<html>
    <head>
        <title>{% block _title1 %}{% endblock %}</title>
    </head>
    <body>
        <h1>{% block _title2 %}{% endblock %}</h1>
    </body>
</html>


# base.html
{% extends "base_helper.html" %}

# Copy title into _title1 & _title2, using "My Cool Website" as a default.
{% block _title1 %}{% block _title2 %}{% block title %}My Cool Website{% endblock %}{% endblock %}{% endblock %}
Run Code Online (Sandbox Code Playgroud)

遗憾的是需要额外的文件,但不要求您从视图中传递标题.

  • 很好的答案。这似乎比@dqd 的答案更干燥,因为您不必在每个继承 base 的模板中重复 &lt;h1&gt; 标签。这可能是公认的答案。 (2认同)

Rob*_*Mao 5

这里有一些讨论:http : //code.djangoproject.com/ticket/4529 显然django核心团队拒绝了这张票,因为他们认为这不是一个常用的场景,但是我不同意。

为此,重复块是一种简单而干净的实现:https : //github.com/SmileyChris/django-repeatblock

模板宏是另一个,但是作者提到它未经仔细测试:http : //www.djangosnippets.org/snippets/363/

我用了repeatblock。

  • 原始的django-repeatblock存储库似乎已被删除。最好的分支似乎是https://github.com/phretor/django-repeatblock; 我还找到了https://github.com/ydm/django-sameas,它不需要“ wontfix” Django补丁。 (3认同)

小智 5

作为对遇到此问题的任何人的更新,我已采用上面提到的代码片段并将其转换为模板标记库 django-macros ,使宏更强大,并且还显式实现了重复块模式:django-macros


man*_*ito 5

这是一个类似于上面do_setdo_get模板标签答案的轻量级解决方案。Django 允许您将整个模板上下文传递到标签中,该标签允许您定义全局变量。

基本.html:

<!DOCTYPE html>
<html lang="en">
<head>
  {% block head %}
    <title>{{ title }}</title>
  {% endblock %}
</head>
<body>
  <h1>{{ title }}</h1>
</body>
</html>
Run Code Online (Sandbox Code Playgroud)

页面.html:

{% extends "base.html" %}

{% block head %}
  {% define 'title' 'Homepage | title' %}
  {{ block.super }}
{% endblock %}
Run Code Online (Sandbox Code Playgroud)

自定义标签(在这里得到了这个想法:/sf/answers/2349549331/):

@register.simple_tag(takes_context=True)
def define(context, key, value):
    context.dicts[0][key] = value
    return ''
Run Code Online (Sandbox Code Playgroud)

另外,不要忘记{% load %}您的自定义标签或将它们添加到模板选项内置列表中,这样您就不必在每个模板中加载它们。此方法的唯一限制是{% define %}必须从块标记内调用,因为子模板仅呈现与父标记匹配的块标记。不确定是否有办法解决这个问题。还要确保define在尝试使用它之前会接到电话。