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

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

以上是 Django:基于类的ListView中的搜索表单 的全部内容, 来源链接: utcz.com/qa/428703.html

回到顶部