2013-12-25 21:13:18 +08:00
|
|
|
from django.db.models.aggregates import Aggregate
|
|
|
|
from django.contrib.gis.db.models.fields import GeometryField, ExtentField
|
2009-01-15 19:06:34 +08:00
|
|
|
|
2013-10-18 19:25:30 +08:00
|
|
|
__all__ = ['Collect', 'Extent', 'Extent3D', 'MakeLine', 'Union']
|
|
|
|
|
2013-11-03 01:18:46 +08:00
|
|
|
|
2013-12-25 21:13:18 +08:00
|
|
|
class GeoAggregate(Aggregate):
|
|
|
|
template = None
|
|
|
|
function = None
|
|
|
|
is_extent = False
|
|
|
|
|
|
|
|
def as_sql(self, compiler, connection):
|
|
|
|
if connection.ops.oracle:
|
|
|
|
if not hasattr(self, 'tolerance'):
|
|
|
|
self.tolerance = 0.05
|
|
|
|
self.extra['tolerance'] = self.tolerance
|
|
|
|
|
|
|
|
template, function = connection.ops.spatial_aggregate_sql(self)
|
|
|
|
if template is None:
|
|
|
|
template = '%(function)s(%(expressions)s)'
|
|
|
|
self.extra['template'] = self.extra.get('template', template)
|
|
|
|
self.extra['function'] = self.extra.get('function', function)
|
|
|
|
return super(GeoAggregate, self).as_sql(compiler, connection)
|
|
|
|
|
|
|
|
def prepare(self, query=None, allow_joins=True, reuse=None, summarize=False):
|
|
|
|
c = super(GeoAggregate, self).prepare(query, allow_joins, reuse, summarize)
|
|
|
|
if not isinstance(self.expressions[0].output_field, GeometryField):
|
|
|
|
raise ValueError('Geospatial aggregates only allowed on geometry fields.')
|
|
|
|
return c
|
|
|
|
|
2014-12-01 15:28:01 +08:00
|
|
|
def convert_value(self, value, connection, context):
|
2013-12-25 21:13:18 +08:00
|
|
|
return connection.ops.convert_geom(value, self.output_field)
|
|
|
|
|
|
|
|
|
|
|
|
class Collect(GeoAggregate):
|
2009-03-31 01:15:49 +08:00
|
|
|
name = 'Collect'
|
|
|
|
|
2013-11-03 01:18:46 +08:00
|
|
|
|
2013-12-25 21:13:18 +08:00
|
|
|
class Extent(GeoAggregate):
|
2009-01-15 19:06:34 +08:00
|
|
|
name = 'Extent'
|
2013-12-25 21:13:18 +08:00
|
|
|
is_extent = '2D'
|
|
|
|
|
|
|
|
def __init__(self, expression, **extra):
|
|
|
|
super(Extent, self).__init__(expression, output_field=ExtentField(), **extra)
|
|
|
|
|
2014-12-01 15:28:01 +08:00
|
|
|
def convert_value(self, value, connection, context):
|
|
|
|
return connection.ops.convert_extent(value, context.get('transformed_srid'))
|
2009-01-15 19:06:34 +08:00
|
|
|
|
2013-11-03 01:18:46 +08:00
|
|
|
|
2013-12-25 21:13:18 +08:00
|
|
|
class Extent3D(GeoAggregate):
|
2009-11-17 02:49:00 +08:00
|
|
|
name = 'Extent3D'
|
2013-12-25 21:13:18 +08:00
|
|
|
is_extent = '3D'
|
|
|
|
|
|
|
|
def __init__(self, expression, **extra):
|
|
|
|
super(Extent3D, self).__init__(expression, output_field=ExtentField(), **extra)
|
|
|
|
|
2014-12-01 15:28:01 +08:00
|
|
|
def convert_value(self, value, connection, context):
|
|
|
|
return connection.ops.convert_extent3d(value, context.get('transformed_srid'))
|
2009-11-17 02:49:00 +08:00
|
|
|
|
2013-11-03 01:18:46 +08:00
|
|
|
|
2013-12-25 21:13:18 +08:00
|
|
|
class MakeLine(GeoAggregate):
|
2009-01-15 19:06:34 +08:00
|
|
|
name = 'MakeLine'
|
|
|
|
|
2013-11-03 01:18:46 +08:00
|
|
|
|
2013-12-25 21:13:18 +08:00
|
|
|
class Union(GeoAggregate):
|
2009-01-15 19:06:34 +08:00
|
|
|
name = 'Union'
|