Django UpdateView 的内联表单集

问题描述 投票:0回答:2

我的 Django 应用程序中有以下表单:

class SurveyAreaForm(ModelForm):
    class Media(object):
        js = formset_media_js

    class Meta:
        model = SurveyArea
        exclude = ['survey', ]
        widgets = {
            'action_by': CustomDateInput(),
        }


AreaFormSet = inlineformset_factory(Survey, SurveyArea, form=SurveyAreaForm)

class SurveyForm(ModelForm):
    class Meta:
        model = Survey
        exclude = ['tech', 'operator', ]
        widgets = {
            'date': CustomDateInput(),
            'client': FilteredJQMRadioSelectWithAdd(),
            'assessorSignature': SignatureInput(attrs={'id': 'assessorSignature'}),
        }

    def __init__(self, *args, **kwargs):
        super(SurveyForm, self).__init__(*args, **kwargs)
        self.fields['client'].empty_label = None

我正在使用 https://pypi.python.org/pypi/django-formset-js/0.3.0 提供 JavaScript 来添加其他表单。

每个 Survey 对象都可以有一个或多个与之关联的 SurveyArea,我想使用相同的表单使它们可编辑。但是,我在渲染初始数据时遇到了问题。这是我的看法:

class SurveyUpdateView(SurveyValidateMixin, UpdateViewWithTech):
    model = Survey
    form_class = SurveyForm
    success_url="/forms/survey/updated/"
    template_name="my_django_app/survey_update.html"

    def get(self, request, *args, **kwargs):
        """
        Handles GET requests and instantiates blank version of the form
        and its inline formsets.
        """
        self.object = self.get_object()
        form_class = self.get_form_class()
        form = self.get_form(form_class)

        # Get areas
        areas = SurveyArea.objects.filter(survey=self.object).order_by('name').values()

        # Render form
        area_form = AreaFormSet(initial=areas)
        return self.render_to_response(
            self.get_context_data(form=form,
                                area_form = area_form))

    def post(self, request, *args, **kwargs):
        """
        Handles POST requests, instantiating a form instance and its inline
        formsets with the passed POST variables and them checking them for
        validity.
        """
        self.object = self.get_object()
        form_class = self.get_form_class()
        form = self.get_form(form_class)
        area_form = AreaFormSet(self.request.POST)
        if (form.is_valid() and area_form.is_valid()):
            return self.form_valid(form, area_form)
        else:
            return self.form_invalid(form, area_form)

我还有一个

SurveyCreateView
,它工作得很好,并且使用
SurveyValidateMixin
在这些视图之间共享验证。
SurveyUpdateView
也继承自
UpdateViewWithTech
,它基本上只是限制用户的查询集并自动设置代表用户的字段。

我遇到的问题是渲染初始数据。在

get()
SurveyUpdateView
方法中,我正在获取当前与该调查相关的所有区域,并使用 PDB 我已经能够在获取与该调查相关的区域时确认这一点(在变量区域),数据似乎是正确的。这是 8 个项目的外观:

[{'description': u'A', 'photo': u'', 'action_required': u'A', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 21L, 'action_taken': True, 'name': u'A'}, {'description': u'A', 'photo': u'', 'action_required': u'A', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 19L, 'action_taken': True, 'name': u'A'}, {'description': u'A', 'photo': u'', 'action_required': u'A', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 29L, 'action_taken': True, 'name': u'A'}, {'description': u'A', 'photo': u'', 'action_required': u'A', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 30L, 'action_taken': True, 'name': u'A'}, {'description': u'B', 'photo': u'', 'action_required': u'B', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 22L, 'action_taken': True, 'name': u'B'}, {'description': u'B', 'photo': u'', 'action_required': u'B', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 20L, 'action_taken': True, 'name': u'B'}, {'description': u'B', 'photo': u'', 'action_required': u'B', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 31L, 'action_taken': True, 'name': u'B'}, {'description': u'C', 'photo': u'', 'action_required': u'C', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 23L, 'action_taken': True, 'name': u'C'}, {'description': u'X', 'photo': u'', 'action_required': u'X', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 24L, 'action_taken': True, 'name': u'X'}, {'description': u'Y', 'photo': u'', 'action_required': u'Y', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 25L, 'action_taken': True, 'name': u'Y'}, {'description': u'Z', 'photo': u'', 'action_required': u'Z', 'action_by': datetime.date(2013, 11, 14), 'survey_id': 12L, u'id': 26L, 'action_taken': True, 'name': u'Z'}]

但是,在实例化

initial
时将该数据作为
AreaFormSet
的值传递不会正确呈现数据。如果在
AreaFormSet
的定义中,我设置了
extra
的值,我会得到渲染的区域数量(如果未定义,则显示三个,我认为这是
extra
的默认值)。我希望看到的行为是以自己的形式呈现每个现有区域。

再次使用PDB,如果我在用

area_form.as_table()
设置之后转储area_form的值,我只能得到
extra
中设置的表单数量,所以问题似乎确实是通过初始数据。

我正确传递了

initial
的值吗?我知道
initial
的值应该是一个字典列表,它对我来说看起来是正确的,但我没有得到正确的渲染区域数量。

python django django-generic-views inline-formset
2个回答
1
投票

你尝试过吗

area_form = AreaFormSet(instance=self.object)

而不是

area_form = AreaFormSet(initial=areas)


0
投票

我看到您正在尝试获取表单集中的对象,以便可以填充表单集。但事实证明 django 会为你做到这一点,所以你只需要将对象本身作为实例传递给表单集,如下所示:

    # Render form
    # put the object instance in the formset
    area_form = AreaFormSet(instance=self.object) 
    return self.render_to_response(
        self.get_context_data(form=form,
                            area_form = area_form))
© www.soinside.com 2019 - 2024. All rights reserved.