将简单搜索添加到Django应用程序

时间:2013-01-09 16:51:23

标签: python django forms postgresql search

所以我正在关注this教程,以便搜索我的一些模型。但是,由于缺少所提供内容的文档,并且作为Django的新手,我对于为了使这项工作缺少什么感到困惑。

所以这就是我所拥有的:

修改

修改搜索模板以包含用于获取查询的输入字段。

myproject.templates.search.html:

<form action="" method="get">
<label for="id_q">Search:</label>
<input id="id_q" name="q" type="text">
<input type="submit" value="Submit">

{% if found_entries %}
    <p>You searched for "{{ query_string }}".</p>
    <ul>
        {% for i in found_entries %}
            {{ i.uid }} {{ i.title }} {{ value|linebreaks }}
        {% endfor %}
    </ul>
{% endif %}

{% if query_string and not found_entries %}
    <p>No results found.</p>
{% else %}
    <p>Type a search query into the box above, and press "Submit" to search.</p>
{% endif %}

</form>

myapp.models.py:

from django.db import models

class Book(models.Model):
    uid = models.IntegerField(primary_key=True)
    title = models.CharField(max_length=30)
    class Meta:
        db_table = u'books'

myapp.search.py​​:

import re

from django.db.models import Q

def normalize_query(query_string,
                findterms=re.compile(r'"([^"]+)"|(\S+)').findall,
                normspace=re.compile(r'\s{2,}').sub):
''' Splits the query string in invidual keywords, getting rid of unecessary spaces
    and grouping quoted words together.
    Example:

    >>> normalize_query('  some random  words "with   quotes  " and   spaces')
    ['some', 'random', 'words', 'with quotes', 'and', 'spaces']

'''
return [normspace(' ', (t[0] or t[1]).strip()) for t in findterms(query_string)] 

def get_query(query_string, search_fields):
''' Returns a query, that is a combination of Q objects. That combination
    aims to search keywords within a model by testing the given search fields.

'''
query = None # Query to search for every search term        
terms = normalize_query(query_string)
for term in terms:
    or_query = None # Query to search for a given term in each field
    for field_name in search_fields:
        q = Q(**{"%s__icontains" % field_name: term})
        if or_query is None:
            or_query = q
        else:
            or_query = or_query | q
    if query is None:
        query = or_query
    else:
        query = query & or_query
return query

myapp.views.py:

from myapp.models import Book
from django.shortcuts import render_to_response
from django.template import RequestContext

def search(request):
    query_string = ''
    found_entries = None
    search_fields=('uid')

    if ('q' in request.GET) and request.GET['q'].strip():

        query_string = request.GET['q']

        entry_query = get_query(query_string, search_fields)

        found_entries = Book.objects.filter(entry_query)

        return render_to_response('search.html',
                         { 'query_string': query_string, 'found_entries': found_entries },
                         context_instance=RequestContext(request))

myproject.templates.search.html:

{% if found_entries %}
    <p>You searched for "{{ query_string }}".</p>
    <ul>
    {% for i in found_entries %}
        <li><a href="{{ q.get_absolute_url }}">{{ found_entries }}</a></li>
    {% endfor %}
    </ul>
{% endif %}
{% if query_string and not found_entries %}
    <p>No results found.</p>
{% else %}
    <p>Type a search query into the box above, and press "Submit" to search.</p>
{% endif %}

myproject.urls.py

from django.conf.urls import patterns, include, url

urlpatterns = patterns('',
url(r'^predictor/$', 'myapp.views.search'),
)

如果我去的话,这就是它的样子:http://localhost:8000/myapp/

图片:http://i.imgur.com/QFaWZ.png

谢谢!

2 个答案:

答案 0 :(得分:2)

您的模板中没有任何表单:您从不在页面上放置搜索框,那么为什么要显示?

查看django documentation on using forms


这需要相当多的工作,但你没有使用任何Django的内置机制来创建表单。这样做可以使您的代码更清晰(并且更容易!)。

答案 1 :(得分:1)

如果您看一下您的代码(我鼓励您这样做),您正在查看的视图是myapp.views.search对吗?查看你的搜索功能,你应该能够看到它查看request.GET中的'q'键,这对于从查询字符串中获取'q'元素是相同的,所以这就是它从中获取搜索的位置。 / p>

所以要搜索,你会转到http://localhost:8000/myapp/?q=searchterm,其中'searchterm'指的是你正在搜索的词。

就像托马斯所说,你可能也想创建一个表单来使用这个功能。