Получить поле формы Django из поля модели

Я хотел бы создать форму, включающую поля из двух отдельных моделей, а также некоторые другие регулярные (немодельные) поля. Форма создаст экземпляр каждой модели. Я не думаю, что я могу использовать встроенные формы для этого, так как я не хочу включать все поля из обеих моделей.

Я хотел бы создать поле формы без жесткого кодирования типа полей модели.

Я знаю, что я могу получить поле формы из поля модели, используя model_field.formfield(). Но как я могу получить конкретное поле модели?

Мое первое решение:

def get_fields(model_class):
    fields = {}
    for f in model_class._meta.fields:
        fields[f.name] = f

class MyForm(forms.Form):
    foo_name = get_fields(Foo)['name'].formfield()
    bar_name = get_fields(Bar)['name'].formfield()
    other_field = ...

Существует ли уже эквивалент get_fields? Это плохая идея? Мне неудобно полагаться на атрибут _meta. Или я об этом совершенно неправильно?

Ответы

Ответ 1

Вы также можете взглянуть на django.forms.models.fields_for_model Это должно дать вам словарь полей, а затем вы можете добавить поля формы

Ответ 2

Вам никогда не придется создавать поля самостоятельно, если вы не хотите особого поведения.

Это должно быть так же просто, как использовать два ModelForm и дополнительный Form внутри одного тега <form> в вашем шаблоне с помощью одной кнопки отправки.

в forms.py:

class Model1Form(forms.ModelForm):
    class Meta:
        model = Model1
        fields = ('fields', 'you', 'want')

class Model2Form(forms.ModelForm):
    class Meta:
        model = Model2
        fields = ('fields', 'you', 'want')

class ExtraFieldsForm(forms.Form):
    extra_field = form.TextField() # or whatever field you're looking for

в views.py:

form1 = Model1Form(request.POST or None)
form2 = Model2Form(request.POST or None)
form3 = ExtraFieldsForm(request.POST or None)

if form1.is_valid() and form2.is_valid() and form3.is_valid():
    form1.save()
    form2.save()
    form3.save()

    ...do other stuff like a redirect...

и в шаблоне:

<form method="POST" action="">{% csrf_token %}
    <fieldset>
        {{ form1|as_uni_form }}
        {{ form2|as_uni_form }}
        {{ form3|as_uni_form }}
        <div class="form_block">
            <input type="submit" value="Save both models"/>
        </div>
    </fieldset>
</form>

Я использую django-uni-form, но вы можете отображать поля формы, как вам нравится. Удачи с вашим сайтом.

Ответ 3

В настоящее время существует documented API для получения поля модели из класса модели:

my_model_field = MyModel._meta.get_field('my_model_field_name')

Хотя он официально не документирован до Django 1.8, это должно работать и с более ранними версиями Django.

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

form_field = my_model_field.formfield()

Ответ 4

Другим решением может быть создание одной "uber" -формы, которая объединяет конкретные модели моделей. Форма поддерживает методы, которые обычно предоставляет форма, и передает их всем дочерним формам. Некоторые из них будут простыми, другими сложными. Большим преимуществом такого подхода является то, что не влияет на код вне формы (код проверки клиента и тому подобное). Концепция не является действительно революционной, но я думаю, что сложно добавить потом. Пол