Django/haystack-仅基于DateField的year元素进行faceting



假设我在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 }}&amp;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()

所有这些加在一起效果都很好。

相关内容

  • 没有找到相关文章

最新更新