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但是还没有弄清楚如何处理这个问题.我尝试了几种方法:
访问者的自定义模型表单,我在其中为目标和活动添加多个选项字段.如果可以独立选择目的地和活动,那就可以了,但在这里它们是相关的,即我想为每个目的地选择一个或多个活动
使用inlineformset_factory生成一套目的地/活动形式,与inlineformset_factory(Destination, Visitor).这会中断,因为Visitor与Destination有M2M关系,而不是FK.
使用formset_factory例如自定义普通formset DestinationActivityFormSet = formset_factory(DestinationActivityForm, extra=2).但是如何设计DestinationActivityForm呢?我没有充分探讨这一点,但看起来并不是很有希望:我不想输入目的地和活动列表,我想要一个复选框列表,标签设置为我想要的目的地/活动选择,但formset_factory会返回具有相同标签的表单列表.
我是django的一个完全新手所以也许解决方案是显而易见的,但我发现这个领域的文档非常弱 - 如果有人对表单/表单集的使用示例有一些指示也会有帮助
谢谢!
最后,我选择在同一视图中处理多个表单,访问者详细信息的访问者模型表单,然后是每个目标的自定义表单列表.
在同一视图中处理多个表单结果非常简单(至少在这种情况下,没有跨字段验证问题).
我仍然感到惊讶的是,对于与中间模型的多对多关系没有内置支持,并且在网络上环顾四周,我发现没有直接引用它.我会发布代码,以防它帮助任何人.
首先是自定义表单:
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)
我通过传递一个Visitor和Destination对象(和为了方便而在外面计算的'访问'标志)来自定义每个表单
我使用布尔字段来允许用户选择每个目的地.该字段称为"已访问",但我将标签设置为目标,以便很好地显示.
活动由通常的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)
我只是在列表中收集目标表单并将此列表传递给我的模板,以便它可以迭代它们并显示它们.只要您不忘记为构造函数中的每个传递一个不同的前缀,它就可以正常工作
如果有人使用更清洁的方法,我会将问题保持开放几天.
谢谢!
| 归档时间: |
|
| 查看次数: |
4080 次 |
| 最近记录: |