-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathobjects.py
323 lines (253 loc) · 9.81 KB
/
objects.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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
from abc import ABC, abstractmethod
from copy import deepcopy
from dataclasses import dataclass
from typing import TYPE_CHECKING, Callable, Optional, Type
from python_ggplot.core.objects import AxisKind, GGException, Scale, UnitType
if TYPE_CHECKING:
from python_ggplot.core.coord.objects import OperatorType
from python_ggplot.graphics.views import ViewPort
def unity_type_to_quantity_cls(kind: UnitType):
data = {
UnitType.CENTIMETER: CentimeterUnit,
UnitType.POINT: PointUnit,
UnitType.INCH: InchUnit,
UnitType.DATA: DataUnit,
UnitType.RELATIVE: RelativeUnit,
UnitType.STR_WIDTH: StrWidthUnit,
UnitType.STR_HEIGHT: StrHeightUnit,
}
return data[kind]
@dataclass
class Quantity(ABC):
val: float
@staticmethod
def from_type(unit_type: UnitType, val: float) -> "Quantity":
return unit_type_from_type(unit_type)(val)
@staticmethod
def from_type_or_none(
unit_type: UnitType, val: Optional[float]
) -> Optional["Quantity"]:
if not val:
return None
return unit_type_from_type(unit_type)(val)
@property
@abstractmethod
def unit_type(self) -> UnitType:
pass
@staticmethod
def centimeters(val: float) -> "Quantity":
return CentimeterUnit(val=val)
@staticmethod
def points(val: float) -> "Quantity":
return PointUnit(val=val)
@staticmethod
def inches(val: float) -> "Quantity":
return InchUnit(val=val)
@staticmethod
def relative(val: float) -> "Quantity":
return RelativeUnit(val=val)
def embed_into(self, axis: AxisKind, view: "ViewPort") -> "Quantity":
from python_ggplot.core.embed import quantity_embed_into # pylint: disable=all
return quantity_embed_into(self, axis, view)
def to_relative_with_view(self, view: "ViewPort", axis: AxisKind):
length = view.to_relative_dimension(axis)
return self.to_relative(length=length, scale=deepcopy(view.x_scale))
def to_relative_from_view(
self, view: "ViewPort", axis_kind: AxisKind
) -> "Quantity":
from python_ggplot.core.units.convert import (
to_relative_from_view,
) # pylint: disable=all
return to_relative_from_view(self, view, axis_kind)
def to(
self,
kind: UnitType,
length: Optional["Quantity"] = None,
scale: Optional[Scale] = None,
) -> "Quantity":
from python_ggplot.core.units.convert import (
convert_quantity,
) # pylint: disable=all
return convert_quantity(self, kind, length=length, scale=scale)
def to_data(
self, length: Optional["Quantity"] = None, scale: Optional[Scale] = None
) -> "Quantity":
return self.to(UnitType.DATA, length=length, scale=scale)
def to_centimeter(
self, length: Optional["Quantity"] = None, scale: Optional[Scale] = None
) -> "Quantity":
return self.to(UnitType.CENTIMETER, length=length, scale=scale)
def to_inch(
self, length: Optional["Quantity"] = None, scale: Optional[Scale] = None
) -> "Quantity":
return self.to(UnitType.INCH, length=length, scale=scale)
def to_points(
self, length: Optional["Quantity"] = None, scale: Optional[Scale] = None
) -> "Quantity":
return self.to(UnitType.POINT, length=length, scale=scale)
def to_relative(
self, length: Optional["Quantity"] = None, scale: Optional[Scale] = None
) -> "Quantity":
return self.to(UnitType.RELATIVE, length=length, scale=scale)
def apply_operator(
self,
other: "Quantity",
length: Optional["Quantity"],
scale: Optional[Scale],
as_coordinate: bool, # noqa TODO fix
operator: Callable[[float, float], float],
operator_type: "OperatorType",
) -> "Quantity":
# TODO fix circular import
from python_ggplot.core.coord.objects import OperatorType
# todo refactor
# this is ugly, needs to become like the conversion eventually
if self.unit_type == other.unit_type:
cls = unit_type_from_type(self.unit_type)
return cls(operator(self.val, other.val))
elif self.unit_type.is_length() and other.unit_type == UnitType.RELATIVE:
if operator_type in {OperatorType.MUL, OperatorType.DIV}:
return PointUnit(operator(self.val, other.val)).to(self.unit_type)
else:
return PointUnit(
operator(self.to_points().val, other.to_points(length=length).val)
).to(self.unit_type, length=length, scale=scale)
elif self.unit_type.is_length() and (other.unit_type.is_length()):
return PointUnit(
operator(self.to_points().val, other.to_points(length=length).val)
).to(self.unit_type, length=length, scale=scale)
elif self.unit_type == UnitType.RELATIVE and other.unit_type.is_length():
if operator_type in {OperatorType.MUL, OperatorType.DIV}:
return PointUnit(operator(self.val, other.val)).to(other.unit_type)
else:
return PointUnit(
operator(self.to_points(length=length).val, other.to_points().val)
).to(self.unit_type, length=length, scale=scale)
elif self.unit_type == UnitType.DATA:
left = deepcopy(self)
if as_coordinate:
if not scale:
raise GGException("Scale is needed to convert unity type DATA")
left = DataUnit(self.val - scale.low)
left = left.to_relative(length=length, scale=scale).val
right = other.to_relative(length=length, scale=scale).val
return RelativeUnit(operator(left, right))
else:
raise GGException(f"Unsupported unit arithmetic for {self.unit_type}")
def multiply(
self,
other: "Quantity",
length: Optional["Quantity"] = None,
scale: Optional[Scale] = None,
as_coordinate: bool = False,
) -> "Quantity":
# TODO fix circular import
from python_ggplot.core.coord.objects import OperatorType
return self.apply_operator(
other, length, scale, as_coordinate, lambda a, b: a * b, OperatorType.MUL
)
def add(
self,
other: "Quantity",
length: Optional["Quantity"] = None,
scale: Optional[Scale] = None,
as_coordinate: bool = False,
) -> "Quantity":
# TODO fix circular import
from python_ggplot.core.coord.objects import OperatorType
return self.apply_operator(
other, length, scale, as_coordinate, lambda a, b: a + b, OperatorType.ADD
)
def divide(
self,
other: "Quantity",
length: Optional["Quantity"] = None,
scale: Optional[Scale] = None,
as_coordinate: bool = False,
) -> "Quantity":
# TODO fix circular import
from python_ggplot.core.coord.objects import OperatorType
return self.apply_operator(
other, length, scale, as_coordinate, lambda a, b: a / b, OperatorType.DIV
)
def subtract(
self,
other: "Quantity",
length: Optional["Quantity"] = None,
scale: Optional[Scale] = None,
as_coordinate: bool = False,
) -> "Quantity":
# TODO fix circular import
from python_ggplot.core.coord.objects import OperatorType
return self.apply_operator(
other, length, scale, as_coordinate, lambda a, b: a - b, OperatorType.SUB
)
class PointUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.POINT
class CentimeterUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.CENTIMETER
class InchUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.INCH
class RelativeUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.RELATIVE
class DataUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.DATA
class StrWidthUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.STR_WIDTH
class StrHeightUnit(Quantity):
@property
def unit_type(self) -> UnitType:
return UnitType.STR_HEIGHT
def add_data_quantity(
left: Quantity,
right: Quantity,
length: Optional[Quantity],
scale: Optional[Scale],
operator: Callable[[float, float], float],
) -> Quantity:
if scale is None:
raise ValueError("Expected scale")
final_q = DataUnit(left.val - scale.low)
left_relative = final_q.to_relative(length=length, scale=scale)
right_relative = right.to_relative(length=length, scale=scale)
val = operator(left_relative.val, right_relative.val)
return RelativeUnit(val)
def add_length_relative_quantity(
length_quantity: Quantity,
relative_quantity: Quantity,
operator: Callable[[float, float], float],
) -> Quantity:
val = operator(length_quantity.val, relative_quantity.val)
cls = unit_type_from_type(length_quantity.unit_type)
return cls(val)
def add_length_quantities(
left: Quantity, right: Quantity, operator: Callable[[float, float], float]
) -> Quantity:
left_converted = left.to(UnitType.POINT)
right_converted = right.to(UnitType.POINT)
val = operator(left_converted.val, right_converted.val)
point = PointUnit(val)
return point.to(left.unit_type)
def unit_type_from_type(kind: UnitType) -> Type[Quantity]:
data = {
UnitType.POINT: PointUnit,
UnitType.CENTIMETER: CentimeterUnit,
UnitType.INCH: InchUnit,
UnitType.RELATIVE: RelativeUnit,
UnitType.DATA: DataUnit,
UnitType.STR_WIDTH: StrWidthUnit,
UnitType.STR_HEIGHT: StrHeightUnit,
}
return data[kind]