Django - 如何使用多个字段和ModelForm执行搜索

时间:2012-04-23 00:11:45

标签: django search django-queryset django-forms

我正在寻找使用ModelForm类构建搜索表单,该表允许用户通过特定字段而不是单个字段关键字搜索来搜索模型中的值。我也希望忽略空字段。

我研究了Haystack和Djapian等搜索引擎选项,以及在Django中手动解决问题,但在查询多个字段时似乎无法获得结果。

这是我到目前为止的观点,部分基于本网站上一些类似问题的答案,以及记录的内容:

def search(request):
    error = False
    form = InfoForm()
    if 'field1' or 'field2' or 'field3' in request.GET:
        form = InfoForm(request.GET)
        if form.is_valid():
            cd = form.cleaned_data
            field1 = cd['field1']
            field2 = cd['field2']
            field3 = cd['field3']
            if not(field1 or field2 or field3):
                error = True
            else:
                query = Q()
                if request.GET['field1']:
                    query &= Q(field1__icontains=request.GET['field1'])
                if request.GET['field2']:
                    query &= Q(field2__icontains=request.GET['field2'])
                if request.GET['field3']:
                    query &= Q(field3__icontains=request.GET['field3'])

                results = PersonInfo.objects.filter(query).distinct()

                return render_to_response('search/personsearch.html',
                        {'query': query,
                         'field1': field1,
                         'field2': field2,
                         'field3': field3,
                         'results': results,
                        },
                    context_instance=RequestContext(request))

    return render_to_response('search/infosearch.html',
            {'error': error,
             'form': form},
        context_instance=RequestContext(request))

我也直接从文档中尝试了这个复杂的查询:

query = InfoModel.objects.get(
    Q(field1__icontains=field1) |
    Q(field2__icontains=field2) |
    Q(field3__icontains=field3)
)

在这两种情况下,查询字符串?field1=&field2=&field3=都会附加到URL中,并且页面将返回字段中提供的字段值,如预期的那样;但如果所有字段都留空,则既不返回结果也不生成错误。

我显然错过了一些东西,但对于我的生活,我无法弄清楚是什么。还有其他人遇到过这个问题吗?

1 个答案:

答案 0 :(得分:1)

在调试搜索视图时,我发现针对if的初始request.GET条件仅返回True一个字段,即使列出的所有字段实际上都在表单中。

从那里,实现我的多字段搜索的最简单方法是将.filter()方法串在一起。搜索表单中的空字段将被忽略,但如果所有字段都为空,则会返回到搜索表单并且处理错误。

def search(request):
    errors = []
    if 'field1' in request.GET:
        field1 = request.GET['field1']
        field2 = request.GET['field2']
        field3 = request.GET['field3']
        if not ((field1 or field2) or field3):
            errors.append('Enter a search term.')
        else:
            results = MyModel.objects.filter(
                field1__icontains=field1
            ).filter(
                field2__icontains=field2
            ).filter(
                field3__icontains=field3
            )
            query = "Field 1: %s, Field 2: %s, Field 3: %s" % (field1, field2, field3)
            return render_to_response('search/search_results.html',
                    {'results': results, 'query': query})
    return render_to_response('search/search_form.html',
            {'errors': errors})

此处的query字符串现在只返回一个格式化的搜索字词列表,您可以将其放入模板中。但是,如果需要,您也可以直接返回变量并将其以另一种形式放在搜索结果页面上。

分页也适用于搜索结果,这对于任何实际应用程序都是必不可少的。将上面的results重命名为results_list,然后将the Django documentation on Pagination中的代码插入到您的视图中。我发现一个警告,您必须添加一个包含搜索字符串的新变量,并将其包含在您的分页链接前面,或者当您单击定义的“下一个”或“上一个”链接时返回搜索表单在文档中。

最后,当您使用HTML构建表单而不依赖于Django的ModelForm类时,它似乎效果最佳。这样您就不必试图调试表单验证错误,并且可以在视图函数中定义自己的验证规则。