-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdmx.py
169 lines (128 loc) · 4.66 KB
/
dmx.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
class JsonSerializeMixin:
def __json__(self, parents=set(), depth=3):
attrs = dir(self)
return {
attr: self.__json__object(getattr(self, attr), parents, depth-1) for attr in attrs if self.__json__suitable(attr)
}
def __json__suitable(self, attr):
# Magic/Private
if attr.startswith('_'):
return False
# Method
value = getattr(self, attr)
if type(value) is type(self.__init__):
return False
# Generator
if type(value) is type((x for x in (1,2))):
return False
return True
def __json__object(self, value, parents, depth):
if type(value) is list:
return [self.__json__object(item, parents, depth) for item in value]
if type(value) is dict:
return {key: self.__json__object(value, parents, depth) for key, value in value.items()}
if value in parents:
return None
new_parents = {self, *parents}
if hasattr(value, '__json__'):
return value.__json__(new_parents, depth)
return value
class Universe(JsonSerializeMixin):
def __init__(self, id, name):
self.name = name
self.id = id
self._devices = []
def __str__(self):
return f'<{self.__class__.__name__} {self.id} "{self.name}">'
def add(self, device):
if not issubclass(type(device), Device):
raise f'{device} does not appear to be a Device'
self._devices.append(device)
@property
def channels(self):
return (c for dev in self._devices for c in dev.channels)
@property
def devices(self):
return self._devices
def kill(self):
for dev in self._devices:
dev.kill()
class Device(JsonSerializeMixin):
def __init__(self, id, name):
self.name = name
self.id = id
self.universe = None
self._channels = []
for attr, channel in vars(self.__class__).items():
if not self._filter_channel(channel):
continue
new_channel = channel.init(self.id, self)
setattr(self, attr, new_channel)
self._channels.append(new_channel)
def __str__(self, specific=None):
if specific is None:
return f'<{self.__class__.__name__} {self.id} "{self.name}">'
return f'<{self.__class__.__name__} {self.id} "{self.name}" {specific}>'
@property
def channels(self):
return self._channels
def _filter_channel(self, var):
return issubclass(type(var), (Channel, OffsetChannel))
def _resolve_channel(self, channel):
return channel.init(self.id)
def kill(self):
for chan in self._channels:
chan.kill()
# TODO: Kill actions (Fade/Persist)
DARK = "DARK"
class Channel(JsonSerializeMixin):
def __init__(self, id, name, value=0, on_kill=DARK):
self.id = id
self.name = name
self._value = value
self.device = None
self.on_kill = on_kill
def __str__(self):
return f'<{self.__class__.__name__} {self.id} "{self.name}">'
@property
def value(self):
return self._value
@value.setter
def value(self, v):
if v > 255 or v < 0:
# TODO: Force limit and warn
raise ValueError(f"Value for {self} of {self.device} must be between 0 and 255 inclusive. Was {v}.")
self._value = v
def init(self, base_channel, device):
self.device = device
return self
def kill(self):
if self.on_kill == DARK:
self._value=0
class OffsetChannel:
def __init__(self, offset, name, value=0, on_kill=DARK):
if offset < 0 or offset > 512:
raise ValueError(f"Value {offset} for offset is invalid. Must be between 0 and 512.")
self._offset = offset
self._kwargs = {
'name': name,
'value': value,
'on_kill': on_kill
}
def __repr__(self):
return f"{self.__class__.__name__}(offset={self._offset}, name=\"{self._kwargs['name']}\", value={self._kwargs['value']})"
def __str__(self):
return f'<{self.__class__.__name__} {self._offset:+} "{self._kwargs["name"]}">'
def init(self, base_channel, device):
chan = Channel(base_channel + self._offset, **self._kwargs)
chan.device = device
return chan
# TODO: Split this out
class RGBAPar(Device):
Red = OffsetChannel(0, "Red")
Green = OffsetChannel(1, "Green")
Blue = OffsetChannel(2, "Blue")
Amber = OffsetChannel(3, "Amber")
def __str__(self):
extra = " ".join(map(lambda c: str(c), self._channels))
return super().__str__(extra)