-
Notifications
You must be signed in to change notification settings - Fork 115
/
Copy pathangle_properties.py
executable file
·137 lines (113 loc) · 4.21 KB
/
angle_properties.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
#!/usr/bin/env python3
# A cheatsheet for how to use Python properties and operator overloading.
# Not intended for use in actual programs, though you're welcome to do
# so if you want. No license, public domain.
import math
class Angle:
"""Stores one angle.
Internally it's stored in radians as a float,
but you can set or get it with a.radians or a.degrees.
If you set it in the constructor to a value over 2*pi,
without specifying radians or degrees,
it will assume degrees and convert accordingly.
NOTE: This isn't necessarily a good idea!
But it's a good way to show how to use property getters/setters
and operator overloading.
"""
TWO_PI = math.pi * 2
TO_RADIANS = math.pi / 180
def __init__(self, *args, **kwargs):
"""Pass in the value of the angle, as radians=r or degrees=d.
If you just pass in a number, Angle will try to guess
based on whether it's over 2*pi.
Usage examples:
Angle(radians=math.pi)
Angle(degrees = 45)
Angle(1.44)
"""
print("Angle constructor: args", args, "kwargs", kwargs)
if 'radians' in kwargs:
self.radians = kwargs['radians']
elif 'degrees' in kwargs:
self.degrees = kwargs['degrees']
elif len(args) == 1:
if abs(args[0]) < Angle.TWO_PI:
self.radians = args[0]
else:
self.degrees = args[0]
else:
raise RuntimeError("Angle: Can't make sense of arguments")
def constrain_radians(self):
"""Make the angle positive and less than 2pi"""
self._radians %= Angle.TWO_PI
@property
def radians(self):
return self._radians
@radians.setter
def radians(self, val):
self._radians = val
self.constrain_radians()
print("radians setter", val, "->", self._radians)
@property
def degrees(self):
return self._radians / Angle.TO_RADIANS
@degrees.setter
def degrees(self, val):
print("degrees_setter", val)
self.radians = val * Angle.TO_RADIANS
def __repr__(self):
return "<Angle %.1f°>" % (self.degrees)
def __add__(self, a):
if not hasattr(a, 'radians'):
print(a, "not an angle; ", end='')
a = Angle(a)
print("->", a)
return Angle(radians=self.radians+a.radians)
if __name__ == '__main__':
# Values to be passed as non-keyword args
vals = [.4, 310, -15, 400]
# and the degrees that should result from each
expected = [22.918, 310, 345, 40]
def too_different(x, y):
return abs(x - y) > .01
angles = []
for (v, ta) in zip(vals, expected):
print("\n==== Setting", v)
ang = Angle(v)
angles.append(ang)
if too_different(ang.degrees, ta):
print("**************", v, "->", ang.degrees, "should be", ta)
else:
print(v, "->", ang)
# Test specifying arguments explicitly
print("\n==== Testing explicit constructors")
a = Angle(degrees=90)
if too_different(a.radians, math.pi/2.):
print(a, "should be 90, and radians is", a.radians)
if too_different(a.degrees, 90):
print(a, "should be 90", a.degrees)
a = Angle(radians=-math.pi/2.)
if too_different(a.radians, math.pi*3//2.):
print(a, "in radians is", a.radians)
if too_different(a.degrees, 270):
print(a, "should be 270", a.degrees)
print("\n==== Testing addition of two angles")
# For addition tests, just use the last angle, which is -90
expected = [22.918+270, 310-90, 345-90, 40+270]
for a1, exp in zip(angles, expected):
print("\n", a, "+", a1)
sum = a + a1
if too_different(sum.degrees, exp):
print("************", sum, "!=", exp)
else:
print(" ====", sum)
print("\n==== Testing addition of angle + scalar")
scalars = [ 10, -10., math.pi/2. ]
expected = [ 280, 260, 0 ]
for v, exp in zip(scalars, expected):
print("\n", a, "+", v)
sum = a + v
if too_different(sum.degrees, exp):
print("************", sum, "!=", exp)
else:
print(" ====", sum)