Python 如何在表单向导中使用ModelFormSet
Django文档在这个主题上没有很好的文档记录。事实上,他们在文件中的唯一参考是本段: 如何使用ModelForm和ModelFormSet WizardView.instance_dict。 WizardView支持模型表单和模型表单集。除了初始_dict之外,as_view()>方法还接受一个instance_dict参数,该参数应包含基于>ModelForm的步骤的模型实例和基于ModelFormSet的步骤的查询集 我还没有找到任何好的和明确的例子如何使用这个。有人能帮我吗 具体而言:Python 如何在表单向导中使用ModelFormSet,python,django,forms,django-forms,django-formwizard,Python,Django,Forms,Django Forms,Django Formwizard,Django文档在这个主题上没有很好的文档记录。事实上,他们在文件中的唯一参考是本段: 如何使用ModelForm和ModelFormSet WizardView.instance_dict。 WizardView支持模型表单和模型表单集。除了初始_dict之外,as_view()>方法还接受一个instance_dict参数,该参数应包含基于>ModelForm的步骤的模型实例和基于ModelFormSet的步骤的查询集 我还没有找到任何好的和明确的例子如何使用这个。有人能帮我吗 具体而言:
- 在forms.py中执行什么操作
- 如果我只在表单的某些步骤上需要一个ModelFormSet,而不是在所有步骤中,该怎么办
- 我需要在views.py和模板中执行什么操作
- 用例:
- 用户希望以多步骤形式注册,在第一步中,他介绍自己的姓名
- 在第二步中,他介绍了他的护照号码和酒店登记,他还想登记他的儿子和妻子,他们将与他一起前往这家酒店(这里我想使用HotelRegistration模型中的modelformset)
- 在第三步中,他输入他的航班信息。然后接收确认消息,确认表单是否有效并保存在数据库中
class Event(models.Model):
name = models.CharField(max_length=100)
date_from = models.DateField(auto_now=False)
date_to = models.DateField(auto_now=False)
description = models.TextField()
def __unicode__(self):
return self.name
class Hotel(models.Model):
name = models.CharField(max_length=50)
address = models.CharField(max_length=100)
def __unicode__(self):
return self.name
class HotelRegistration(models.Model):
pax_first_name = models.CharField(max_length=50)
pax_last_name = models.CharField(max_length=50)
hotel = models.ForeignKey(Hotel)
def __unicode__(self):
return self.pax_first_name
class Registration(models.Model):
first_name = models.CharField(max_length=50)
last_name = models.CharField(max_length=50)
passport = models.CharField(max_length=15)
city_origin = models.CharField(max_length=50)
flight_date_from = models.DateField(auto_now=False)
flight_date_to = models.DateField(auto_now=False)
require_transfer = models.BooleanField(default=None)
event = models.ForeignKey(Event)
hotel_registration = models.ForeignKey(HotelRegistration, blank=True, null=True)
def __unicode__(self):
return self.first_name
forms.py
from django import forms
from .models import Registration
class FormStep1(forms.ModelForm):
class Meta:
model = Registration
fields = ['first_name', 'last_name']
widgets = {
'first_name': forms.TextInput(attrs={'placeholder': 'Nombre de la persona que esta reservando', 'label_tag': 'Nombre'}),
'last_name': forms.TextInput(attrs={'placeholder': 'Apellido'})
}
class FormStep2(forms.ModelForm):
class Meta:
model = Registration
fields = ['passport', 'hotel_registration']
exclude = ('first_name', 'last_name', 'event' , 'city_origin', 'flight_date_from', 'flight_date_to', 'require_transfer')
widgets = {
'passport': forms.NumberInput(attrs={'placeholder':'Escriba su pasaporte'})
}
class FormStep3(forms.ModelForm):
class Meta:
model = Registration
fields = ['city_origin', 'flight_date_from', 'flight_date_to', 'require_transfer', 'event']
exclude = ('first_name', 'last_name', 'hotel_registration')
widgets = {
'city_origin': forms.TextInput(attrs={'placeholder':'Ciudad desde donde esta viajando'}),
'flight_date_from': forms.DateInput(format=('%d-%m-%Y'), attrs={'class':'myDateClass', 'placeholder':'Select a date'}),
'flight_date_to': forms.DateInput(format=('%d-%m-%Y'), attrs={'class':'myDateClass', 'placeholder':'Select a date'}),
'require_transfer': forms.Select(),
'event': forms.Select()
}
views.py
from django.shortcuts import render
from django.contrib.formtools.wizard.views import SessionWizardView
from django.http import HttpResponseRedirect
from django.views.generic import TemplateView
from django.forms.models import inlineformset_factory
from .models import Registration, HotelRegistration
from .forms import FormStep1, FormStep2, FormStep3
FORMS = [
("step1", FormStep1),
("step2", FormStep2),
("step3", FormStep3)
]
TEMPLATES = {
"step1" : "wizard/step1.html",
"step2" : "wizard/step2.html",
"step3" : "wizard/step3.html"
}
class TestFormWizard(SessionWizardView):
instance = None
def get_form_instance(self, step):
if self.instance is None:
self.instance = Registration()
return self.instance
def get_form(self, step=None, data=None, files=None):
form = super(TestFormWizard, self).get_form(step, data, files)
HotelRegistFormSet = inlineformset_factory(HotelRegistration, Registration, can_delete=True, extra=1)
# determine the step if not given
if step is None:
step = self.steps.current
if step == '2':
hotel_registration_formset = HotelRegistFormSet(self.steps.current, self.steps.files, prefix="step2")
return form
def get_template_names(self):
return [TEMPLATES[self.steps.current]]
def done(self, form_list, **kwargs):
self.instance.save()
return HttpResponseRedirect('/register/confirmation')
class ConfirmationView(TemplateView):
template_name = 'wizard/confirmation.html'
模板
{% extends "base.html" %}
{% load i18n %}
{% block content %}
<p>Step {{ wizard.steps.step1 }} of {{ wizard.steps.count }}</p>
<form action="" method="post">{% csrf_token %}
<table>
{{ wizard.management_form }}
{% if wizard.form.forms %}
{{ wizard.form.management_form }}
{% for form in wizard.form.forms %}
{{ form }}
{% endfor %}
{% else %}
{{ wizard.form }}
{% endif %}
</table>
{% if wizard.steps.prev %}
<button name="wizard_goto_step" type="submit" value="{{ wizard.steps.first }}">{% trans "first step" %}</button>
<button name="wizard_goto_step" type="submit" value="{{ wizard.steps.prev }}">{% trans "prev step" %}</button>
{% endif %}
<input type="submit" value="{% trans "submit" %}"/>
</form>
{% endblock %}
{%extends“base.html”%}
{%load i18n%}
{%block content%}
{{wizard.steps.count}的步骤{wizard.steps.step1}
{%csrf_令牌%}
{{wizard.management_form}
{%if wizard.form.forms%}
{{wizard.form.management_form}
{wizard.form.forms%中窗体的%s}
{{form}}
{%endfor%}
{%else%}
{{wizard.form}
{%endif%}
{%if wizard.steps.prev%}
{%trans“第一步”%}
{%trans“上一步”%}
{%endif%}
{%endblock%}
在forms.py中执行什么操作
重写SessionWizardView的get_form_实例方法。这是FormWizard用于确定模型实例是否与模型表单一起使用的方法
向导视图。获取表单实例(步骤)
仅当ModelForm用作步骤的表单时,才会调用此方法。
返回一个模型对象,在为步骤实例化ModelForm时,该对象将作为实例参数传递。如果初始化表单向导时未提供实例对象,则不会返回任何实例对象
这可以在SessionWizardView实现中的每个步骤有条件地完成。我不明白你想做什么,给你一个确切的例子,所以这里有一个更一般的例子
def get_form_instance(self, step):
if step == u'3':
past_data = self.get_cleaned_data_for_step(u'2')
hotel_name = past_data['hotel_field']
hotel = Hotel.objects.get(name=hotel_name)
return hotel #do NOT set self.instance, just return the model instance you want
return self.instance_dict.get(step, None) # the default implementation
如果我只在表单的某些步骤上需要一个ModelFormSet,而不是在所有步骤中,该怎么办
见上文;使用“if step==(form/step name)”表达式确定每个步骤中发生的情况
我需要在views.py和模板中执行什么操作
使用ModelForm并向其传递模型对象实例将设置初始表单值。你还需要更多吗
希望这能向您展示FormWizard中所期望的结构。与我使用过的Django的任何其他部分相比,FormWizard需要非常特殊的结构,并且是一个整体类。您解决过这个问题吗?我将放弃在向导视图中使用模型表单集。您解决过这个问题吗?你呢,怀旧?