当前位置: 首页 > 面试题库 >

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

秦涵涤
2023-03-14
问题内容

我试图实现一个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

forms.py

class ProfileSearchForm(forms.Form):
    name = forms.CharField(required=False)

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

在下面,我添加了完成任务的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))

问题答案:

我认为你的目标是尝试使用GET过滤基于表单提交的queryset:

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

然后,你所需要做的就是编写一个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

我们可以像上面那样做,但是不能使用这种方法进行动态过滤。通过使用get_queryset我们可以进行动态过滤,使用我们拥有的任何数据/值/信息,这意味着我们还可以使用name由发送的参数GET,并且该参数在上可用kwargs,在这种情况下,你指定的任何参数在self.kwargs["some_key"]哪里some_key



 类似资料:
  • 问题内容: 主视图是一个简单的分页ListView,我想向其中添加搜索表单。 我认为这样可以解决问题: 但是显然我错了..我在官方文档中找不到该怎么做的方法。 建议? 问题答案: 这些答案对引导我朝正确的方向大有帮助。谢谢大家 对于我的实现,我需要一个窗体视图,该窗体视图同时在get和post上返回ListView。我不喜欢重复get函数的内容,但需要进行一些更改。现在,self.form也可以从

  • 问题内容: 如何在Django 1.3中使用分页? 该文档对此不是很清楚。 我views.py怎么了? 我的模板如何处理? URLconf文件中有什么内容? 问题答案: 我认为你需要有关将分页与基于新类的视图一起使用的信息,因为使用基于传统函数的视图很容易找到。我发现仅通过设置变量就足以激活分页。请参见基于类的通用视图。 例如,在你的: 在你的模板(),你可以包括这样的分页部分(我们有一些情境变量

  • 问题内容: 有谁知道或可以请提供一个简单的Django基于类的通用DeleteView示例吗?我想继承DeleteView的子类,并确保当前登录的用户在删除对象之前拥有该对象的所有权。任何帮助将不胜感激。先感谢你。 问题答案: 这是一个简单的例子: 注意事项: 该不会删除请求; 这是你提供带有此视图的“是,我确定”按钮的确认模板(你可以在class属性中提供名称)的机会。 你可能更喜欢错误消息而不

  • 我想做一个字段折叠热门点击聚合,正如这里所记录的: https://www.elastic.co/guide/en/elasticsearch/reference/current/search-aggregations-metrics-top-hits-aggregation.html#_field_collapse_example 特别是,这一部分是一个问题: 因为生产环境配置有: 这意味着我不

  • 本文向大家介绍edittext + listview 实现搜索listview中的内容方法(推荐),包括了edittext + listview 实现搜索listview中的内容方法(推荐)的使用技巧和注意事项,需要的朋友参考一下 主要原理:是在主界面有两个空间,一个是EditText,一个是ListView,ListView是放在EditText下面的,然后自定义建立一个adapter适配器,

  • 基于另一个子DTO的属性值从arraylist中搜索元素的最佳方法是什么。 场景:我有Employee对象,它有另一个ContactDetails对象。我需要从extNumber为1234的员工列表中查找员工(extNumber是ContactDetails中的属性。我需要在创建新员工和分配新ext时检查ext是否可用。 无法直接使用JDBC查询,因为我正在使用corba api。没有任何提供此类