Python 如何在表单向导中使用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的步骤的查询集 我还没有找到任何好的和明确的例子如何使用这个。有人能帮我吗 具体而言:

Django文档在这个主题上没有很好的文档记录。事实上,他们在文件中的唯一参考是本段:

如何使用ModelForm和ModelFormSet

WizardView.instance_dict。 WizardView支持模型表单和模型表单集。除了初始_dict之外,as_view()>方法还接受一个instance_dict参数,该参数应包含基于>ModelForm的步骤的模型实例和基于ModelFormSet的步骤的查询集

我还没有找到任何好的和明确的例子如何使用这个。有人能帮我吗

具体而言:

  • 在forms.py中执行什么操作
  • 如果我只在表单的某些步骤上需要一个ModelFormSet,而不是在所有步骤中,该怎么办
  • 我需要在views.py和模板中执行什么操作
以我正在处理的用例和小项目为例,我分享了我的代码:

  • 用例:
    • 用户希望以多步骤形式注册,在第一步中,他介绍自己的姓名
  • 在第二步中,他介绍了他的护照号码和酒店登记,他还想登记他的儿子和妻子,他们将与他一起前往这家酒店(这里我想使用HotelRegistration模型中的modelformset)
  • 在第三步中,他输入他的航班信息。然后接收确认消息,确认表单是否有效并保存在数据库中
这是我的密码:

models.py

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需要非常特殊的结构,并且是一个整体类。

您解决过这个问题吗?我将放弃在向导视图中使用模型表单集。您解决过这个问题吗?你呢,怀旧?