Django:基于类的ListView中的搜索表单

neu*_*rix 25 forms django listview django-class-based-views class-based-views

我试图实现一个Class Based ListView显示选择的表集.如果第一次请求站点,则应显示数据集.我更喜欢POST提交,但GET也没关系.

这是一个容易处理的问题,function based views但是基于类的视图我很难理解.

我的问题是我得到了各种错误,这是由于我对基于分类的观点的理解有限.我已阅读各种文档,我理解直接查询请求的视图,但只要我想在查询语句中添加一个表单,我就遇到了不同的错误.对于下面的代码,我会收到一个ValueError: Cannot use None as a query value.

根据表单条目(否则选择整个数据库),基于类的ListView的最佳实践工作流程是什么?

这是我的示例代码:

models.py

class Profile(models.Model):
    name = models.CharField(_('Name'), max_length=255)

    def __unicode__(self):
        return '%name' % {'name': self.name}

    @staticmethod
    def get_queryset(params):

        date_created = params.get('date_created')
        keyword = params.get('keyword')
        qset = Q(pk__gt = 0)
        if keyword:
            qset &= Q(title__icontains = keyword)
        if date_created:
            qset &= Q(date_created__gte = date_created)
        return qset
Run Code Online (Sandbox Code Playgroud)

forms.py

class ProfileSearchForm(forms.Form):
    name = forms.CharField(required=False)
Run Code Online (Sandbox Code Playgroud)

views.py

class ProfileList(ListView):
    model = Profile
    form_class = ProfileSearchForm
    context_object_name = 'profiles'
    template_name = 'pages/profile/list_profiles.html'
    profiles = []


    def post(self, request, *args, **kwargs):
        self.show_results = False
        self.object_list = self.get_queryset()
        form = form_class(self.request.POST or None)
        if form.is_valid():
            self.show_results = True
            self.profiles = Profile.objects.filter(name__icontains=form.cleaned_data['name'])
        else:
            self.profiles = Profile.objects.all()
        return self.render_to_response(self.get_context_data(object_list=self.object_list, form=form))

    def get_context_data(self, **kwargs):
        context = super(ProfileList, self).get_context_data(**kwargs)
        if not self.profiles:
            self.profiles = Profile.objects.all()
        context.update({
            'profiles': self.profiles
        })
        return context
Run Code Online (Sandbox Code Playgroud)

下面我添加了FBV来完成这项工作.如何将此功能转换为CBV? 它在基于函数的视图中似乎很简单,但在基于类的视图中却不是这么简单.

def list_profiles(request):
    form_class = ProfileSearchForm
    model = Profile
    template_name = 'pages/profile/list_profiles.html'
    paginate_by = 10

    form = form_class(request.POST or None)
    if form.is_valid():
        profile_list = model.objects.filter(name__icontains=form.cleaned_data['name'])
    else:
        profile_list = model.objects.all()

    paginator = Paginator(profile_list, 10) # Show 10 contacts per page
    page = request.GET.get('page')
    try:
        profiles = paginator.page(page)
    except PageNotAnInteger:
        profiles = paginator.page(1)
    except EmptyPage:
        profiles = paginator.page(paginator.num_pages)

    return render_to_response(template_name, 
            {'form': form, 'profiles': suppliers,}, 
            context_instance=RequestContext(request))
Run Code Online (Sandbox Code Playgroud)

Ots*_*lal 39

我认为您的目标是尝试根据表单提交过滤查询集,如果是这样,使用GET:

class ProfileSearchView(ListView)
    template_name = '/your/template.html'
    model = Person

    def get_queryset(self):
        try:
            name = self.kwargs['name']
        except:
            name = ''
        if (name != ''):
            object_list = self.model.objects.filter(name__icontains = name)
        else:
            object_list = self.model.objects.all()
        return object_list
Run Code Online (Sandbox Code Playgroud)

然后,您需要做的就是编写一个get方法来渲染模板和上下文.

也许不是最好的方法.通过使用上面的代码,您无需定义django表单.

以下是它的工作原理:基于类的视图分离了渲染模板,处理表单等的方式.比如,get处理GET响应,post处理POST响应,get_queryset并且get_object是自我解释的,等等.知道什么方法可用的简单方法,启动shell并键入:

from django.views.generic import ListView 如果你想了解 ListView

然后输入dir(ListView).在那里你可以看到定义的所有方法,并访问源代码来理解它.get_queryset用于获取查询集的方法.为什么不像这样定义它,它也有效:

class FooView(ListView):
    template_name = 'foo.html'
    queryset = Photo.objects.all() # or anything
Run Code Online (Sandbox Code Playgroud)

我们可以像上面那样做,但我们不能通过使用这种方法进行动态过滤.通过使用get_queryset我们可以使用我们所拥有的任何数据/值/信息进行动态过滤,这意味着我们也可以使用name由您指定的任何参数在何处发送的参数,或者在其中GET可用的参数.kwargsself.kwargs["some_key"]some_key

  • 而不是使用尝试你可以使用默认值的get,我认为它有助于提高可读性.`name = self.kargs.get('name',None)`然后``如果名字:#做一些东西` (6认同)

jas*_*isz 14

好吧,我认为将验证留给形式是个好主意.在这种特殊情况下可能不值得,因为它是非常简单的形式 - 但肯定有更复杂的一个(也许你的也会增长),所以我会做类似的事情:

class ProfileList(ListView):
    model = Profile
    form_class = ProfileSearchForm
    context_object_name = 'profiles'
    template_name = 'pages/profile/list_profiles.html'
    profiles = []


    def get_queryset(self):
        form = self.form_class(self.request.GET)
        if form.is_valid():
            return Profile.objects.filter(name__icontains=form.cleaned_data['name'])
        return Profile.objects.all()
Run Code Online (Sandbox Code Playgroud)

  • @VicNicethemer它应该是form = self.form_class(s​​elf.request.GET). (3认同)

sup*_*amp 6

这类似于@jasisz的方法,但更简单。

class ProfileList(ListView):
    template_name = 'your_template.html'
    model = Profile

    def get_queryset(self):
        query = self.request.GET.get('q')
        if query:
            object_list = self.model.objects.filter(name__icontains=query)
        else:
            object_list = self.model.objects.none()
        return object_list
Run Code Online (Sandbox Code Playgroud)

然后,您在html模板上要做的就是:

<form method='GET'>
  <input type='text' name='q' value='{{ request.GET.q }}'>
  <input class="button" type='submit' value="Search Profile">
</form>
Run Code Online (Sandbox Code Playgroud)