如何让Django的通用视图使用带初始值的表单?

2 投票
3 回答
3950 浏览
提问于 2025-04-15 11:49

我知道怎么从视图中给表单设置初始值。但是,怎么让一个通用视图给表单设置初始值呢?我可以为这个通用视图写一个包装视图,但我还是无法访问到表单对象的实例。

我想实现的具体目标是让用户创建一个新对象,使用 create_object 这个通用视图。不过,我想把这个对象的 'user' 字段设置为当前登录的用户,而这个用户信息只能通过 request.user 来获取。那我该怎么初始化表单,让这个字段有这个值呢?

补充:我遇到了 __new__。这个方法可以用一些默认参数调用它自己的构造函数吗?

非常感谢。

3 个回答

0

如果你想要通用视图的所有功能,那么你可以用原来的视图作为模板,创建一个新的通用视图。

比如:

def create_object_with_initial(request, model=None, template_name=None,
    template_loader=loader, extra_context=None, post_save_redirect=None,
    login_required=False, context_processors=None, form_class=None, initial=None):

if extra_context is None: extra_context = {}
if login_required and not request.user.is_authenticated():
    return redirect_to_login(request.path)

model, form_class = get_model_and_form_class(model, form_class)
if request.method == 'POST':
    form = form_class(request.POST, request.FILES)
    if form.is_valid():
        new_object = form.save()

        msg = ugettext("The %(verbose_name)s was created successfully.") %\
                                {"verbose_name": model._meta.verbose_name}
        messages.success(request, msg, fail_silently=True)
        return redirect(post_save_redirect, new_object)
else:
    print "creating", form_class, " with initial data ", initial
    form = form_class(initial=initial)

# Create the template, context, response
if not template_name:
    template_name = "%s/%s_form.html" % (model._meta.app_label, model._meta.object_name.lower())
t = template_loader.get_template(template_name)
c = RequestContext(request, {
    'form': form,
}, context_processors)
apply_extra_context(extra_context, c)
return HttpResponse(t.render(c))

这个代码是从 /site-packages/django/views/generic/create_update.py 复制过来的,只有第3行和第21行做了修改,以便加入初始数据。

然后你可以像你想的那样使用它:

object_info = {
'model': YourModel,
'initial': {'data' : 'Initial Value'},
'template_name': 'template.html'
}

url(r'^path/$',
    login_required(create_object_with_initial),
    object_info,
    name='url_name'),

这样应该就能正常工作了。

3

你可以通过一个通用的视图包装器来实现这个功能,方法是动态创建一个表单类,然后把它传递给通用视图。不过,这样的解决办法可能比问题本身还要复杂。最简单的办法就是自己写一个视图,然后耐心等待这个功能的更新

4

很遗憾,你不能通过Django的 create_object 通用视图来实现这个功能;你需要自己写一个。不过,这其实很简单。

首先,你需要创建一个新的表单类,像这样:

from django import forms
class MyForm(forms.ModelForm):
    class Meta:
        model = MyModel  # model has a user field

然后你就可以创建一个视图,像这样:

from django.shortcuts import render_to_response
from django.template import RequestContext
from django.http import HttpResponseRedirect
from django.contrib.auth.decorators import login_required

@login_required
def create_mymodel(request):
    if request.method == 'POST':
        # Get data from form
        form = MyForm(request.POST)
        # If the form is valid, create a new object and redirect to it.
        if form.is_valid():
            newObject = form.save()
            return HttpResponseRedirect(newObject.get_absolute_url())
    else:
        # Fill in the field with the current user by default
        form = MyForm(initial={'user': request.user})
    # Render our template
    return render_to_response('path/to/template.html',
        {'form': form},
        context_instance=RequestContext(request))

撰写回答