Django(主干)和基于类的通用视图:一个表单的初始数据出现在另一个表单中

时间:2011-02-16 16:24:26

标签: python django django-class-based-views

我遇到了一个奇怪的问题,即数据似乎在不同的视图和请求中持续存在,直到服务器重新启动。

我已设法将问题缩减为以下代码:

# foobar/models.py
from django.db import models

class Foo(models.Model):
    bug = models.CharField(max_length=10)


# foobar/forms.py
from django import forms
from foobar.models import Foo

class CreateForm(forms.ModelForm):
    class Meta:
        model = Foo

class UpdateForm(forms.ModelForm):
    class Meta:
        model = Foo

    def __init__(self, *args, **kwargs):
        kwargs.setdefault('initial', {})
        kwargs['initial'].update({'bug': 'WHY??'})
        super(UpdateForm, self).__init__(*args, **kwargs)


# foobar/views.py
from django.views.generic.edit import CreateView, UpdateView
from foobar.forms import CreateForm, UpdateForm
from foobar.models import Foo

class FooCreateView(CreateView):
    form_class = CreateForm
    template_name = 'foobar/foo_form.html'

create = FooCreateView.as_view()

class FooUpdateView(UpdateView):
    form_class = UpdateForm
    template_name = 'foobar/foo_form.html'
    queryset = Foo.objects.all()

update = FooUpdateView.as_view()


# foobar/urls.py
from django.conf.urls.defaults import *

urlpatterns = patterns('foobar.views',
    ('^$', 'create'),
    (r'^(?P<pk>\d+)/$', 'update'),
)

您还应该添加一个模板(例如foobar/templates/foo_form.html):

<form action="" method="post">
{{ form.as_p }}
<input type="submit" />
{% csrf_token %}
</form>

要重现,请执行以下操作:

  • 将foobar应用添加到settings.INSTALLED_APPS
  • 运行syncdb
  • foobar.urls添加到您的root urlconf
  • 导航到/foobar/(实际网址取决于您的root urlconf)
  • 提交表单(从而创建新的Foo对象)
  • 导航至/foobar/1/。请注意,表单字段已预先填充(这是预期的)
  • 导航至/foobar/。请注意,表单字段仍然填充(这不是预期的)。

这是一个错误还是我在做一些我不应该做的事情(或者两者都是......)?

- 编辑 -

在forms.py中,如果我用此替换update调用:

kwargs['initial']['bug'] = 'WHY???'

然后问题仍然存在。

注释掉该行可以解决问题(但是表格显然没有初始数据)。

2 个答案:

答案 0 :(得分:1)

因为你正在改变传入的kwargs,它来自视图类中的类级属性。

而是复制它们并更新副本:

initial_defaults = {'bug': 'no'}
initial_defaults.update(kwargs.get('initial', {}))
defaults = kwargs.copy()
defaults['initial'] = initial_defaults 

答案 1 :(得分:0)

您可能想要指定Django-1.3开发,Django 1.2.5中不存在泛型类视图。在forms.py文件中,您可以对以下行进行评论,然后重试:

class UpdateForm(forms.ModelForm):
    class Meta:
        model = Foo

    def __init__(self, *args, **kwargs):
        #kwargs.setdefault('initial', {})
        #kwargs['initial'].update({'bug': 'WHY??'})
        super(UpdateForm, self).__init__(*args, **kwargs)