2012-09-30 20:31:28 +04:00
|
|
|
from rest_framework import serializers
|
2012-11-08 01:07:24 +04:00
|
|
|
from rest_framework.templatetags.rest_framework import replace_query_param
|
2012-09-30 20:31:28 +04:00
|
|
|
|
|
|
|
# TODO: Support URLconf kwarg-style paging
|
|
|
|
|
|
|
|
|
2012-11-08 01:09:26 +04:00
|
|
|
class NextPageField(serializers.Field):
|
2012-10-01 18:49:19 +04:00
|
|
|
"""
|
|
|
|
Field that returns a link to the next page in paginated results.
|
|
|
|
"""
|
2012-11-08 01:09:26 +04:00
|
|
|
page_field = 'page'
|
|
|
|
|
2012-09-30 20:31:28 +04:00
|
|
|
def to_native(self, value):
|
|
|
|
if not value.has_next():
|
|
|
|
return None
|
|
|
|
page = value.next_page_number()
|
2012-10-01 18:49:19 +04:00
|
|
|
request = self.context.get('request')
|
2012-11-09 01:47:54 +04:00
|
|
|
url = request and request.build_absolute_uri() or ''
|
2012-11-08 01:07:24 +04:00
|
|
|
return replace_query_param(url, self.page_field, page)
|
2012-09-30 20:31:28 +04:00
|
|
|
|
|
|
|
|
2012-11-08 01:09:26 +04:00
|
|
|
class PreviousPageField(serializers.Field):
|
2012-10-01 18:49:19 +04:00
|
|
|
"""
|
|
|
|
Field that returns a link to the previous page in paginated results.
|
|
|
|
"""
|
2012-11-08 01:09:26 +04:00
|
|
|
page_field = 'page'
|
|
|
|
|
2012-09-30 20:31:28 +04:00
|
|
|
def to_native(self, value):
|
|
|
|
if not value.has_previous():
|
|
|
|
return None
|
|
|
|
page = value.previous_page_number()
|
2012-10-01 18:49:19 +04:00
|
|
|
request = self.context.get('request')
|
2012-11-09 01:47:54 +04:00
|
|
|
url = request and request.build_absolute_uri() or ''
|
2012-11-08 01:07:24 +04:00
|
|
|
return replace_query_param(url, self.page_field, page)
|
2012-09-30 20:31:28 +04:00
|
|
|
|
|
|
|
|
2012-10-01 18:49:19 +04:00
|
|
|
class PaginationSerializerOptions(serializers.SerializerOptions):
|
|
|
|
"""
|
|
|
|
An object that stores the options that may be provided to a
|
|
|
|
pagination serializer by using the inner `Meta` class.
|
|
|
|
|
|
|
|
Accessible on the instance as `serializer.opts`.
|
|
|
|
"""
|
|
|
|
def __init__(self, meta):
|
|
|
|
super(PaginationSerializerOptions, self).__init__(meta)
|
|
|
|
self.object_serializer_class = getattr(meta, 'object_serializer_class',
|
|
|
|
serializers.Field)
|
|
|
|
|
|
|
|
|
|
|
|
class BasePaginationSerializer(serializers.Serializer):
|
|
|
|
"""
|
|
|
|
A base class for pagination serializers to inherit from,
|
|
|
|
to make implementing custom serializers more easy.
|
|
|
|
"""
|
|
|
|
_options_class = PaginationSerializerOptions
|
2012-10-02 13:40:43 +04:00
|
|
|
results_field = 'results'
|
2012-10-01 18:49:19 +04:00
|
|
|
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
|
|
"""
|
|
|
|
Override init to add in the object serializer field on-the-fly.
|
|
|
|
"""
|
|
|
|
super(BasePaginationSerializer, self).__init__(*args, **kwargs)
|
2012-10-02 13:40:43 +04:00
|
|
|
results_field = self.results_field
|
2012-10-01 18:49:19 +04:00
|
|
|
object_serializer = self.opts.object_serializer_class
|
2013-01-23 11:38:13 +04:00
|
|
|
|
|
|
|
if 'context' in kwargs:
|
|
|
|
context_kwarg = {'context': kwargs['context']}
|
|
|
|
else:
|
|
|
|
context_kwarg = {}
|
|
|
|
|
|
|
|
self.fields[results_field] = object_serializer(source='object_list', **context_kwarg)
|
2012-09-30 20:31:28 +04:00
|
|
|
|
|
|
|
def to_native(self, obj):
|
|
|
|
"""
|
|
|
|
Prevent default behaviour of iterating over elements, and serializing
|
|
|
|
each in turn.
|
|
|
|
"""
|
|
|
|
return self.convert_object(obj)
|
2012-10-01 18:49:19 +04:00
|
|
|
|
|
|
|
|
|
|
|
class PaginationSerializer(BasePaginationSerializer):
|
|
|
|
"""
|
|
|
|
A default implementation of a pagination serializer.
|
|
|
|
"""
|
|
|
|
count = serializers.Field(source='paginator.count')
|
|
|
|
next = NextPageField(source='*')
|
|
|
|
previous = PreviousPageField(source='*')
|