Подтвердить что ты не робот

Выполнение поиска стиля getattr() в шаблоне django

Метод Python getattr() полезен, если вы заранее не знаете имя определенного атрибута.

Эта функциональность также пригодится в шаблонах, но я никогда не понял, как это сделать. Есть ли встроенный тег или не встроенный тег, который может выполнять динамический поиск атрибутов?

4b9b3361

Ответ 1

Мне также пришлось написать этот код в качестве настраиваемого тега шаблона. Чтобы обрабатывать все сценарии поиска, он сначала выполняет поиск стандартных атрибутов, затем пытается выполнить поиск по словарю, затем пытается найти getitem (для работы списков), а затем следует стандартным Поведение шаблона Django, когда объект не найден.

(обновлено 2009-08-26, чтобы теперь обрабатывать запросы индекса списка)

# app/templatetags/getattribute.py

import re
from django import template
from django.conf import settings

numeric_test = re.compile("^\d+$")
register = template.Library()

def getattribute(value, arg):
    """Gets an attribute of an object dynamically from a string name"""

    if hasattr(value, str(arg)):
        return getattr(value, arg)
    elif hasattr(value, 'has_key') and value.has_key(arg):
        return value[arg]
    elif numeric_test.match(str(arg)) and len(value) > int(arg):
        return value[int(arg)]
    else:
        return settings.TEMPLATE_STRING_IF_INVALID

register.filter('getattribute', getattribute)

Использование шаблона:

{% load getattribute %}
{{ object|getattribute:dynamic_string_var }}


Ответ 2

Я так не думаю. Но было бы нелегко написать настраиваемый тег шаблона, чтобы вернуть атрибут в контексте dict. Если вы просто пытаетесь вернуть строку, попробуйте что-то вроде этого:

class GetAttrNode(template.Node):
    def __init__(self, attr_name):
        self.attr_name = attr_name

    def render(self, context):
        try:
            return context[self.attr_name]
        except:
            # (better yet, return an exception here)
            return ''

@register.tag
def get_attr(parser, token):
    return GetAttrNode(token)

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

Ответ 3

В итоге я добавил метод к рассматриваемой модели, и этот метод можно получить как атрибут в шаблоне.

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

Ответ 4

Сохраняя различие между get и getattr,

@register.filter(name='get')
def get(o, index):
    try:
        return o[index]
    except:
        return settings.TEMPLATE_STRING_IF_INVALID


@register.filter(name='getattr')
def getattrfilter(o, attr):
    try:
        return getattr(o, attr)
    except:
        return settings.TEMPLATE_STRING_IF_INVALID

Ответ 6

Этот сниппет сохранил мой день, но мне нужно было его охватить отношениями, поэтому я изменил его, чтобы разделить arg на ".". и рекурсивно получить значение. Это можно сделать в одной строке: return getattribute(getattribute(value,str(arg).split(".")[0]),".".join(str(arg).split(".")[1:])) но я оставил его в 4 для удобства чтения. Я надеюсь, что у кого-то есть это.

import re
from django import template
from django.conf import settings

numeric_test = re.compile("^\d+$")
register = template.Library()

def getattribute(value, arg):
"""Gets an attribute of an object dynamically AND recursively from a string name"""
    if "." in str(arg):
        firstarg = str(arg).split(".")[0]
        value = getattribute(value,firstarg)
        arg = ".".join(str(arg).split(".")[1:])
        return getattribute(value,arg)
    if hasattr(value, str(arg)):
        return getattr(value, arg)
    elif hasattr(value, 'has_key') and value.has_key(arg):
        return value[arg]
    elif numeric_test.match(str(arg)) and len(value) > int(arg):
        return value[int(arg)]
    else:
        #return settings.TEMPLATE_STRING_IF_INVALID
        return 'no attr.' + str(arg) + 'for:' + str(value)

register.filter('getattribute', getattribute)