假设我在Django实现中有以下模型:
class Broadcast(models.Model):
broadcast_date = models.DateField()
...
在我的大海捞针/Solr驱动的搜索结果中,我希望能够仅根据广播年份而不是确切的日期来划分。
我可以为此创建一个新的模型属性,但似乎我应该能够在search_indexes模型定义中以某种方式做到这一点。以下是我目前在search_indexes.py中的内容,仅截止到完整日期:
class BroadcastIndex(indexes.SearchIndex, indexes.Indexable):
text = [...]
broadcast_date = indexes.DateField(model_attr="broadcast_date", faceted=True)
有什么想法吗?
您应该能够使用date_facet()。如果您将其gap_by
参数设置为year
,则它将按年份分面。
只是概述一下我最终做了什么。
首先,在我的urls.py文件中,我使用date_facet()
包含了一个自定义的SearchQuerySet(感谢Adrian Ghiuta!),我将其传递给了我的搜索视图:
...
from haystack.forms import FacetedSearchForm
from haystack.query import SearchQuerySet
sqs = SearchQuerySet().date_facet('broadcast_date', start_date=date(1955, 1, 1), end_date=date(2015, 1, 1), gap_by="year")
urlpatterns = [
...
url(r'^search/', views.SarkSearch(form_class=FacetedSearchForm, searchqueryset=sqs), name='haystack_search'),
]
date_facet()
函数确实提供了一个带有年份计数的分面日期字典,但返回的键的格式是yyyy-01:00:00:00,所以我最终只是从该字符串中截取了年份。我使用列表理解创建了一个新的列表,其中只包含我想要的数据,然后我将其添加到facets上下文变量中。我使用的是一个基于类的视图子类FacetedSearchView,所以我需要将所有处理代码放在extra_context()
函数中:
class FooSearch(FacetedSearchView):
def extra_content(self):
extra = super(FooSearch, self).extra_context()
dates = extra['facets']['dates']['broadcast_date']
dates = sorted([[year[:4], count] for year, count in dates.items() if count > 0])
extra['facets']['dates']['broadcast_date'] = dates
return extra
现在我有了返回的年份方面和计数,我需要想出一种方法,将实际搜索结果缩小到仅在给定年份内的搜索结果。在我的搜索模板中,我使用GET变量"year_facet"将每个返回的方面日期绑定到一个方面链接。看起来像这样:
<dl>
{% if facets.dates.broadcast_date %}
<dt>Broadcast dates</dt>
{% for date in facets.dates.broadcast_date %}
<dd><a href="{{ request.get_full_path }}&year_facet={{ date.0 }}">{{ date.0 }}</a> ({{ date.1 }})</dd>
{% endfor %}
{% endif %}
{% if 'year_facet' in request.get_full_path %}
<!-- I want a way to clear the facet if one is present -->
<dd><a href="?q={{ query }}"><em>see all</em></a></dd>
{% endif %}
</dl>
现在链接已经设置好了,当选择基于年份的日期方面时,我需要一种方法来缩小视图的范围。为此,在我的搜索视图中,我覆盖了get_results()
函数,以包含给定年份内所有日期的日期过滤器:
def get_results(self):
if 'year_facet' in self.request.GET:
year = int(self.request.GET['year_facet'])
return self.form.search().filter(broadcast_date__lte=datetime.date(year, 12, 31)).filter(broadcast_date__gte=datetime.date(year, 1, 1))
return self.form.search()
所有这些加在一起效果都很好。