У меня есть модельный набор, который мне нужно передать ключевому слову initial in to (чтобы иметь некоторые начальные значения в наборе форм, например дата, установленная сегодня). Насколько я могу судить, аргумент queryset (который по умолчанию относится к MyModel.objects.all()) переопределяет эту способность для создания пустых форм. Я задал запрос как: YourModel.objects.none(), который не позволяет ему заполнить набор форм записями из базы данных, но я до сих пор не могу понять, как эффективно проходить исходный код. Мне нужно подклассифицировать метод из ModelForm?
Передача начального значения в Django ModelFormSet
Ответ 1
Новое в Django 1.4... если вы установите initial на modelformset, теперь он применяется только к "дополнительным" формам, а не к объектам, связанным с экземплярами запроса:
https://docs.djangoproject.com/en/dev/topics/forms/modelforms/#providing-initial-values
Таким образом, все вышеперечисленные хаки больше не нужны, и использование начального ответа - это ответ.
Ответ 2
Вы передаете пустой набор запросов в ModelFormset.
ExampleModelFormSet = modelformset_factory(ExampleModel, extra=2)
formset = ExampleModelFormSet(queryset=ExampleModel.objects.none(),
initial=[{'name': 'Some Name'},
{'name': 'Another Name'}])
который действительно является способом достижения желаемого. Имеет смысл передать пустой запрос, иначе начальные значения переопределяют значение базы данных модели.
И если я полагаю, вы передаете непустой запрос, и все же хотите, чтобы ваши дополнительные формы были заполнены, будут начальные значения, вы можете переопределить Formset
.
from django.forms.models import modelformset_factory
ExampleFormset = modelformset_factory(OpenIDProfile, extra=3)
class myFormset(ExampleFormset):
def _construct_form(self, i, **kwargs):
"""
Instantiates and returns the i-th form instance in a formset.
"""
if self.is_bound and i < self.initial_form_count():
pk_key = "%s-%s" % (self.add_prefix(i), self.model._meta.pk.name)
pk = self.data[pk_key]
pk_field = self.model._meta.pk
pk = pk_field.get_db_prep_lookup('exact', pk)
if isinstance(pk, list):
pk = pk[0]
kwargs['instance'] = self._existing_object(pk)
if i < self.initial_form_count() and not kwargs.get('instance'):
kwargs['instance'] = self.get_queryset()[i]
defaults = {'auto_id': self.auto_id, 'prefix': self.add_prefix(i)}
if self.data or self.files:
defaults['data'] = self.data
defaults['files'] = self.files
# Check to confirm we are not overwriting the database value.
if not i in range(self.initial_form_count()) and self.initial:
try:
defaults['initial'] = self.initial[i - self.initial_form_count()]
except IndexError:
pass
# Allow extra forms to be empty.
if i >= self.initial_form_count():
defaults['empty_permitted'] = True
defaults.update(kwargs)
form = self.form(**defaults)
self.add_fields(form, i)
return form
Примечание: initial
- список dicts. Желательно иметь элементы в списке, в точности равные количеству дополнительных.
Ответ 3
Лучший способ, который я нашел для этого, - просто вручную отредактировать атрибут "extra" объекта formet после инициализации. Это было протестировано и работает в Django 1.9.
ExampleModelFormSet = modelformset_factory(ExampleModel, extra=1)
formset = ExampleModelFormSet(initial=[{'name': 'Some Name'},
{'name': 'Another Name'}])
formset.extra = 3
Это выплюнет 3 формы (2 предварительно заполненных и 1 пробел).
Если вы хотите что-то более динамичное (например, initial
- это var, вычисленное где-то в другом месте), вы можете сделать что-то вроде этого:
ExampleModelFormSet = modelformset_factory(ExampleModel, extra=1)
formset = ExampleModelFormSet(initial=initial)
formset.extra += len(initial)
Надеюсь, что это поможет.
Ответ 4
По-прежнему можно передать исходный код в ModelFormSet.
from django.forms.models import modelformset_factory
from example.models import ExampleModel
ExampleModelFormSet = modelformset_factory(ExampleModel)
formset = ExampleModelFormSet(initial=[{'name': 'Some Name'},
{'name': 'Another Name'}])
Если это не то, что вы после, можете ли вы объяснить свой вопрос больше.
EDIT:
ExampleModelFormSet = modelformset_factory(ExampleModel, extra=2)
formset = ExampleModelFormSet(queryset=ExampleModel.objects.none(),
initial=[{'name': 'Some Name'},
{'name': 'Another Name'}])
Ответ 5
После этого вы можете изменить исходные данные.
Например:
# Construct the FormSet class
ExampleFormSet = modelformset_factory(ParentModel, ChildModel)
# Set the initial data
initial_data = {'field1': 1}
for form in ExampleFormSet.extra_forms:
for field in form.fields:
field.initial = initial_data[field.name]
Ответ 6
В Django 1.4 правило изменилось: число инициалов, которые должны отображаться, соответствует параметру "extra". https://docs.djangoproject.com/en/dev/topics/forms/modelforms/#providing-initial-values
В моем случае количество исходных параметров изменяется динамически, поэтому Я нашел следующий способ:
initial=[{'name': 'Some Name'},{'name': 'Another Name'}]
formset = ExampleModelFormSet(queryset=ExampleModel.objects.none(),
initial=initial, extra=len(initial))
class MyBaseFormSet(BaseModelFormSet):
def __init__(self, *args, **kwargs):
self.extra = kwargs.pop('extra', self.extra)
super(MyBaseFormSet, self).__init__(*args, **kwargs)
ExampleModelFormSet основан на MyBaseFormSet
Ответ 7
Простым решением может быть то, что вы передаете форму, которая расширяет модельную форму() до набора форм (а не modelformset).
Таким образом, вы можете установить начальные значения в форме.
Ответ 8
Я думаю, вы можете использовать начальный аргумент, как и для обычных форм http://docs.djangoproject.com/en/dev/topics/forms/formsets/#using-initial-data-with-a-formset
Ответ 9
Простой способ передачи исходных данных с тем же количеством дополнительных форм:
MyModelFormSet = lambda *a, **kw: modelformset_factory(MyModel, extra=kw.pop('extra', 1))(*a, **kw)
Использование в представлениях:
initial=[{'name': 'Some Name'},{'name': 'Another Name'}
formset = MyModelFormSet(queryset=MyModel.objects.none(), initial, extra=len(initial))})
Ответ 10
Через два дня нашел это: http://streamhacker.com/2010/03/01/django-model-formsets/
def custom_field_callback(field):
return field.formfield(required=False)
FormSet = modelformset_factory(model, formfield_callback=custom_field_callback)
def custom_field_callback(field):
if field.name == 'optional':
return field.formfield(required=False)
elif field.name == 'text':
return field.formfield(widget=Textarea)
elif field.name == 'integer':
return IntegerField()
else:
return field.formfield()