Fixed #12312 -- Set the coordinate dimension on each component of geometry collections after transform (refines GDAL bug workaround introduced in r11628). Thanks, yourcelf for bug report.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@12878 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
parent
7d25682108
commit
87302ac009
|
@ -48,11 +48,11 @@ from django.contrib.gis.gdal.base import GDALBase
|
||||||
from django.contrib.gis.gdal.envelope import Envelope, OGREnvelope
|
from django.contrib.gis.gdal.envelope import Envelope, OGREnvelope
|
||||||
from django.contrib.gis.gdal.error import OGRException, OGRIndexError, SRSException
|
from django.contrib.gis.gdal.error import OGRException, OGRIndexError, SRSException
|
||||||
from django.contrib.gis.gdal.geomtype import OGRGeomType
|
from django.contrib.gis.gdal.geomtype import OGRGeomType
|
||||||
|
from django.contrib.gis.gdal.libgdal import GEOJSON, GDAL_VERSION
|
||||||
from django.contrib.gis.gdal.srs import SpatialReference, CoordTransform
|
from django.contrib.gis.gdal.srs import SpatialReference, CoordTransform
|
||||||
|
|
||||||
# Getting the ctypes prototype functions that interface w/the GDAL C library.
|
# Getting the ctypes prototype functions that interface w/the GDAL C library.
|
||||||
from django.contrib.gis.gdal.prototypes import geom as capi, srs as srs_api
|
from django.contrib.gis.gdal.prototypes import geom as capi, srs as srs_api
|
||||||
GEOJSON = capi.GEOJSON
|
|
||||||
|
|
||||||
# For recognizing geometry input.
|
# For recognizing geometry input.
|
||||||
from django.contrib.gis.geometry.regex import hex_regex, wkt_regex, json_regex
|
from django.contrib.gis.geometry.regex import hex_regex, wkt_regex, json_regex
|
||||||
|
@ -400,7 +400,8 @@ class OGRGeometry(GDALBase):
|
||||||
# afterwards. This is done because of GDAL bug (in versions prior
|
# afterwards. This is done because of GDAL bug (in versions prior
|
||||||
# to 1.7) that turns geometries 3D after transformation, see:
|
# to 1.7) that turns geometries 3D after transformation, see:
|
||||||
# http://trac.osgeo.org/gdal/changeset/17792
|
# http://trac.osgeo.org/gdal/changeset/17792
|
||||||
orig_dim = self.coord_dim
|
if GDAL_VERSION < (1, 7):
|
||||||
|
orig_dim = self.coord_dim
|
||||||
|
|
||||||
# Depending on the input type, use the appropriate OGR routine
|
# Depending on the input type, use the appropriate OGR routine
|
||||||
# to perform the transformation.
|
# to perform the transformation.
|
||||||
|
@ -412,11 +413,22 @@ class OGRGeometry(GDALBase):
|
||||||
sr = SpatialReference(coord_trans)
|
sr = SpatialReference(coord_trans)
|
||||||
capi.geom_transform_to(self.ptr, sr.ptr)
|
capi.geom_transform_to(self.ptr, sr.ptr)
|
||||||
else:
|
else:
|
||||||
raise TypeError('Transform only accepts CoordTransform, SpatialReference, string, and integer objects.')
|
raise TypeError('Transform only accepts CoordTransform, '
|
||||||
|
'SpatialReference, string, and integer objects.')
|
||||||
|
|
||||||
# Setting with original dimension, see comment above.
|
# Setting with original dimension, see comment above.
|
||||||
if self.coord_dim != orig_dim:
|
if GDAL_VERSION < (1, 7):
|
||||||
self.coord_dim = orig_dim
|
if isinstance(self, GeometryCollection):
|
||||||
|
# With geometry collections have to set dimension on
|
||||||
|
# each internal geometry reference, as the collection
|
||||||
|
# dimension isn't affected.
|
||||||
|
for i in xrange(len(self)):
|
||||||
|
internal_ptr = capi.get_geom_ref(self.ptr, i)
|
||||||
|
if orig_dim != capi.get_coord_dim(internal_ptr):
|
||||||
|
capi.set_coord_dim(internal_ptr, orig_dim)
|
||||||
|
else:
|
||||||
|
if self.coord_dim != orig_dim:
|
||||||
|
self.coord_dim = orig_dim
|
||||||
|
|
||||||
def transform_to(self, srs):
|
def transform_to(self, srs):
|
||||||
"For backwards-compatibility."
|
"For backwards-compatibility."
|
||||||
|
|
|
@ -456,6 +456,30 @@ class OGRGeomTest(unittest.TestCase):
|
||||||
self.assertEqual(4326, g2.srs.srid)
|
self.assertEqual(4326, g2.srs.srid)
|
||||||
self.assertEqual(g1.srs.wkt, g2.srs.wkt)
|
self.assertEqual(g1.srs.wkt, g2.srs.wkt)
|
||||||
|
|
||||||
|
def test18_ogrgeometry_transform_workaround(self):
|
||||||
|
"Testing coordinate dimensions on geometries after transformation."
|
||||||
|
# A bug in GDAL versions prior to 1.7 changes the coordinate
|
||||||
|
# dimension of a geometry after it has been transformed.
|
||||||
|
# This test ensures that the bug workarounds employed within
|
||||||
|
# `OGRGeometry.transform` indeed work.
|
||||||
|
wkt_2d = "MULTILINESTRING ((0 0,1 1,2 2))"
|
||||||
|
wkt_3d = "MULTILINESTRING ((0 0 0,1 1 1,2 2 2))"
|
||||||
|
srid = 4326
|
||||||
|
|
||||||
|
# For both the 2D and 3D MultiLineString, ensure _both_ the dimension
|
||||||
|
# of the collection and the component LineString have the expected
|
||||||
|
# coordinate dimension after transform.
|
||||||
|
geom = OGRGeometry(wkt_2d, srid)
|
||||||
|
geom.transform(srid)
|
||||||
|
self.assertEqual(2, geom.coord_dim)
|
||||||
|
self.assertEqual(2, geom[0].coord_dim)
|
||||||
|
self.assertEqual(wkt_2d, geom.wkt)
|
||||||
|
|
||||||
|
geom = OGRGeometry(wkt_3d, srid)
|
||||||
|
geom.transform(srid)
|
||||||
|
self.assertEqual(3, geom.coord_dim)
|
||||||
|
self.assertEqual(3, geom[0].coord_dim)
|
||||||
|
self.assertEqual(wkt_3d, geom.wkt)
|
||||||
|
|
||||||
def suite():
|
def suite():
|
||||||
s = unittest.TestSuite()
|
s = unittest.TestSuite()
|
||||||
|
|
Loading…
Reference in New Issue