django表单:在单个表单中编辑多组相关对象

xul*_*vez 6 django django-forms django-orm

我正在尝试做一些非常常见的事情:在一个表单中添加/编辑一堆相关模型.例如:

Visitor Details:
Select destinations and activities:
    Miami  []   -  swimming [], clubbing [], sunbathing[]
    Cancun []   -  swimming [], clubbing [], sunbathing[]
Run Code Online (Sandbox Code Playgroud)

我的模型是访问者,目的地和活动,访问者通过中间模型VisitorDestination将ManyToMany字段导入Destination,其中包含要在目标上完成的活动的详细信息(本身是Activity中的ManyToMany字段).

Visitor ---->(M2M though VisitorDestination) -------------> Destination
                                            |
                       activities            ---->(M2M)---> Activity  
Run Code Online (Sandbox Code Playgroud)

请注意,我不想输入新的目的地/活动值,只需从数据库中可用的那些中选择(但这是对M2M字段的完全合法使用吗?)

对我来说,这看起来像一个非常常见的情况(与其他模型中的FK或M2M字段的其他细节有很多甚至很多关系),这看起来像是最明智的建模,但如果我错了,请纠正我.

我花了几天时间搜索Django docs/SO/googling但是还没有弄清楚如何处理这个问题.我尝试了几种方法:

  1. 访问者的自定义模型表单,我在其中为目标和活动添加多个选项字段.如果可以独立选择目的地和活动,那就可以了,但在这里它们是相关的,即我想为每个目的地选择一个或多个活动

  2. 使用inlineformset_factory生成一套目的地/活动形式,与inlineformset_factory(Destination, Visitor).这会中断,因为Visitor与Destination有M2M关系,而不是FK.

  3. 使用formset_factory例如自定义普通formset DestinationActivityFormSet = formset_factory(DestinationActivityForm, extra=2).但是如何设计DestinationActivityForm呢?我没有充分探讨这一点,但看起来并不是很有希望:我不想输入目的地和活动列表,我想要一个复选框列表,标签设置为我想要的目的地/活动选择,但formset_factory会返回具有相同标签的表单列表.

我是django的一个完全新手所以也许解决方案是显而易见的,但我发现这个领域的文档非常弱 - 如果有人对表单/表单集的使用示例有一些指示也会有帮助

谢谢!

xul*_*vez 8

最后,我选择在同一视图中处理多个表单,访问者详细信息的访问者模型表单,然后是每个目标的自定义表单列表.

在同一视图中处理多个表单结果非常简单(至少在这种情况下,没有跨字段验证问题).

我仍然感到惊讶的是,对于与中间模型的多对多关系没有内置支持,并且在网络上环顾四周,我发现没有直接引用它.我会发布代码,以防它帮助任何人.

首先是自定义表单:

class VisitorForm(ModelForm):
    class Meta:
      model = Visitor
      exclude = ['destinations']

class VisitorDestinationForm(Form):
    visited = forms.BooleanField(required=False)
    activities = forms.MultipleChoiceField(choices = [(obj.pk, obj.name) for obj in Activity.objects.all()], required=False, 
                                                      widget = CheckboxSelectMultipleInline(attrs={'style' : 'display:inline'}))

    def __init__(self, visitor, destination, visited,  *args, **kwargs):
        super(VisitorDestinationForm, self).__init__(*args, **kwargs)
        self.destination = destination
        self.fields['visited'].initial = visited
        self.fields['visited'].label= destination.destination

        # load initial choices for activities
        activities_initial = []
        try:
            visitorDestination_entry = VisitorDestination.objects.get(visitor=visitor, destination=destination)
            activities = visitorDestination_entry.activities.all()
            for activity in Activity.objects.all():
                if activity in activities: 
                    activities_initial.append(activity.pk)
        except VisitorDestination.DoesNotExist:
            pass
        self.fields['activities'].initial = activities_initial
Run Code Online (Sandbox Code Playgroud)

我通过传递一个VisitorDestination对象(和为了方便而在外面计算的'访问'标志)来自定义每个表单

我使用布尔字段来允许用户选择每个目的地.该字段称为"已访问",但我将标签设置为目标,以便很好地显示.

活动由通常的MultipleChoiceField处理(我使用我自定义的小部件来获取在桌面上显示的复选框,非常简单,但如果有人需要,可以发布)

然后是查看代码:

def edit_visitor(request, pk):
    visitor_obj = Visitor.objects.get(pk=pk)
    visitorDestinations = visitor_obj.destinations.all()
    if request.method == 'POST':
        visitorForm = VisitorForm(request.POST, instance=visitor_obj)

        # set up the visitor destination forms
        destinationForms = []
        for destination in Destination.objects.all():
            visited = destination in visitorDestinations
            destinationForms.append(VisitorDestinationForm(visitor_obj, destination, visited, request.POST, prefix=destination.destination))

        if visitorForm.is_valid() and all([form.is_valid() for form in destinationForms]):
            visitor_obj = visitorForm.save()
            # clear any existing entries,
            visitor_obj.destinations.clear()
            for form in destinationForms:
                if form.cleaned_data['visited']: 
                    visitorDestination_entry = VisitorDestination(visitor = visitor_obj, destination=form.destination)
                    visitorDestination_entry.save()
                    for activity_pk in form.cleaned_data['activities']: 
                        activity = Activity.objects.get(pk=activity_pk)
                        visitorDestination_entry.activities.add(activity)
                    print 'activities: %s' % visitorDestination_entry.activities.all()
                    visitorDestination_entry.save()

            success_url = reverse('visitor_detail', kwargs={'pk' : visitor_obj.pk})
            return HttpResponseRedirect(success_url)
    else:
        visitorForm = VisitorForm(instance=visitor_obj)
        # set up the visitor destination forms
        destinationForms = []
        for destination in Destination.objects.all():
            visited = destination in visitorDestinations
            destinationForms.append(VisitorDestinationForm(visitor_obj, destination, visited,  prefix=destination.destination))

    return render_to_response('testapp/edit_visitor.html', {'form': visitorForm, 'destinationForms' : destinationForms, 'visitor' : visitor_obj}, context_instance= RequestContext(request))
Run Code Online (Sandbox Code Playgroud)

我只是在列表中收集目标表单并将此列表传递给我的模板,以便它可以迭代它们并显示它们.只要您不忘记为构造函数中的每个传递一个不同的前缀,它就可以正常工作

如果有人使用更清洁的方法,我会将问题保持开放几天.

谢谢!