使用下拉菜单保存表单外键:IntegrityError XXX_id可能不为空 [英] Saving formset with drop-down-menu Foreignkey: IntegrityError XXX_id may not be NULL

查看:132
本文介绍了使用下拉菜单保存表单外键:IntegrityError XXX_id可能不为空的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我试图拥有一个表单,每个表单(PropertySelector)都有一个下拉菜单(PropertySelector.property),而该菜单的每个项目都是ForeignKey引用另一个模型(Property)。

I am trying to have a formset where each form (PropertySelector) has a drop-down menu (PropertySelector.property) whereas each item of that menu is ForeignKey reference to another model (Property).

不知何故当我试图提交并保存表单时,我得到:

Somehow when I am trying to submit and save the formset I am getting:

Exception Type:     IntegrityError
Exception Value:    testproj_propertyselector.property_id may not be NULL

它有什么问题,我该怎么办?我的整个代码如下。谢谢。

What is wrong with it and how can I get around it? My entire code is below. Thanks.

编辑:看起来像我的inline_formset问题(也许MySQL也)。请帮助我解决问题。

it looks like inline_formset problem to me (maybe MySQL also). Please, help me with workaround.

该项目称为testproj,我的应用程序也称为testproj。

The project is called testproj and my app is called testproj too.

首先我们填写属性:

>>> from testproj.models import Property
>>> p = Property(name='prop1', basic=True)
>>> p.save() 
>>> p = Property(name='prop2', basic=True)
>>> p.save()

models.py

models.py

from django.db import models

class PropertySet(models.Model):
    name = models.CharField(max_length=50)

class Property(models.Model):
    name = models.CharField(max_length=50)
    basic = models.BooleanField()

    def __unicode__(self):
        return u'%s' % (self.name)

class PropertySelector(models.Model):
    property_set = models.ForeignKey(PropertySet)
    title = models.CharField(max_length=50)
    property = models.ForeignKey(Property)



forms.py

from django.forms import ModelForm, TextInput, Select, ModelChoiceField
from django.db.models import Q
from testproj.models import Property, PropertySet, PropertySelector

class PropertySetForm(ModelForm):
    class Meta:
        model = PropertySet

def PropertySelForm():
    PropertyQueryset = Property.objects.filter(Q(basic=True))

    class PropertySelectorForm(ModelForm):
        property = ModelChoiceField(
            queryset=PropertyQueryset,
            widget=Select(attrs={'class': 'property'})
        )

        def __init__(self, *args, **kwargs): 
            super(ModelForm, self).__init__(*args, **kwargs)
            self.css_class = "prop_sel"

        class Meta:
            model = PropertySelector
            fields = ("property_set", "title")
            widgets = {"title" : TextInput(attrs={"class" : "title"})}

    return PropertySelectorForm

views.py

from django.shortcuts import render_to_response
from django.template import RequestContext
from django.forms.models import inlineformset_factory

from testproj.models import PropertySet, PropertySelector
from testproj.forms import PropertySetForm, PropertySelForm, PropertySelForm

def index(request):

    property_selector_form = PropertySelForm()
    PropertySelectorFormSet = inlineformset_factory(PropertySet, PropertySelector, form=property_selector_form)

    if request.method == "POST":

        property_set_form = PropertySetForm(request.POST)

        if property_set_form.is_valid():            
            saved_property_set = property_set_form.save()

            prop_sel_formset = PropertySelectorFormSet(request.POST, instance=saved_property_set)

            if prop_sel_formset.is_valid():
                prop_sel_formset.save()

        else:

            property_set_form = PropertySetForm()

            prop_sel_formset = PropertySelectorFormSet()    

        return render_to_response(
            "testproj/index.html",
            {
                "property_set_form": property_set_form,
                "prop_sel_formset": prop_sel_formset
            },
            context_instance=RequestContext(request)
        )

index.html(模板):

index.html (template):

{% block content %}

<head>

</head>

<body>

<form method="post" action=""> {% csrf_token %}

    {{ property_set_form.as_p }}

    {{ prop_sel_formset.management_form }}

    {% for form in prop_sel_formset %}
        {{ form }}

    {% endfor %}

    <input type="submit" value="Submit">    
</form>

</body>
{% endblock %}


推荐答案

code> property_id 未保存,因为您不包含在 form.Meta 中。离开字段,它的工作原理:

The property_id isn't saved because you don't include it in your form.Meta. Leave away the fields, and it works:

def PropertySelForm():
    PropertyQueryset = Property.objects.filter(Q(basic=True))

    class PropertySelectorForm(ModelForm):
        property = ModelChoiceField(
            queryset=PropertyQueryset,
            widget=Select(attrs={'class': 'property'})
        )

        def __init__(self, *args, **kwargs): 
            super(ModelForm, self).__init__(*args, **kwargs)
            self.css_class = "prop_sel"

        class Meta:
            model = PropertySelector
            #fields = ("property_set", "title")
            widgets = {"title" : TextInput(attrs={"class" : "title"})}

    return PropertySelectorForm

我会将 PropertySelForm 重命名为 property_selectorform_factory ,但这是
只是我:)

I'd rename PropertySelForm to property_selectorform_factory, but that's just me. :)

这篇关于使用下拉菜单保存表单外键:IntegrityError XXX_id可能不为空的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆