2017-11-07 11:08:23 -05:00
|
|
|
import django_filters
|
2019-02-14 09:43:56 -05:00
|
|
|
from django.conf import settings
|
2019-01-31 13:36:30 -05:00
|
|
|
from django.db.models import Q
|
2018-11-12 15:48:58 -05:00
|
|
|
from taggit.models import Tag
|
2016-09-15 17:12:53 -04:00
|
|
|
|
2017-03-22 09:39:30 -04:00
|
|
|
|
|
|
|
|
class NumericInFilter(django_filters.BaseInFilter, django_filters.NumberFilter):
|
|
|
|
|
"""
|
|
|
|
|
Filters for a set of numeric values. Example: id__in=100,200,300
|
|
|
|
|
"""
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
2017-07-27 09:53:41 -04:00
|
|
|
class NullableCharFieldFilter(django_filters.CharFilter):
|
2018-07-30 14:00:37 -04:00
|
|
|
"""
|
|
|
|
|
Allow matching on null field values by passing a special string used to signify NULL.
|
|
|
|
|
"""
|
2017-07-27 09:53:41 -04:00
|
|
|
|
|
|
|
|
def filter(self, qs, value):
|
2019-02-14 09:43:56 -05:00
|
|
|
if value != settings.FILTERS_NULL_CHOICE_VALUE:
|
2018-11-27 10:52:24 -05:00
|
|
|
return super().filter(qs, value)
|
2019-02-14 09:43:56 -05:00
|
|
|
qs = self.get_method(qs)(**{'{}__isnull'.format(self.field_name): True})
|
2017-07-27 09:53:41 -04:00
|
|
|
return qs.distinct() if self.distinct else qs
|
|
|
|
|
|
|
|
|
|
|
2018-11-12 15:48:58 -05:00
|
|
|
class TagFilter(django_filters.ModelMultipleChoiceFilter):
|
|
|
|
|
"""
|
|
|
|
|
Match on one or more assigned tags. If multiple tags are specified (e.g. ?tag=foo&tag=bar), the queryset is filtered
|
|
|
|
|
to objects matching all tags.
|
|
|
|
|
"""
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
|
|
2018-11-12 16:03:52 -05:00
|
|
|
kwargs.setdefault('field_name', 'tags__slug')
|
2018-11-12 15:48:58 -05:00
|
|
|
kwargs.setdefault('to_field_name', 'slug')
|
|
|
|
|
kwargs.setdefault('conjoined', True)
|
|
|
|
|
kwargs.setdefault('queryset', Tag.objects.all())
|
|
|
|
|
|
2018-11-27 10:52:24 -05:00
|
|
|
super().__init__(*args, **kwargs)
|
2019-01-31 13:36:30 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
class NameSlugSearchFilterSet(django_filters.FilterSet):
|
|
|
|
|
"""
|
|
|
|
|
A base class for adding the search method to models which only expose the `name` and `slug` fields
|
|
|
|
|
"""
|
|
|
|
|
q = django_filters.CharFilter(
|
|
|
|
|
method='search',
|
|
|
|
|
label='Search',
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def search(self, queryset, name, value):
|
|
|
|
|
if not value.strip():
|
|
|
|
|
return queryset
|
|
|
|
|
return queryset.filter(
|
|
|
|
|
Q(name__icontains=value) |
|
|
|
|
|
Q(slug__icontains=value)
|
|
|
|
|
)
|