Source code for grass.temporal.spatial_extent

"""
Spatial extents classes for map layer and space time datasets

Usage:

.. code-block:: python

    >>> import grass.temporal as tgis
    >>> tgis.init()
    >>> extent = tgis.RasterSpatialExtent(
    ...     ident="raster@PERMANENT",
    ...     north=90,
    ...     south=90,
    ...     east=180,
    ...     west=180,
    ...     top=100,
    ...     bottom=-20,
    ... )
    >>> extent = tgis.Raster3DSpatialExtent(
    ...     ident="raster3d@PERMANENT",
    ...     north=90,
    ...     south=90,
    ...     east=180,
    ...     west=180,
    ...     top=100,
    ...     bottom=-20,
    ... )
    >>> extent = tgis.VectorSpatialExtent(
    ...     ident="vector@PERMANENT",
    ...     north=90,
    ...     south=90,
    ...     east=180,
    ...     west=180,
    ...     top=100,
    ...     bottom=-20,
    ... )
    >>> extent = tgis.STRDSSpatialExtent(
    ...     ident="strds@PERMANENT",
    ...     north=90,
    ...     south=90,
    ...     east=180,
    ...     west=180,
    ...     top=100,
    ...     bottom=-20,
    ... )
    >>> extent = tgis.STR3DSSpatialExtent(
    ...     ident="str3ds@PERMANENT",
    ...     north=90,
    ...     south=90,
    ...     east=180,
    ...     west=180,
    ...     top=100,
    ...     bottom=-20,
    ... )
    >>> extent = tgis.STVDSSpatialExtent(
    ...     ident="stvds@PERMANENT",
    ...     north=90,
    ...     south=90,
    ...     east=180,
    ...     west=180,
    ...     top=100,
    ...     bottom=-20,
    ... )

(C) 2012-2013 by the GRASS Development Team
This program is free software under the GNU General Public
License (>=v2). Read the file COPYING that comes with GRASS
for details.

:authors: Soeren Gebbert
"""

from typing import Literal

from .base import SQLDatabaseInterface


[docs]class SpatialExtent(SQLDatabaseInterface): """This is the spatial extent base class for all maps and space time datasets This class implements a three dimensional axis aligned bounding box and functions to compute topological relationships Usage: .. code-block:: python >>> init() >>> extent = SpatialExtent( ... table="raster_spatial_extent", ... ident="soil@PERMANENT", ... north=90, ... south=90, ... east=180, ... west=180, ... top=100, ... bottom=-20, ... ) >>> extent.id 'soil@PERMANENT' >>> extent.north 90.0 >>> extent.south 90.0 >>> extent.east 180.0 >>> extent.west 180.0 >>> extent.top 100.0 >>> extent.bottom -20.0 >>> extent.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 90.0 | South:...................... 90.0 | East:.. .................... 180.0 | West:....................... 180.0 | Top:........................ 100.0 | Bottom:..................... -20.0 >>> extent.print_shell_info() north=90.0 south=90.0 east=180.0 west=180.0 top=100.0 bottom=-20.0 """ def __init__( self, table=None, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, proj="XY", ) -> None: SQLDatabaseInterface.__init__(self, table, ident) self.set_id(ident) self.set_spatial_extent_from_values(north, south, east, west, top, bottom) self.set_projection(proj)
[docs] def overlapping_2d(self, extent) -> bool: """Return True if this (A) and the provided spatial extent (B) overlaps in two dimensional space. Code is lend from wind_overlap.c in lib/gis Overlapping includes the spatial relations: - contain - in - cover - covered - equivalent .. code-block:: python >>> A = SpatialExtent(north=80, south=20, east=60, west=10) >>> B = SpatialExtent(north=80, south=20, east=60, west=10) >>> A.overlapping_2d(B) True :param extent: The spatial extent to check overlapping with :return: True or False """ if self.get_projection() != extent.get_projection(): self.msgr.error( _( "Projections are different. Unable to compute " "overlapping_2d for spatial extents" ) ) return False N = extent.get_north() S = extent.get_south() E = extent.get_east() W = extent.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while self.get_west() > E: E += 360.0 W += 360.0 while self.get_east() < W: E -= 360.0 W -= 360.0 return not ( self.get_north() <= S or self.get_south() >= N or self.get_east() <= W or self.get_west() >= E )
[docs] def overlapping(self, extent) -> bool: """Return True if this (A) and the provided spatial extent (B) overlaps in three dimensional space. Overlapping includes the spatial relations: - contain - in - cover - covered - equivalent Usage: .. code-block:: python >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> A.overlapping(B) True :param extent: The spatial extent to check overlapping with :return: True or False """ if not self.overlapping_2d(extent): return False T = extent.get_top() B = extent.get_bottom() return not (self.get_top() <= B or self.get_bottom() >= T)
[docs] def intersect_2d(self, extent): """Return the two dimensional intersection as spatial_extent object or None in case no intersection was found. :param extent: The spatial extent to intersect with :return: The intersection spatial extent """ if not self.overlapping_2d(extent): return None eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 # Compute the extent nN = N nS = S nE = E nW = W if eW > W: nW = eW if eE < E: nE = eE if eN < N: nN = eN if eS > S: nS = eS return SpatialExtent( north=nN, south=nS, east=nE, west=nW, top=0, bottom=0, proj=self.get_projection(), )
[docs] def intersect(self, extent): """Return the three dimensional intersection as spatial_extent object or None in case no intersection was found. Usage: .. code-block:: python >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> C = A.intersect(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 80.0 | South:...................... 20.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=10, bottom=-50, top=50 ... ) >>> C = A.intersect(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 40.0 | South:...................... 30.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=30, bottom=-50, top=50 ... ) >>> C = A.intersect(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 40.0 | South:...................... 30.0 | East:.. .................... 60.0 | West:....................... 30.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=30, bottom=-30, top=50 ... ) >>> C = A.intersect(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 40.0 | South:...................... 30.0 | East:.. .................... 60.0 | West:....................... 30.0 | Top:........................ 50.0 | Bottom:..................... -30.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=30, bottom=-30, top=30 ... ) >>> C = A.intersect(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 40.0 | South:...................... 30.0 | East:.. .................... 60.0 | West:....................... 30.0 | Top:........................ 30.0 | Bottom:..................... -30.0 :param extent: The spatial extent to intersect with :return: The intersection spatial extent """ # noqa: E501 if not self.overlapping(extent): return None new = self.intersect_2d(extent) eT = extent.get_top() eB = extent.get_bottom() T = self.get_top() B = self.get_bottom() nT = T nB = B if eB > B: nB = eB if eT < T: nT = eT new.set_top(nT) new.set_bottom(nB) return new
[docs] def union_2d(self, extent): """Return the two dimensional union as spatial_extent object or None in case the extents does not overlap or meet. :param extent: The spatial extent to create a union with :return: The union spatial extent """ if not self.overlapping_2d(extent) and not self.meet_2d(extent): return None return self.disjoint_union_2d(extent)
[docs] def disjoint_union_2d(self, extent): """Return the two dimensional union as spatial_extent. :param extent: The spatial extent to create a union with :return: The union spatial extent """ eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 # Compute the extent nN = N nS = S nE = E nW = W if eW < W: nW = eW if eE > E: nE = eE if eN > N: nN = eN if eS < S: nS = eS return SpatialExtent( north=nN, south=nS, east=nE, west=nW, top=0, bottom=0, proj=self.get_projection(), )
[docs] def union(self, extent): """Return the three dimensional union as spatial_extent object or None in case the extents does not overlap or meet. :param extent: The spatial extent to create a union with :return: The union spatial extent """ if not self.overlapping(extent) and not self.meet(extent): return None return self.disjoint_union(extent)
[docs] def disjoint_union(self, extent): """Return the three dimensional union as spatial_extent . Usage: .. code-block:: python >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> C = A.disjoint_union(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 80.0 | South:...................... 20.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=10, bottom=-50, top=50 ... ) >>> C = A.disjoint_union(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 80.0 | South:...................... 20.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=30, bottom=-50, top=50 ... ) >>> C = A.disjoint_union(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 80.0 | South:...................... 20.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=30, bottom=-30, top=50 ... ) >>> C = A.disjoint_union(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 80.0 | South:...................... 20.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> B = SpatialExtent( ... north=40, south=30, east=60, west=30, bottom=-30, top=30 ... ) >>> C = A.disjoint_union(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 80.0 | South:...................... 20.0 | East:.. .................... 60.0 | West:....................... 10.0 | Top:........................ 50.0 | Bottom:..................... -50.0 >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=90, south=80, east=70, west=20, bottom=-30, top=60 ... ) >>> C = A.disjoint_union(B) >>> C.print_info() +-------------------- Spatial extent ----------------------------------------+ | North:...................... 90.0 | South:...................... 20.0 | East:.. .................... 70.0 | West:....................... 10.0 | Top:........................ 60.0 | Bottom:..................... -50.0 :param extent: The spatial extent to create a disjoint union with :return: The union spatial extent """ # noqa: E501 new = self.disjoint_union_2d(extent) eT = extent.get_top() eB = extent.get_bottom() T = self.get_top() B = self.get_bottom() nT = T nB = B if eB < B: nB = eB if eT > T: nT = eT new.set_top(nT) new.set_bottom(nB) return new
[docs] def is_in_2d(self, extent) -> bool: """Return True if this extent (A) is located in the provided spatial extent (B) in two dimensions. :: _____ |A _ | | |_| | |_____|B :param extent: The spatial extent :return: True or False """ if self.get_projection() != extent.get_projection(): self.msgr.error( _( "Projections are different. Unable to compute " "is_in_2d for spatial extents" ) ) return False eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 return not (eW >= W or eE <= E or eN <= N or eS >= S)
[docs] def is_in(self, extent) -> bool: """Return True if this extent (A) is located in the provided spatial extent (B) in three dimensions. Usage: .. code-block:: python >>> A = SpatialExtent( ... north=79, south=21, east=59, west=11, bottom=-49, top=49 ... ) >>> B = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> A.is_in(B) True >>> B.is_in(A) False :param extent: The spatial extent :return: True or False """ if not self.is_in_2d(extent): return False eT = extent.get_top() eB = extent.get_bottom() T = self.get_top() B = self.get_bottom() return not (eB >= B or eT <= T)
[docs] def contain_2d(self, extent): """Return True if this extent (A) contains the provided spatial extent (B) in two dimensions. Usage: .. code-block:: python >>> A = SpatialExtent(north=80, south=20, east=60, west=10) >>> B = SpatialExtent(north=79, south=21, east=59, west=11) >>> A.contain_2d(B) True >>> B.contain_2d(A) False :param extent: The spatial extent :return: True or False """ return extent.is_in_2d(self)
[docs] def contain(self, extent): """Return True if this extent (A) contains the provided spatial extent (B) in three dimensions. Usage: .. code-block:: python >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=79, south=21, east=59, west=11, bottom=-49, top=49 ... ) >>> A.contain(B) True >>> B.contain(A) False :param extent: The spatial extent :return: True or False """ return extent.is_in(self)
[docs] def equivalent_2d(self, extent) -> bool: """Return True if this extent (A) is equal to the provided spatial extent (B) in two dimensions. Usage: .. code-block:: python >>> A = SpatialExtent(north=80, south=20, east=60, west=10) >>> B = SpatialExtent(north=80, south=20, east=60, west=10) >>> A.equivalent_2d(B) True >>> B.equivalent_2d(A) True :param extent: The spatial extent :return: True or False """ if self.get_projection() != extent.get_projection(): self.msgr.error( _( "Projections are different. Unable to compute " "equivalent_2d for spatial extents" ) ) return False eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 return not (eW != W or eE != E or eN != N or eS != S)
[docs] def equivalent(self, extent) -> bool: """Return True if this extent (A) is equal to the provided spatial extent (B) in three dimensions. Usage: .. code-block:: python >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> A.equivalent(B) True >>> B.equivalent(A) True :param extent: The spatial extent :return: True or False """ if not self.equivalent_2d(extent): return False eT = extent.get_top() eB = extent.get_bottom() T = self.get_top() B = self.get_bottom() return not (eB != B or eT != T)
[docs] def cover_2d(self, extent) -> bool: """Return True if this extent (A) covers the provided spatial extent (B) in two dimensions. :: _____ _____ _____ _____ |A __| |__ A| |A | B| |B | A| | |B | | B| | | |__| |__| | |__|__| |__|__| |_____| |_____| _____ _____ _____ _____ |A|B| | |A __| |A _ | |__ A| | |_| | | |__|B | |B| | B|__| | |_____| |_____| |_|_|_| |_____| _____ _____ _____ _____ |A|B | |_____|A |A|B|A| |_____|A | | | |B | | | | | |_____|B |_|___| |_____| |_|_|_| |_____|A The following cases are excluded: - contain - in - equivalent :param extent: The spatial extent :return: True or False """ if self.get_projection() != extent.get_projection(): self.msgr.error( _( "Projections are different. Unable to compute" " cover_2d for spatial extents" ) ) return False # Exclude equivalent_2d if self.equivalent_2d(extent): return False eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 # Edges of extent located outside of self are not allowed if eW >= E: return False if eE <= W: return False if eS >= N: return False if eN <= S: return False # First we check that at least one edge of extent meets an edge of self if eW != W and eE != E and eN != N and eS != S: return False # We check that at least one edge of extent is located in self edge_count = 0 if W < eW < E: edge_count += 1 if W < eE < E: edge_count += 1 if S < eN < N: edge_count += 1 if S < eS < N: edge_count += 1 return edge_count != 0
[docs] def cover(self, extent) -> bool: """Return True if this extent covers the provided spatial extent in three dimensions. The following cases are excluded: - contain - in - equivalent :param extent: The spatial extent :return: True or False """ if self.get_projection() != extent.get_projection(): self.msgr.error( _( "Projections are different. Unable to compute " "cover for spatial extents" ) ) return False # Exclude equivalent_2d if self.equivalent_2d(extent): return False eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() eT = extent.get_top() eB = extent.get_bottom() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() T = self.get_top() B = self.get_bottom() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 # Edges of extent located outside of self are not allowed if (eW >= E) or (eE <= W) or (eS >= N) or (eN <= S) or (eB >= T) or (eT <= B): return False # First we check that at least one edge of extent meets an edge of self if eW != W and eE != E and eN != N and eS != S and eB != B and eT != T: return False # We check that at least one edge of extent is located in self edge_count = 0 if W < eW < E: edge_count += 1 if W < eE < E: edge_count += 1 if S < eN < N: edge_count += 1 if S < eS < N: edge_count += 1 if S < eN < N: edge_count += 1 if S < eS < N: edge_count += 1 if B < eT < T: edge_count += 1 if B < eB < T: edge_count += 1 return edge_count != 0
[docs] def covered_2d(self, extent): """Return True if this extent is covered by the provided spatial extent in two dimensions. The following cases are excluded: - contain - in - equivalent :param extent: The spatial extent :return: True or False """ return extent.cover_2d(self)
[docs] def covered(self, extent): """Return True if this extent is covered by the provided spatial extent in three dimensions. The following cases are excluded: - contain - in - equivalent :param extent: The spatial extent :return: True or False """ return extent.cover(self)
[docs] def overlap_2d(self, extent) -> bool: """Return True if this extent (A) overlaps with the provided spatial extent (B) in two dimensions. Code is lend from wind_overlap.c in lib/gis :: _____ |A __|__ | | | B| |__|__| | |_____| The following cases are excluded: - contain - in - cover - covered - equivalent :param extent: The spatial extent :return: True or False """ if self.contain_2d(extent): return False if self.is_in_2d(extent): return False if self.cover_2d(extent): return False if self.covered_2d(extent): return False if self.equivalent_2d(extent): return False N = extent.get_north() S = extent.get_south() E = extent.get_east() W = extent.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while self.get_west() > E: E += 360.0 W += 360.0 while self.get_east() < W: E -= 360.0 W -= 360.0 return not ( self.get_north() <= S or self.get_south() >= N or self.get_east() <= W or self.get_west() >= E )
[docs] def overlap(self, extent) -> bool: """Return True if this extent overlaps with the provided spatial extent in three dimensions. The following cases are excluded: - contain - in - cover - covered - equivalent :param extent: The spatial extent :return: True or False """ if self.is_in(extent): return False if self.contain(extent): return False if self.cover(extent): return False if self.covered(extent): return False if self.equivalent(extent): return False N = extent.get_north() S = extent.get_south() E = extent.get_east() W = extent.get_west() T = extent.get_top() B = extent.get_bottom() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while self.get_west() > E: E += 360.0 W += 360.0 while self.get_east() < W: E -= 360.0 W -= 360.0 return not ( self.get_north() <= S or self.get_south() >= N or self.get_east() <= W or self.get_west() >= E or self.get_top() <= B or self.get_bottom() >= T )
[docs] def meet_2d(self, extent) -> bool: """Return True if this extent (A) meets with the provided spatial extent (B) in two dimensions. :: _____ _____ | A | B | |_____| | |_____| _____ _____ | B | A | | | | |_____|_____| ___ | A | | | |___| | B | | | |_____| _____ | B | | | |_____|_ | A | | | |_____| :param extent: The spatial extent :return: True or False """ eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 edge = None edge_count = 0 if eW == E: edge = "E" edge_count += 1 if eE == W: edge = "W" edge_count += 1 if eS == N: edge = "N" edge_count += 1 if eN == S: edge = "S" edge_count += 1 # Meet a a single edge only if edge_count != 1: return False # Check boundaries of the faces if edge in {"E", "W"}: if eS > N or eN < S: return False if edge in {"N", "S"}: if eW > E or eE < W: return False return True
[docs] def meet(self, extent) -> bool: """Return True if this extent meets with the provided spatial extent in three dimensions. :param extent: The spatial extent :return: True or False """ eN = extent.get_north() eS = extent.get_south() eE = extent.get_east() eW = extent.get_west() eT = extent.get_top() eB = extent.get_bottom() N = self.get_north() S = self.get_south() E = self.get_east() W = self.get_west() T = self.get_top() B = self.get_bottom() # Adjust the east and west in case of LL projection if self.get_projection() == "LL": while eE < W: eE += 360.0 eW += 360.0 while eW > E: eE -= 360.0 eW -= 360.0 edge = None edge_count = 0 if eW == E: edge = "E" edge_count += 1 if eE == W: edge = "W" edge_count += 1 if eS == N: edge = "N" edge_count += 1 if eN == S: edge = "S" edge_count += 1 if eB == T: edge = "T" edge_count += 1 if eT == B: edge = "B" edge_count += 1 # Meet a single edge only if edge_count != 1: return False # Check boundaries of the faces if edge in {"E", "W"}: if eS > N or eN < S: return False if eB > T or eT < B: return False if edge in {"N", "S"}: if eW > E or eE < W: return False if eB > T or eT < B: return False if edge in {"T", "B"}: if eW > E or eE < W: return False if eS > N or eN < S: return False return True
[docs] def disjoint_2d(self, extent) -> bool: """Return True if this extent (A) is disjoint with the provided spatial extent (B) in three dimensions. :: _____ | A | |_____| _______ | B | |_______| :param extent: The spatial extent :return: True or False """ return not ( self.is_in_2d(extent) or self.contain_2d(extent) or self.cover_2d(extent) or self.covered_2d(extent) or self.equivalent_2d(extent) or self.overlapping_2d(extent) or self.meet_2d(extent) )
[docs] def disjoint(self, extent) -> bool: """Return True if this extent is disjoint with the provided spatial extent in three dimensions. :param extent: The spatial extent :return: True or False """ return not ( self.is_in(extent) or self.contain(extent) or self.cover(extent) or self.covered(extent) or self.equivalent(extent) or self.overlapping(extent) or self.meet(extent) )
[docs] def spatial_relation_2d(self, extent) -> Literal[ "equivalent", "contain", "in", "cover", "covered", "overlap", "meet", "disjoint", "unknown", ]: """Returns the two dimensional spatial relation between this extent and the provided spatial extent in two dimensions. Spatial relations are: - disjoint - meet - overlap - cover - covered - in - contain - equivalent Usage: see self.spatial_relation() """ if self.equivalent_2d(extent): return "equivalent" if self.contain_2d(extent): return "contain" if self.is_in_2d(extent): return "in" if self.cover_2d(extent): return "cover" if self.covered_2d(extent): return "covered" if self.overlap_2d(extent): return "overlap" if self.meet_2d(extent): return "meet" if self.disjoint_2d(extent): return "disjoint" return "unknown"
[docs] def spatial_relation(self, extent) -> Literal[ "equivalent", "contain", "in", "cover", "covered", "overlap", "meet", "disjoint", "unknown", ]: """Returns the two dimensional spatial relation between this extent and the provided spatial extent in three dimensions. Spatial relations are: - disjoint - meet - overlap - cover - covered - in - contain - equivalent Usage: .. code-block:: python >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation(B) 'equivalent' >>> B.spatial_relation(A) 'equivalent' >>> B = SpatialExtent( ... north=70, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'cover' >>> A.spatial_relation(B) 'cover' >>> B = SpatialExtent( ... north=70, south=30, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'cover' >>> A.spatial_relation(B) 'cover' >>> B.spatial_relation_2d(A) 'covered' >>> B.spatial_relation(A) 'covered' >>> B = SpatialExtent( ... north=70, south=30, east=50, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'cover' >>> B.spatial_relation_2d(A) 'covered' >>> A.spatial_relation(B) 'cover' >>> B = SpatialExtent( ... north=70, south=30, east=50, west=20, bottom=-50, top=50 ... ) >>> B.spatial_relation(A) 'covered' >>> B = SpatialExtent( ... north=70, south=30, east=50, west=20, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'contain' >>> A.spatial_relation(B) 'cover' >>> B = SpatialExtent( ... north=70, south=30, east=50, west=20, bottom=-40, top=50 ... ) >>> A.spatial_relation(B) 'cover' >>> B = SpatialExtent( ... north=70, south=30, east=50, west=20, bottom=-40, top=40 ... ) >>> A.spatial_relation(B) 'contain' >>> B.spatial_relation(A) 'in' >>> B = SpatialExtent( ... north=90, south=30, east=50, west=20, bottom=-40, top=40 ... ) >>> A.spatial_relation_2d(B) 'overlap' >>> A.spatial_relation(B) 'overlap' >>> B = SpatialExtent( ... north=90, south=5, east=70, west=5, bottom=-40, top=40 ... ) >>> A.spatial_relation_2d(B) 'in' >>> A.spatial_relation(B) 'overlap' >>> B = SpatialExtent( ... north=90, south=5, east=70, west=5, bottom=-40, top=60 ... ) >>> A.spatial_relation(B) 'overlap' >>> B = SpatialExtent( ... north=90, south=5, east=70, west=5, bottom=-60, top=60 ... ) >>> A.spatial_relation(B) 'in' >>> A = SpatialExtent( ... north=80, south=60, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=60, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'meet' >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=60, south=40, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=60, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'meet' >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=40, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'meet' >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=90, south=30, east=60, west=40, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'meet' >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=70, south=50, east=60, west=40, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'meet' >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=60, south=20, east=60, west=40, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'meet' >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=40, south=20, east=60, west=40, bottom=-50, top=50 ... ) >>> A.spatial_relation_2d(B) 'disjoint' >>> A.spatial_relation(B) 'disjoint' >>> A = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=60, south=20, east=60, west=40, bottom=-60, top=60 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=40, west=20, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=90, south=30, east=60, west=40, bottom=-40, top=40 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=0, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=-50, top=0 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=0, top=50 ... ) >>> B = SpatialExtent( ... north=80, south=50, east=60, west=30, bottom=-50, top=0 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=0, top=50 ... ) >>> B = SpatialExtent( ... north=70, south=50, east=50, west=30, bottom=-50, top=0 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=0, top=50 ... ) >>> B = SpatialExtent( ... north=90, south=30, east=70, west=10, bottom=-50, top=0 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=0, top=50 ... ) >>> B = SpatialExtent( ... north=70, south=30, east=50, west=10, bottom=-50, top=0 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=-50, top=0 ... ) >>> B = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=0, top=50 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=-50, top=0 ... ) >>> B = SpatialExtent( ... north=80, south=50, east=60, west=30, bottom=0, top=50 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=-50, top=0 ... ) >>> B = SpatialExtent( ... north=70, south=50, east=50, west=30, bottom=0, top=50 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=-50, top=0 ... ) >>> B = SpatialExtent( ... north=90, south=30, east=70, west=10, bottom=0, top=50 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=40, east=60, west=20, bottom=-50, top=0 ... ) >>> B = SpatialExtent( ... north=70, south=30, east=50, west=10, bottom=0, top=50 ... ) >>> A.spatial_relation(B) 'meet' >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=90, south=81, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation(B) 'disjoint' >>> A = SpatialExtent( ... north=80, south=20, east=60, west=10, bottom=-50, top=50 ... ) >>> B = SpatialExtent( ... north=90, south=80, east=60, west=10, bottom=-50, top=50 ... ) >>> A.spatial_relation(B) 'meet' """ if self.equivalent(extent): return "equivalent" if self.contain(extent): return "contain" if self.is_in(extent): return "in" if self.cover(extent): return "cover" if self.covered(extent): return "covered" if self.overlap(extent): return "overlap" if self.meet(extent): return "meet" if self.disjoint(extent): return "disjoint" return "unknown"
[docs] def set_spatial_extent_from_values( self, north, south, east, west, top, bottom ) -> None: """Set the three dimensional spatial extent :param north: The northern edge :param south: The southern edge :param east: The eastern edge :param west: The western edge :param top: The top edge :param bottom: The bottom edge """ self.set_north(north) self.set_south(south) self.set_east(east) self.set_west(west) self.set_top(top) self.set_bottom(bottom)
[docs] def set_spatial_extent(self, spatial_extent) -> None: """Set the three dimensional spatial extent :param spatial_extent: An object of type SpatialExtent or its subclasses """ self.set_north(spatial_extent.get_north()) self.set_south(spatial_extent.get_south()) self.set_east(spatial_extent.get_east()) self.set_west(spatial_extent.get_west()) self.set_top(spatial_extent.get_top()) self.set_bottom(spatial_extent.get_bottom())
[docs] def set_projection(self, proj) -> None: """Set the projection of the spatial extent it should be XY or LL. As default the projection is XY """ if proj is None or (proj not in {"XY", "LL"}): self.D["proj"] = "XY" else: self.D["proj"] = proj
[docs] def set_spatial_extent_from_values_2d(self, north, south, east, west) -> None: """Set the two dimensional spatial extent from values :param north: The northern edge :param south: The southern edge :param east: The eastern edge :param west: The western edge """ self.set_north(north) self.set_south(south) self.set_east(east) self.set_west(west)
[docs] def set_spatial_extent_2d(self, spatial_extent) -> None: """Set the three dimensional spatial extent :param spatial_extent: An object of type SpatialExtent or its subclasses """ self.set_north(spatial_extent.north) self.set_south(spatial_extent.south) self.set_east(spatial_extent.east) self.set_west(spatial_extent.west)
[docs] def set_id(self, ident) -> None: """Convenient method to set the unique identifier (primary key)""" self.ident = ident self.D["id"] = ident
[docs] def set_north(self, north) -> None: """Set the northern edge of the map""" if north is not None: self.D["north"] = float(north) else: self.D["north"] = None
[docs] def set_south(self, south) -> None: """Set the southern edge of the map""" if south is not None: self.D["south"] = float(south) else: self.D["south"] = None
[docs] def set_west(self, west) -> None: """Set the western edge of the map""" if west is not None: self.D["west"] = float(west) else: self.D["west"] = None
[docs] def set_east(self, east) -> None: """Set the eastern edge of the map""" if east is not None: self.D["east"] = float(east) else: self.D["east"] = None
[docs] def set_top(self, top) -> None: """Set the top edge of the map""" if top is not None: self.D["top"] = float(top) else: self.D["top"] = None
[docs] def set_bottom(self, bottom) -> None: """Set the bottom edge of the map""" if bottom is not None: self.D["bottom"] = float(bottom) else: self.D["bottom"] = None
[docs] def get_id(self): """Convenient method to get the unique identifier (primary key) :return: None if not found """ if "id" in self.D: return self.D["id"] return None
[docs] def get_projection(self): """Get the projection of the spatial extent""" return self.D["proj"]
[docs] def get_volume(self): """Compute the volume of the extent, in case z is zero (top == bottom or top - bottom = 1) the area is returned""" if self.get_projection() == "LL": self.msgr.error(_("Volume computation is not supported for LL projections")) area = self.get_area() bbox = self.get_spatial_extent_as_tuple() z = abs(bbox[4] - bbox[5]) if z == 0: z = 1.0 return area * z
[docs] def get_area(self): """Compute the area of the extent, extent in z direction is ignored""" if self.get_projection() == "LL": self.msgr.error(_("Area computation is not supported for LL projections")) bbox = self.get_spatial_extent_as_tuple() y = abs(bbox[0] - bbox[1]) x = abs(bbox[2] - bbox[3]) return x * y
[docs] def get_spatial_extent_as_tuple(self): """Return a tuple (north, south, east, west, top, bottom) of the spatial extent""" return (self.north, self.south, self.east, self.west, self.top, self.bottom)
[docs] def get_spatial_extent_as_tuple_2d(self): """Return a tuple (north, south, east, west,) of the 2d spatial extent""" return (self.north, self.south, self.east, self.west)
[docs] def get_north(self): """Get the northern edge of the map :return: None if not found""" if "north" in self.D: return self.D["north"] return None
[docs] def get_south(self): """Get the southern edge of the map :return: None if not found""" if "south" in self.D: return self.D["south"] return None
[docs] def get_east(self): """Get the eastern edge of the map :return: None if not found""" if "east" in self.D: return self.D["east"] return None
[docs] def get_west(self): """Get the western edge of the map :return: None if not found""" if "west" in self.D: return self.D["west"] return None
[docs] def get_top(self): """Get the top edge of the map :return: None if not found""" if "top" in self.D: return self.D["top"] return None
[docs] def get_bottom(self): """Get the bottom edge of the map :return: None if not found""" if "bottom" in self.D: return self.D["bottom"] return None
id = property(fget=get_id, fset=set_id) north = property(fget=get_north, fset=set_north) south = property(fget=get_south, fset=set_south) east = property(fget=get_east, fset=set_east) west = property(fget=get_west, fset=set_west) top = property(fget=get_top, fset=set_top) bottom = property(fget=get_bottom, fset=set_bottom)
[docs] def print_info(self) -> None: """Print information about this class in human readable style""" # 0123456789012345678901234567890 print( " +-------------------- Spatial extent ----------------------------------------+" # noqa: E501 ) print(" | North:...................... " + str(self.get_north())) print(" | South:...................... " + str(self.get_south())) print(" | East:.. .................... " + str(self.get_east())) print(" | West:....................... " + str(self.get_west())) print(" | Top:........................ " + str(self.get_top())) print(" | Bottom:..................... " + str(self.get_bottom()))
[docs] def print_shell_info(self) -> None: """Print information about this class in shell style""" print("north=" + str(self.get_north())) print("south=" + str(self.get_south())) print("east=" + str(self.get_east())) print("west=" + str(self.get_west())) print("top=" + str(self.get_top())) print("bottom=" + str(self.get_bottom()))
###############################################################################
[docs]class RasterSpatialExtent(SpatialExtent): def __init__( self, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, ) -> None: SpatialExtent.__init__( self, "raster_spatial_extent", ident, north, south, east, west, top, bottom )
[docs]class Raster3DSpatialExtent(SpatialExtent): def __init__( self, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, ) -> None: SpatialExtent.__init__( self, "raster3d_spatial_extent", ident, north, south, east, west, top, bottom, )
[docs]class VectorSpatialExtent(SpatialExtent): def __init__( self, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, ) -> None: SpatialExtent.__init__( self, "vector_spatial_extent", ident, north, south, east, west, top, bottom )
[docs]class STRDSSpatialExtent(SpatialExtent): def __init__( self, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, ) -> None: SpatialExtent.__init__( self, "strds_spatial_extent", ident, north, south, east, west, top, bottom )
[docs]class STR3DSSpatialExtent(SpatialExtent): def __init__( self, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, ) -> None: SpatialExtent.__init__( self, "str3ds_spatial_extent", ident, north, south, east, west, top, bottom )
[docs]class STVDSSpatialExtent(SpatialExtent): def __init__( self, ident=None, north=None, south=None, east=None, west=None, top=None, bottom=None, ) -> None: SpatialExtent.__init__( self, "stvds_spatial_extent", ident, north, south, east, west, top, bottom )
############################################################################### if __name__ == "__main__": import doctest doctest.testmod()