Как объединить строки в шаблонах Django? - PullRequest
177 голосов
/ 08 декабря 2010

Я хочу объединить строку в теге шаблона Django, например:

{% extend shop/shop_name/base.html %}

Здесь shop_name - моя переменная, и я хочу объединить ее с остатком пути.

Предположим, у меня есть shop_name=example.com, и я хочу, чтобы результат увеличился до shop/example.com/base.html.

Ответы [ 10 ]

356 голосов
/ 24 декабря 2010

Использовать с:

{% with "shop/"|add:shop_name|add:"/base.html" as template %}
{% include template %}
{% endwith %}
95 голосов
/ 21 мая 2014

Не используйте add для строк, вы должны определить пользовательский тег следующим образом:

Создать файл: <appname>\templatetags\<appname>_extras.py

from django import template

register = template.Library()

@register.filter
def addstr(arg1, arg2):
    """concatenate arg1 & arg2"""
    return str(arg1) + str(arg2)

, а затем используйте его, как говорит @Steven

{% with "shop/"|addstr:shop_name|addstr:"/base.html" as template %}
    {% include template %}
{% endwith %}

Причина ухода add

Согласно документам

Этот фильтр сначала попытается привести оба значения к целым числам ... Строки, которые можно привести к целым числам, будут суммированы, не объединены ...

Если обе переменные являются целыми числами, результат будет неожиданным.

10 голосов
/ 08 декабря 2010

Я изменил иерархию папок

/ shop / shop_name / base.html На /shop_name/shop/base.html

и тогда ниже будет работать.

{% extends shop_name|add:"/shop/base.html"%} 

Теперь можно расширять страницу base.html.

3 голосов
/ 05 марта 2018

См. Объединение строк в шаблонах Django :

  1. Для более ранних версий Django:

    {{ "Mary had a little"|stringformat:"s lamb." }}

«У Марии был маленький ягненок».

Остальное:

{{ "Mary had a little"|add:" lamb." }}

«У Марии был маленький ягненок».

3 голосов
/ 08 декабря 2010

Посмотрите на фильтр add .

Редактировать: Вы можете связать фильтры, чтобы вы могли сделать "shop/"|add:shop_name|add:"/base.html". Но это не сработает, потому что это зависит от тега шаблона для оценки фильтров в аргументах, а расширение не может.

Я думаю, вы не можете сделать это в шаблонах.

2 голосов
/ 08 декабря 2010

Из документов:

Этот тег можно использовать двумя способами:

  • {% extends "base.html" %} (с кавычками) использует буквальное значение "base.html" в качестве имениродительского шаблона для расширения.
  • {% extends variable %} использует значение переменной.Если переменная оценивается как строка, Django будет использовать эту строку в качестве имени родительского шаблона.Если переменная оценивается как объект Template, Django будет использовать этот объект в качестве родительского шаблона.

Похоже, вы не можете использовать фильтр для манипулирования аргументом.В вызывающем представлении вы должны либо создать экземпляр шаблона предка, либо создать строковую переменную с правильным путем и передать ее в контексте.

1 голос
/ 02 августа 2017

Работа с тегом {% with %} показалась мне довольно сложной. Вместо этого я создал следующий тег шаблона, который должен работать со строками и целыми числами.

from django import template

register = template.Library()


@register.filter
def concat_string(value_1, value_2):
    return str(value_1) + str(value_2)

Затем загрузите тег шаблона в свой шаблон вверху, используя следующее:

{% load concat_string %}

Затем вы можете использовать его следующим образом:

<a href="{{ SOME_DETAIL_URL|concat_string:object.pk }}" target="_blank">123</a>

Лично мне показалось, что работать с этим намного чище.

1 голос
/ 22 апреля 2016

@ ответ об ошибке в корне верен, для этого вы должны использовать тег шаблона. Однако я предпочитаю чуть более универсальный тег шаблона, который я могу использовать для выполнения любых операций, подобных этим:

from django import template
register = template.Library()


@register.tag(name='captureas')
def do_captureas(parser, token):
    """
    Capture content for re-use throughout a template.
    particularly handy for use within social meta fields 
    that are virtually identical. 
    """
    try:
        tag_name, args = token.contents.split(None, 1)
    except ValueError:
        raise template.TemplateSyntaxError("'captureas' node requires a variable name.")
    nodelist = parser.parse(('endcaptureas',))
    parser.delete_first_token()
    return CaptureasNode(nodelist, args)


class CaptureasNode(template.Node):
    def __init__(self, nodelist, varname):
        self.nodelist = nodelist
        self.varname = varname

    def render(self, context):
        output = self.nodelist.render(context)
        context[self.varname] = output
        return ''

и затем вы можете использовать это в своем шаблоне:

{% captureas template %}shop/{{ shop_name }}/base.html{% endcaptureas %}
{% include template %}

Как отмечается в комментарии, этот тег шаблона особенно полезен для информации, которая повторяется в шаблоне, но требует логики и других вещей, которые будут мешать вашим шаблонам, или в случаях, когда вы хотите повторно использовать данные, передаваемые между шаблонами через блоки:

{% captureas meta_title %}{% spaceless %}{% block meta_title %}
    {% if self.title %}{{ self.title }}{% endif %}
    {% endblock %}{% endspaceless %} - DEFAULT WEBSITE NAME
{% endcaptureas %}

, а затем:

<title>{{ meta_title }}</title>
<meta property="og:title" content="{{ meta_title }}" />
<meta itemprop="name" content="{{ meta_title }}">
<meta name="twitter:title" content="{{ meta_title }}">

Кредит для тега captureas можно получить здесь: https://www.djangosnippets.org/snippets/545/

0 голосов
/ 08 декабря 2010

extends не имеет возможности для этого.Либо поместите весь путь шаблона в переменную контекста и используйте его, либо скопируйте существующий тег шаблона и измените его соответствующим образом.

0 голосов
/ 08 декабря 2010

Вы не можете делать переменные в шаблонах django. У вас есть два варианта: либо написать свой собственный тег шаблона, либо сделать это на виду,

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...