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

Передача начального значения в Django ModelFormSet

У меня есть модельный набор, который мне нужно передать ключевому слову initial in to (чтобы иметь некоторые начальные значения в наборе форм, например дата, установленная сегодня). Насколько я могу судить, аргумент queryset (который по умолчанию относится к MyModel.objects.all()) переопределяет эту способность для создания пустых форм. Я задал запрос как: YourModel.objects.none(), который не позволяет ему заполнить набор форм записями из базы данных, но я до сих пор не могу понять, как эффективно проходить исходный код. Мне нужно подклассифицировать метод из ModelForm?

4b9b3361

Ответ 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).

Таким образом, вы можете установить начальные значения в форме.

Ответ 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()