-
Notifications
You must be signed in to change notification settings - Fork 54
/
Copy pathbase.py
160 lines (134 loc) · 5.86 KB
/
base.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
"""Class for representing 1- or 2-dimensional regions of interest (RoIs)."""
from __future__ import annotations
from collections.abc import Sequence
from typing import Literal, TypeAlias
import shapely
from shapely.coords import CoordinateSequence
from movement.utils.logging import log_error
LineLike: TypeAlias = shapely.LinearRing | shapely.LineString
PointLike: TypeAlias = tuple[float, float]
PointLikeList: TypeAlias = Sequence[PointLike]
RegionLike: TypeAlias = shapely.Polygon
SupportedGeometry: TypeAlias = LineLike | RegionLike
class BaseRegionOfInterest:
"""Base class for representing regions of interest (RoIs).
Regions of interest can be either 1 or 2 dimensional, and are represented
by appropriate ``shapely.Geometry`` objects depending on which. Note that
there are a number of discussions concerning subclassing ``shapely``
objects;
- https://github.com/shapely/shapely/issues/1233.
- https://stackoverflow.com/questions/10788976/how-do-i-properly-inherit-from-a-superclass-that-has-a-new-method
To avoid the complexities of subclassing ourselves, we simply elect to wrap
the appropriate ``shapely`` object in the ``_shapely_geometry`` attribute,
accessible via the property ``region``. This also has the benefit of
allowing us to 'forbid' certain operations (that ``shapely`` would
otherwise interpret in a set-theoretic sense, giving confusing answers to
users).
This class is not designed to be instantiated directly. It can be
instantiated, however its primary purpose is to reduce code duplication.
"""
__default_name: str = "Un-named region"
_name: str | None
_shapely_geometry: SupportedGeometry
@property
def coords(self) -> CoordinateSequence:
"""Coordinates of the points that define the region.
These are the points passed to the constructor argument ``points``.
Note that for Polygonal regions, these are the coordinates of the
exterior boundary, interior boundaries must be accessed via
``self.region.interior.coords``.
"""
return (
self.region.coords
if self.dimensions < 2
else self.region.exterior.coords
)
@property
def dimensions(self) -> int:
"""Dimensionality of the region."""
return shapely.get_dimensions(self.region)
@property
def is_closed(self) -> bool:
"""Return True if the region is closed.
A closed region is either:
- A polygon (2D RoI).
- A 1D LoI whose final point connects back to its first.
"""
return self.dimensions > 1 or (
self.dimensions == 1
and self.region.coords[0] == self.region.coords[-1]
)
@property
def name(self) -> str:
"""Name of the instance."""
return self._name if self._name else self.__default_name
@property
def region(self) -> SupportedGeometry:
"""``shapely.Geometry`` representation of the region."""
return self._shapely_geometry
def __init__(
self,
points: PointLikeList,
dimensions: Literal[1, 2] = 2,
closed: bool = False,
holes: Sequence[PointLikeList] | None = None,
name: str | None = None,
) -> None:
"""Initialise a region of interest.
Parameters
----------
points : Sequence of (x, y) values
Sequence of (x, y) coordinate pairs that will form the region.
dimensions : Literal[1, 2], default 2
The dimensionality of the region to construct.
'1' creates a sequence of joined line segments,
'2' creates a polygon whose boundary is defined by ``points``.
closed : bool, default False
Whether the line to be created should be closed. That is, whether
the final point should also link to the first point.
Ignored if ``dimensions`` is 2.
holes : sequence of sequences of (x, y) pairs, default None
A sequence of items, where each item will be interpreted like
``points``. These items will be used to construct internal holes
within the region. See the ``holes`` argument to
``shapely.Polygon`` for details. Ignored if ``dimensions`` is 1.
name : str, default None
Human-readable name to assign to the given region, for
user-friendliness. Default name given is 'Un-named region' if no
explicit name is provided.
"""
self._name = name
if len(points) < dimensions + 1:
raise log_error(
ValueError,
f"Need at least {dimensions + 1} points to define a "
f"{dimensions}D region (got {len(points)}).",
)
elif dimensions < 1 or dimensions > 2:
raise log_error(
ValueError,
"Only regions of interest of dimension 1 or 2 are supported "
f"(requested {dimensions})",
)
elif dimensions == 1 and len(points) < 3 and closed:
raise log_error(
ValueError,
"Cannot create a loop from a single line segment.",
)
if dimensions == 2:
self._shapely_geometry = shapely.Polygon(shell=points, holes=holes)
else:
self._shapely_geometry = (
shapely.LinearRing(coordinates=points)
if closed
else shapely.LineString(coordinates=points)
)
def __repr__(self) -> str: # noqa: D105
return str(self)
def __str__(self) -> str: # noqa: D105
display_type = "-gon" if self.dimensions > 1 else " line segment(s)"
n_points = len(self.coords) - 1
return (
f"{self.__class__.__name__} {self.name} "
f"({n_points}{display_type})\n"
) + " -> ".join(f"({c[0]}, {c[1]})" for c in self.coords)