POST后的Django表单重定向不可靠,需要success_url

时间:2015-08-31 12:05:28

标签: python django-forms http-post django-class-based-views

我完全不理解这种行为,并找到了一个我不太喜欢的解决方法。有人可以帮忙赐教吗?上下文是我有一个bootstrap样式表单来创建新记录(继承自generic.CreateView)

url.py:

url(r'^$', home, name='home'),
url(r'^main/$', views.MainView.as_view(), name='MainView'),
url(r'^topic/(?P<pk>[0-9]+)/$', catalogue_views.TopicView.as_view(), name='TopicView'),
url(r'^resource/(?P<pk>[0-9]+)/$', catalogue_views.DetailView.as_view(), name='ResourceDetail'),
url(r'^contribute/$', catalogue_views.ContributeView.as_view(success_url="/main/"), name='Contribute'),

views.py:

class ContributeView(generic.CreateView):
    template_name = "openeye/contribute.html"
    form_class = ContributeForm

    @method_decorator(login_required)
    def dispatch(self, *args, **kwargs):
        return super(ContributeView, self).dispatch(*args, **kwargs)

class MainView(generic.ListView):

    template_name = "openeye/main.html"
    context_object_name = 'topic_list'

    # TODO Make this only active topic areas?
    def get_queryset(self):
        return TopicArea.objects.all().order_by('name')

    @method_decorator(login_required)
    def dispatch(self, *args, **kwargs):
        return super(MainView, self).dispatch(*args, **kwargs)

forms.py:

class ContributeForm(forms.ModelForm):
    class Meta:
        model = CatalogueItem
        fields = ['title', 'topic_area', 'description', 'link', 'what_learn', 'how_apply', 'level', 'relevant_to', 'discovered_by']

    ROLE_CHOICES = [[x.id, x.job] for x in JobType.objects.all()]

    title = forms.CharField(widget=forms.TextInput(attrs={'placeholder': 'To sell this resource to others'}), max_length=80, required=True)
    description = forms.CharField(widget=forms.Textarea(attrs={'rows': 2, 'placeholder': 'Clear, e.g. format, duration, activities...'}))
    link = forms.CharField(widget=forms.URLInput(attrs={'placeholder': 'If required, link to resource http://...'}), required=False)
    what_learn = forms.CharField(widget=forms.Textarea(attrs={'rows': 3, 'placeholder':"This is important,."}), label='What will you learn?')
    how_apply = forms.CharField(widget=forms.Textarea(attrs={'rows': 3, 'placeholder':"How could this be put into action afterwards?"}), label='How could you apply this?')
    relevant_to = forms.MultipleChoiceField(widget=forms.CheckboxSelectMultiple, choices=ROLE_CHOICES)

以及带有表单的模板:

<div class="container">
        <div class="entry-form row">
            <div class="col-md-10 col-md-offset-1 col-sm-10 col-sm-offset-1 col-xs-10 col-xs-offset-1">
                <form action="{% url 'MainView' %}" method="post" class="form">
                    <input type="hidden" name="next" value="{{ next }}">
                    {% bootstrap_form form %}
                    <button class="btn btn-primary btn-lg" type="submit">Submit Suggestion</button>
                    {% csrf_token %}
                </form>
        </div>
    </div>

表单完美运行,数据很好地保存到数据库中。问题是之后,浏览器转到正确的URL / main /但是SCREEN IS BLANK。服务器显示HTTP 405 0,如果我刷新页面它可以工作。

如果我改变模板,那么动作=&#34; {%url&#39; Contribute&#39; %}&#34;为了返回相同的表单,我收到HTTP 500和一条关于“没有URL”的Django消息,以重定向到&#39;。所以两个不同的错误由重新定位确定。在这两种情况下,如果我只是点击浏览器网址字段并点击返回就可以了。

我确信这最初起作用然后破了,但我解决了它如下。 success_url中的硬编码使用它的路径

url(r'^contribute/$', catalogue_views.ContributeView.as_view(success_url="/main/"), name='Contribute'),

删除模板中的任何操作链接:

<form action="" method="post" class="form">

这是正确的做法吗?为什么,尽管使用了正确的网址,但是我的原始方法是不加载页面还是出错?我很想明白这一点。

1 个答案:

答案 0 :(得分:2)

您确定您的数据实际上已保存在服务器上吗?从您发布的内容来看,似乎不太可能。这是Django遵循的正常过程:

  1. 表单视图上的GET(ContributeView)   →返回空表格
  2. 表单视图上的POST(ContributeView)   →如果无效,请返回步骤1.如果有效,则返回302 Redirectsuccess_url
  3. 获取success_url
  4. 通常情况下,在模板中,表单操作应为空,因此表单会回发到生成它的视图中。并且ContributeView应该有一个成功的网址,可以重定向到您希望在以下位置发送给用户的位置:

    from django.core.urlresolvers import reverse_lazy
    class ContributeView(generic.CreateView):
        # other stuff
        success_url = reverse_lazy('MainView')
    

    使用405获得的行为是因为浏览器尝试将表单直接发送到MainView,而不是表单视图,它告诉浏览器它不知道如何处理{{ 1}}方法。