Skip to content

Commit

Permalink
Add transform methods for TPoint and STBoxes
Browse files Browse the repository at this point in the history
  • Loading branch information
Diviloper committed Feb 11, 2024
1 parent 1a6e547 commit b637659
Show file tree
Hide file tree
Showing 2 changed files with 86 additions and 56 deletions.
122 changes: 69 additions & 53 deletions pymeos/pymeos/boxes/stbox.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,10 +41,10 @@ class STBox:
_mobilitydb_name = "stbox"

def _get_box(
self,
other: Union[shp.BaseGeometry, STBox, Temporal, Time],
allow_space_only: bool = True,
allow_time_only: bool = False,
self,
other: Union[shp.BaseGeometry, STBox, Temporal, Time],
allow_space_only: bool = True,
allow_time_only: bool = False,
) -> STBox:
if allow_space_only and isinstance(other, shp.BaseGeometry):
other_box = geo_to_stbox(geo_to_gserialized(other, self.geodetic()))
Expand All @@ -68,31 +68,31 @@ def _get_box(

# ------------------------- Constructors ----------------------------------
def __init__(
self,
string: Optional[str] = None,
*,
xmin: Optional[Union[str, float]] = None,
xmax: Optional[Union[str, float]] = None,
ymin: Optional[Union[str, float]] = None,
ymax: Optional[Union[str, float]] = None,
zmin: Optional[Union[str, float]] = None,
zmax: Optional[Union[str, float]] = None,
tmin: Optional[Union[str, datetime]] = None,
tmax: Optional[Union[str, datetime]] = None,
tmin_inc: bool = True,
tmax_inc: bool = True,
geodetic: bool = False,
srid: Optional[int] = None,
_inner=None,
self,
string: Optional[str] = None,
*,
xmin: Optional[Union[str, float]] = None,
xmax: Optional[Union[str, float]] = None,
ymin: Optional[Union[str, float]] = None,
ymax: Optional[Union[str, float]] = None,
zmin: Optional[Union[str, float]] = None,
zmax: Optional[Union[str, float]] = None,
tmin: Optional[Union[str, datetime]] = None,
tmax: Optional[Union[str, datetime]] = None,
tmin_inc: bool = True,
tmax_inc: bool = True,
geodetic: bool = False,
srid: Optional[int] = None,
_inner=None,
):
assert (_inner is not None) or (string is not None) != (
(
xmin is not None
and xmax is not None
and ymin is not None
and ymax is not None
)
or (tmin is not None and tmax is not None)
(
xmin is not None
and xmax is not None
and ymin is not None
and ymax is not None
)
or (tmin is not None and tmax is not None)
), (
"Either string must be not None or at least a bound pair (xmin/max"
" and ymin/max, or tmin/max) must be not None"
Expand All @@ -106,10 +106,10 @@ def __init__(
tstzspan = None
hast = tmin is not None and tmax is not None
hasx = (
xmin is not None
and xmax is not None
and ymin is not None
and ymax is not None
xmin is not None
and xmax is not None
and ymin is not None
and ymax is not None
)
hasz = zmin is not None and zmax is not None
if hast:
Expand Down Expand Up @@ -224,9 +224,9 @@ def from_time(time: Time) -> STBox:

@staticmethod
def from_geometry_time(
geometry: shp.BaseGeometry,
time: Union[datetime, TsTzSpan],
geodetic: bool = False,
geometry: shp.BaseGeometry,
time: Union[datetime, TsTzSpan],
geodetic: bool = False,
) -> STBox:
"""
Returns a `STBox` from a space and time dimension.
Expand Down Expand Up @@ -272,9 +272,9 @@ def from_tpoint(temporal: TPoint) -> STBox:

@staticmethod
def from_expanding_bounding_box(
value: Union[shp.BaseGeometry, TPoint, STBox],
expansion: float,
geodetic: Optional[bool] = False,
value: Union[shp.BaseGeometry, TPoint, STBox],
expansion: float,
geodetic: Optional[bool] = False,
) -> STBox:
"""
Returns a `STBox` from a `shp.BaseGeometry`, `TPoint` or `STBox` instance,
Expand Down Expand Up @@ -661,7 +661,7 @@ def scale_time(self, duration: timedelta) -> STBox:
return self.shift_scale_time(duration=duration)

def shift_scale_time(
self, shift: Optional[timedelta] = None, duration: Optional[timedelta] = None
self, shift: Optional[timedelta] = None, duration: Optional[timedelta] = None
) -> STBox:
"""
Returns a new `STBox` with the time dimension shifted by `shift` and
Expand All @@ -681,7 +681,7 @@ def shift_scale_time(
:meth:`TsTzSpan.shift_scale`
"""
assert (
shift is not None or duration is not None
shift is not None or duration is not None
), "shift and scale deltas must not be both None"
result = stbox_shift_scale_time(
self._inner,
Expand All @@ -707,6 +707,22 @@ def round(self, max_decimals: Optional[int] = 0) -> STBox:
stbox_round(new_inner, max_decimals)
return STBox(_inner=new_inner)

def transform(self, srid: int) -> STBox:
"""
Returns a new :class:`STBox` transformed to another SRID
Args:
srid: The desired SRID
Returns:
A new :class:`STBox` instance
MEOS Functions:
stbox_transform
"""
result = stbox_transform(self._inner, srid)
return STBox(_inner=result)

# ------------------------- Set Operations --------------------------------
def union(self, other: STBox, strict: Optional[bool] = False) -> STBox:
"""
Expand Down Expand Up @@ -776,7 +792,7 @@ def __mul__(self, other):

# ------------------------- Topological Operations ------------------------
def is_adjacent(
self, other: Union[shp.BaseGeometry, STBox, Temporal, Time]
self, other: Union[shp.BaseGeometry, STBox, Temporal, Time]
) -> bool:
"""
Returns whether ``self`` and `other` are adjacent. Two spatiotemporal
Expand All @@ -799,7 +815,7 @@ def is_adjacent(
)

def is_contained_in(
self, container: Union[shp.BaseGeometry, STBox, Temporal, Time]
self, container: Union[shp.BaseGeometry, STBox, Temporal, Time]
) -> bool:
"""
Returns whether ``self`` is contained in `container`. Note that for
Expand Down Expand Up @@ -1161,7 +1177,7 @@ def is_over_or_after(self, other: Union[Box, Temporal, Time]) -> bool:

# ------------------------- Distance Operations ---------------------------
def nearest_approach_distance(
self, other: Union[shp.BaseGeometry, STBox, TPoint]
self, other: Union[shp.BaseGeometry, STBox, TPoint]
) -> float:
"""
Returns the distance between the nearest points of ``self`` and `other`.
Expand Down Expand Up @@ -1261,11 +1277,11 @@ def quad_split(self) -> Union[List[List[STBox]], List[List[List[STBox]]]]:
]

def tile(
self,
size: Optional[float] = None,
duration: Optional[Union[timedelta, str]] = None,
origin: Optional[shp.BaseGeometry] = None,
start: Union[datetime, str, None] = None,
self,
size: Optional[float] = None,
duration: Optional[Union[timedelta, str]] = None,
origin: Optional[shp.BaseGeometry] = None,
start: Union[datetime, str, None] = None,
) -> List[STBox]:
"""
Returns a list of `STBox` instances representing the tiles of
Expand All @@ -1290,12 +1306,12 @@ def tile(
stbox_tile_list
"""
sz = size or (
max(
self.xmax() - self.xmin(),
self.ymax() - self.ymin(),
(self.zmax() - self.zmin() if self.has_z() else 0),
)
+ 1
max(
self.xmax() - self.xmin(),
self.ymax() - self.ymin(),
(self.zmax() - self.zmin() if self.has_z() else 0),
)
+ 1
)
dt = (
timedelta_to_interval(duration)
Expand Down
20 changes: 17 additions & 3 deletions pymeos/pymeos/main/tpoint.py
Original file line number Diff line number Diff line change
Expand Up @@ -465,9 +465,7 @@ def make_simple(self) -> List[TPoint]:
tpoint_make_simple
"""
result, count = tpoint_make_simple(self._inner)
from ..factory import _TemporalFactory

return [_TemporalFactory.create_temporal(result[i]) for i in range(count)]
return [Temporal._factory(result[i]) for i in range(count)]

def expand(self, other: Union[int, float]) -> STBox:
"""
Expand All @@ -489,6 +487,22 @@ def expand(self, other: Union[int, float]) -> STBox:
result = tpoint_expand_space(self._inner, float(other))
return STBox(_inner=result)

def transform(self: Self, srid: int) -> Self:
"""
Returns a new :class:`TPoint` of the same subclass of ``self`` transformed to another SRID
Args:
srid: The desired SRID
Returns:
A new :class:`TPoint` instance
MEOS Functions:
tpoint_transform
"""
result = tpoint_transform(self._inner, srid)
return Temporal._factory(result)

# ------------------------- Restrictions ----------------------------------
def at(self, other: Union[shpb.BaseGeometry, GeoSet, STBox, Time]) -> TG:
"""
Expand Down

0 comments on commit b637659

Please sign in to comment.