diff --git a/timeseriesviewer/utils.py b/timeseriesviewer/utils.py index 451f893248a30fcb15d637d610736fb78d46a849..e5813802174698715418d1542718607605a3a073 100644 --- a/timeseriesviewer/utils.py +++ b/timeseriesviewer/utils.py @@ -58,6 +58,8 @@ class SpatialPoint(QgsPoint): return SpatialPoint(crs, spatialExtent.center()) def __init__(self, crs, *args): + if not isinstance(crs, QgsCoordinateReferenceSystem): + crs = QgsCoordinateReferenceSystem(crs) assert isinstance(crs, QgsCoordinateReferenceSystem) super(SpatialPoint, self).__init__(*args) self.mCrs = crs @@ -69,6 +71,29 @@ class SpatialPoint(QgsPoint): def crs(self): return self.mCrs + def toPixelPosition(self, rasterDataSource, allowOutOfRaster=False): + """ + Returns the pixel position of this SpatialPoint within the rasterDataSource + :param rasterDataSource: gdal.Dataset + :param allowOutOfRaster: set True to return out-of-raster pixel positions, e.g. QPoint(-1,0) + :return: the pixel position as QPoint + """ + ds = gdalDataset(rasterDataSource) + ns, nl = ds.RasterXSize, ds.RasterYSize + gt = ds.GetGeoTransform() + + pt = self.toCrs(ds.GetProjection()) + if pt is None: + return None + + px = geo2px(pt, gt) + if not allowOutOfRaster: + if px.x() < 0 or px.x() >= ns: + return None + if px.y() < 0 or px.y() >= nl: + return None + return px + def toCrs(self, crs): assert isinstance(crs, QgsCoordinateReferenceSystem) pt = QgsPoint(self) @@ -78,8 +103,21 @@ class SpatialPoint(QgsPoint): return SpatialPoint(crs, pt) if pt else None + def __reduce_ex__(self, protocol): + return self.__class__, (self.crs().toWkt(), self.x(), self.y()), {} + + def __eq__(self, other): + if not isinstance(other, SpatialPoint): + return False + return self.x() == other.x() and \ + self.y() == other.y() and \ + self.crs() == other.crs() + def __copy__(self): - return SpatialExtent(self.crs(), QgsRectangle(self)) + return SpatialPoint(self.crs(), self.x(), self.y()) + + def __str__(self): + return self.__repr__() def __repr__(self): return '{} {} {}'.format(self.x(), self.y(), self.crs().authid()) @@ -124,6 +162,41 @@ def saveTransform(geom, crs1, crs2): crs1.description(), crs2.description(), str(geom))) return result + +def gdalDataset(pathOrDataset, eAccess=gdal.GA_ReadOnly): + """ + + :param pathOrDataset: path or gdal.Dataset + :return: gdal.Dataset + """ + if not isinstance(pathOrDataset, gdal.Dataset): + pathOrDataset = gdal.Open(pathOrDataset, eAccess) + assert isinstance(pathOrDataset, gdal.Dataset) + return pathOrDataset + + +def geo2pxF(geo, gt): + """ + Returns the pixel position related to a Geo-Coordinate in floating point precision. + :param geo: Geo-Coordinate as QgsPoint + :param gt: GDAL Geo-Transformation tuple, as described in http://www.gdal.org/gdal_datamodel.html + :return: pixel position as QPointF + """ + assert isinstance(geo, QgsPoint) + # see http://www.gdal.org/gdal_datamodel.html + px = (geo.x() - gt[0]) / gt[1] # x pixel + py = (geo.y() - gt[3]) / gt[5] # y pixel + return QPointF(px,py) + +geo2px = lambda geo, gt : geo2pxF(geo,gt).toPoint() + +def px2geo(px, gt): + #see http://www.gdal.org/gdal_datamodel.html + gx = gt[0] + px.x()*gt[1]+px.y()*gt[2] + gy = gt[3] + px.x()*gt[4]+px.y()*gt[5] + return QgsPoint(gx,gy) + + class SpatialExtent(QgsRectangle): """ Object to keep QgsRectangle and QgsCoordinateReferenceSystem together @@ -146,6 +219,29 @@ class SpatialExtent(QgsRectangle): return SpatialExtent(crs, ext) + @staticmethod + def fromRasterSource(pathSrc): + ds = gdalDataset(pathSrc) + assert isinstance(ds, gdal.Dataset) + ns, nl = ds.RasterXSize, ds.RasterYSize + gt = ds.GetGeoTransform() + crs = QgsCoordinateReferenceSystem(ds.GetProjection()) + + xValues = [] + yValues = [] + for x in [0, ns]: + for y in [0, nl]: + px = px2geo(QPoint(x,y), gt) + xValues.append(px.x()) + yValues.append(px.y()) + + return SpatialExtent(crs, min(xValues), min(yValues), + max(xValues), max(yValues)) + + + + + @staticmethod def fromLayer(mapLayer): assert isinstance(mapLayer, QgsMapLayer) @@ -154,6 +250,8 @@ class SpatialExtent(QgsRectangle): return SpatialExtent(crs, extent) def __init__(self, crs, *args): + if not isinstance(crs, QgsCoordinateReferenceSystem): + crs = QgsCoordinateReferenceSystem(crs) assert isinstance(crs, QgsCoordinateReferenceSystem) super(SpatialExtent, self).__init__(*args) self.mCrs = crs @@ -172,8 +270,8 @@ class SpatialExtent(QgsRectangle): box = saveTransform(box, self.mCrs, crs) return SpatialExtent(crs, box) if box else None - def __copy__(self): - return SpatialExtent(self.crs(), QgsRectangle(self)) + def spatialCenter(self): + return SpatialPoint(self.crs(), self.center()) def combineExtentWith(self, *args): if args is None: @@ -204,15 +302,6 @@ class SpatialExtent(QgsRectangle): if other is None: return 1 s = "" - def __eq__(self, other): - return self.toString() == other.toString() - - def __sub__(self, other): - raise NotImplementedError() - - def __mul__(self, other): - raise NotImplementedError() - def upperRightPt(self): return QgsPoint(*self.upperRight()) @@ -239,6 +328,26 @@ class SpatialExtent(QgsRectangle): return self.xMinimum(), self.yMinimum() + def __eq__(self, other): + return self.toString() == other.toString() + + def __sub__(self, other): + raise NotImplementedError() + + def __mul__(self, other): + raise NotImplementedError() + + def __copy__(self): + return SpatialExtent(self.crs(), QgsRectangle(self)) + + def __reduce_ex__(self, protocol): + return self.__class__, (self.crs().toWkt(), + self.xMinimum(), self.yMinimum(), + self.xMaximum(), self.yMaximum() + ), {} + def __str__(self): + return self.__repr__() + def __repr__(self): return '{} {} {}'.format(self.upperLeft(), self.lowerRight(), self.crs().authid())