django表单动态属性查找

Posted

技术标签:

【中文标题】django表单动态属性查找【英文标题】:django form dynamic attribute lookup 【发布时间】:2016-08-06 16:55:15 【问题描述】:

我遇到了一个问题,不知道如何解决。

我的申请中有以下表格:

class ContentForm(ModelForm):
    class Meta:
        model = Content
        fields = ('url_1','url_2','url_3','url_4','url_5',)
        widgets = 
            'url_1': forms.URLInput(attrs='class': 'form-control', 'maxlength': 100),
            'url_2': forms.URLInput(attrs='class': 'form-control', 'maxlength': 100),
            'url_3': forms.URLInput(attrs='class': 'form-control', 'maxlength': 100),
            'url_4': forms.URLInput(attrs='class': 'form-control', 'maxlength': 100),
            'url_5': forms.URLInput(attrs='class': 'form-control', 'maxlength': 100),
        

我要做的是通过循环动态生成这些 url 字段。我采用了this 方法(它实现了用于动态属性查找的模板标签):

% for i in 12345|make_list %
    % with url='url_'|add:i %
       form|getattribute:url 
    % endwith %
% endfor %

但是它不起作用。例如,Django 不呈现输入字段并抱怨我的表单对象没有名为url_1 的属性。如果我直接为同一个表单调用url_1 属性,它就可以工作。

getattribute模板标签的实现与here相同。

[更新:在下面发布了我的代码]

views.py

@login_required
def new(request):
    form = ContentForm()
    return render(request, "new.html", 'form': form)

new.html

<form action="% url 'create_content' %" method="POST">
...
% include 'partials/urls.html' %
...
</form>

urls.html

% load getattribute %

% for i in 12345|make_list %
    % with url='url_'|add:i %
      <label>URL i</label>
       form|getattribute:url 
    % endwith %
% endfor %

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"""

        return getattr(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

register.filter('getattribute', getattribute)

ps.:运行的django版本是1.9.2

【问题讨论】:

python 版本 2.7.10 【参考方案1】:

如下更改getattribute.py

import re
from django.forms import Form, ModelForm
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)]
    elif issubclass(value.__class__, (Form, ModelForm)) and arg in value.fields:
        return value[arg]
    else:
        return settings.TEMPLATE_STRING_IF_INVALID

register.filter('getattribute', getattribute)

您可以在第 17 行和第 18 行看到预期的特殊情况,与表单相关并返回相关的渲染小部件;)

【讨论】:

您不需要为表单添加特殊检查。 has_key 检查适用于 Python 2(应针对 Python 3 进行更改)。 has_key 方法和arg in value.fields 一样,谢谢@Alasdair 你的说法是对的,但是value中的键等于value.fields...我只为你改变它:) 更改我的代码后,看不到字段...当前代码可用 @Mohsen 我的错误,看起来has_key 检查对于 Django 表单返回 False。我认为最好的解决方案可能是将has_key 检查替换为return value[arg],并抓住可能的KeyError。那么你就没有Form 类的特殊代码了。

以上是关于django表单动态属性查找的主要内容,如果未能解决你的问题,请参考以下文章

Django 动态名称属性并在views.py 中获取值

Django 动态表单,带有 HTML 数组的表单集

Django:在表单向导中为步骤动态设置表单集

Django Admin中的动态只读字段

Django:使用 Jquery 的动态表单集仅保存第一个表单集实例

Django 动态表单保存