]> git.immae.eu Git - perso/Immae/Projets/Python/MusicSampler.git/blame - music_sampler/mapping.py
Improve error message and handling
[perso/Immae/Projets/Python/MusicSampler.git] / music_sampler / mapping.py
CommitLineData
4b2d79ca 1from kivy.uix.relativelayout import RelativeLayout
ab47d2a1 2from kivy.properties import NumericProperty, ListProperty, StringProperty
4b2d79ca 3from kivy.core.window import Window
30d8796f 4from kivy.clock import Clock
4b2d79ca 5
be27763f 6import threading
4b2d79ca 7import yaml
b68b4e8f 8import sys
05d0d2ed 9from collections import defaultdict
4b2d79ca 10
8ba7d831
IB
11from transitions.extensions import HierarchicalMachine as Machine
12
e55b29bb 13from .music_file import MusicFile
22514f3a 14from .mixer import Mixer
6ebe6247 15from .helpers import Config, gain, error_print, warn_print
3aaddc9d 16from .action import Action
4b2d79ca
IB
17
18class Mapping(RelativeLayout):
8ba7d831
IB
19 STATES = [
20 'initial',
21 'configuring',
22 'configured',
23 'loading',
24 'loaded',
25 'failed'
26 ]
27
28 TRANSITIONS = [
29 {
30 'trigger': 'configure',
31 'source': 'initial',
32 'dest': 'configuring'
33 },
34 {
35 'trigger': 'fail',
36 'source': 'configuring',
37 'dest': 'failed'
38 },
39 {
40 'trigger': 'success',
41 'source': 'configuring',
42 'dest': 'configured',
43 'after': 'load'
44 },
45 {
46 'trigger': 'load',
47 'source': 'configured',
48 'dest': 'loading'
49 },
50 {
51 'trigger': 'fail',
52 'source': 'loading',
53 'dest': 'failed'
54 },
55 {
56 'trigger': 'success',
57 'source': 'loading',
58 'dest': 'loaded'
ab47d2a1
IB
59 },
60 {
61 'trigger': 'reload',
62 'source': 'loaded',
63 'dest': 'configuring'
8ba7d831
IB
64 }
65 ]
66
1b4b78f5 67 master_volume = NumericProperty(100)
30d8796f 68 ready_color = ListProperty([1, 165/255, 0, 1])
ab47d2a1 69 state = StringProperty("")
4b2d79ca
IB
70
71 def __init__(self, **kwargs):
ab47d2a1
IB
72 self.keys = []
73 self.running = []
74 self.wait_ids = {}
75 self.open_files = {}
76
77 Machine(model=self, states=self.STATES,
78 transitions=self.TRANSITIONS, initial='initial',
79 ignore_invalid_triggers=True, queued=True)
80 super(Mapping, self).__init__(**kwargs)
81 self.keyboard = Window.request_keyboard(self.on_keyboard_closed, self)
82 self.keyboard.bind(on_key_down=self.on_keyboard_down)
83
84 self.configure()
85
86 def on_enter_configuring(self):
d6290f14 87 if Config.builtin_mixing:
af27d782 88 self.mixer = Mixer()
d6290f14
IB
89 else:
90 self.mixer = None
9c4f705f
IB
91
92 try:
93 self.key_config, self.open_files = self.parse_config()
94 except Exception as e:
05d0d2ed 95 error_print("Error while loading configuration: {}".format(e),
2010311b 96 with_trace=True, exit=True)
ab47d2a1
IB
97 else:
98 self.success()
9c4f705f 99
ab47d2a1
IB
100 def on_enter_loading(self):
101 for key in self.keys:
102 key.reload()
103 self.success()
1b4b78f5 104
8ba7d831
IB
105 # Kivy events
106 def add_widget(self, widget, index=0):
107 if type(widget).__name__ == "Key" and widget not in self.keys:
108 self.keys.append(widget)
109 return super(Mapping, self).add_widget(widget, index)
3aaddc9d 110
8ba7d831
IB
111 def remove_widget(self, widget, index=0):
112 if type(widget).__name__ == "Key" and widget in self.keys:
113 self.keys.remove(widget)
114 return super(Mapping, self).remove_widget(widget, index)
3aaddc9d 115
8ba7d831
IB
116 def on_keyboard_closed(self):
117 self.keyboard.unbind(on_key_down=self.on_keyboard_down)
118 self.keyboard = None
4b2d79ca 119
8ba7d831 120 def on_keyboard_down(self, keyboard, keycode, text, modifiers):
4b2d79ca 121 key = self.find_by_key_code(keycode)
4b6d1836
IB
122 if self.allowed_modifiers(modifiers) and key is not None:
123 modifiers.sort()
124 threading.Thread(name="MSKeyAction", target=key.run,
125 args=['-'.join(modifiers)]).start()
b68b4e8f 126 elif 'ctrl' in modifiers and (keycode[0] == 113 or keycode[0] == '99'):
a1d7f30a 127 self.stop_all_running()
b68b4e8f
IB
128 for thread in threading.enumerate():
129 if thread.getName()[0:2] != "MS":
130 continue
131 thread.join()
132
b68b4e8f 133 sys.exit()
ab47d2a1
IB
134 elif 'ctrl' in modifiers and keycode[0] == 114:
135 threading.Thread(name="MSReload", target=self.reload).start()
4b2d79ca
IB
136 return True
137
4b6d1836
IB
138 # Helpers
139 def allowed_modifiers(self, modifiers):
140 allowed = []
141 return len([a for a in modifiers if a not in allowed]) == 0
142
4b2d79ca
IB
143 def find_by_key_code(self, key_code):
144 if "Key_" + str(key_code[0]) in self.ids:
145 return self.ids["Key_" + str(key_code[0])]
be27763f
IB
146 return None
147
8ba7d831
IB
148 def all_keys_ready(self):
149 partial = False
150 for key in self.keys:
e55b29bb 151 if not key.is_loaded_or_failed():
8ba7d831
IB
152 return "not_ready"
153 partial = partial or key.is_failed()
154
155 if partial:
156 return "partial"
157 else:
158 return "success"
159
160 # Callbacks
161 def key_loaded_callback(self):
162 result = self.all_keys_ready()
163 if result == "success":
164 self.ready_color = [0, 1, 0, 1]
165 elif result == "partial":
166 self.ready_color = [1, 0, 0, 1]
ab47d2a1
IB
167 else:
168 self.ready_color = [1, 165/255, 0, 1]
30d8796f 169
8ba7d831 170 ## Some global actions
62a8b07a 171 def stop_all_running(self, except_key=None, key_start_time=0):
0deb82a5 172 running = self.running
62a8b07a
IB
173 self.running = [r for r in running\
174 if r[0] == except_key and r[1] == key_start_time]
0deb82a5 175 for (key, start_time) in running:
62a8b07a
IB
176 if (key, start_time) != (except_key, key_start_time):
177 key.interrupt()
be27763f 178
8ba7d831
IB
179 # Master volume methods
180 @property
181 def master_gain(self):
182 return gain(self.master_volume)
183
184 def set_master_volume(self, value, delta=False, fade=0):
185 [db_gain, self.master_volume] = gain(
186 value + int(delta) * self.master_volume,
187 self.master_volume)
188
189 for music in self.open_files.values():
190 music.set_gain_with_effect(db_gain, fade=fade)
191
192 # Wait handler methods
193 def add_wait_id(self, wait_id, action_or_wait):
194 self.wait_ids[wait_id] = action_or_wait
195
196 def interrupt_wait(self, wait_id):
197 if wait_id in self.wait_ids:
198 action_or_wait = self.wait_ids[wait_id]
199 del(self.wait_ids[wait_id])
200 if isinstance(action_or_wait, Action):
201 action_or_wait.interrupt()
202 else:
203 action_or_wait.set()
204
205 # Methods to control running keys
be27763f
IB
206 def start_running(self, key, start_time):
207 self.running.append((key, start_time))
208
209 def keep_running(self, key, start_time):
210 return (key, start_time) in self.running
211
212 def finished_running(self, key, start_time):
213 if (key, start_time) in self.running:
214 self.running.remove((key, start_time))
215
8ba7d831 216 # YML config parser
4b2d79ca 217 def parse_config(self):
05d0d2ed
IB
218 def update_alias(prop_hash, aliases, key):
219 if isinstance(aliases[key], dict):
220 prop_hash.update(aliases[key], **prop_hash)
221 else:
222 warn_print("Alias {} is not a hash, ignored".format(key))
223
224 def include_aliases(prop_hash, aliases):
225 if 'include' not in prop_hash:
226 return
227
228 included = prop_hash['include']
229 del(prop_hash['include'])
230 if isinstance(included, str):
231 update_alias(prop_hash, aliases, included)
232 elif isinstance(included, list):
233 for included_ in included:
234 if isinstance(included_, str):
235 update_alias(prop_hash, aliases, included_)
236 else:
237 warn_print("Unkown alias include type, ignored: "
238 "{} in {}".format(included_, included))
239 else:
240 warn_print("Unkown alias include type, ignored: {}"
241 .format(included))
242
243 def check_key_property(key_property, key):
244 if 'description' in key_property:
245 desc = key_property['description']
246 if not isinstance(desc, list):
247 warn_print("description in key_property '{}' is not "
248 "a list, ignored".format(key))
249 del(key_property['description'])
250 if 'color' in key_property:
251 color = key_property['color']
252 if not isinstance(color, list)\
253 or len(color) != 3\
254 or not all(isinstance(item, int) for item in color)\
255 or any(item < 0 or item > 255 for item in color):
256 warn_print("color in key_property '{}' is not "
257 "a list of 3 valid integers, ignored".format(key))
258 del(key_property['color'])
259
260 def check_key_properties(config):
261 if 'key_properties' in config:
262 if isinstance(config['key_properties'], dict):
263 return config['key_properties']
264 else:
265 warn_print("key_properties config is not a hash, ignored")
266 return {}
267 else:
268 return {}
269
270 def check_mapped_keys(config):
271 if 'keys' in config:
272 if isinstance(config['keys'], dict):
273 return config['keys']
274 else:
275 warn_print("keys config is not a hash, ignored")
276 return {}
277 else:
278 return {}
279
280 def check_mapped_key(mapped_keys, key):
281 if not isinstance(mapped_keys[key], list):
282 warn_print("key config '{}' is not an array, ignored"
283 .format(key))
284 return []
285 else:
286 return mapped_keys[key]
287
288 def check_music_property(music_property, filename):
289 if not isinstance(music_property, dict):
290 warn_print("music_property config '{}' is not a hash, ignored"
291 .format(filename))
292 return {}
293 if 'name' in music_property:
294 music_property['name'] = str(music_property['name'])
295 if 'gain' in music_property:
296 try:
297 music_property['gain'] = float(music_property['gain'])
298 except ValueError as e:
299 del(music_property['gain'])
300 warn_print("gain for music_property '{}' is not "
301 "a float, ignored".format(filename))
302 return music_property
303
75d6cdba 304 stream = open(Config.yml_file, "r")
6c44b231 305 try:
05d0d2ed 306 config = yaml.safe_load(stream)
aee1334c 307 except Exception as e:
2010311b
IB
308 error_print("Error while loading config file: {}".format(e),
309 exit=True)
4b2d79ca
IB
310 stream.close()
311
05d0d2ed 312 if not isinstance(config, dict):
2010311b
IB
313 error_print("Top level config is supposed to be a hash",
314 exit=True)
05d0d2ed
IB
315
316 if 'aliases' in config and isinstance(config['aliases'], dict):
317 aliases = config['aliases']
318 else:
319 aliases = defaultdict(dict)
320 if 'aliases' in config:
321 warn_print("aliases config is not a hash, ignored")
322
323 music_properties = defaultdict(dict)
324 if 'music_properties' in config and\
325 isinstance(config['music_properties'], dict):
326 music_properties.update(config['music_properties'])
327 elif 'music_properties' in config:
328 warn_print("music_properties config is not a hash, ignored")
329
4b2d79ca
IB
330 seen_files = {}
331
05d0d2ed
IB
332 key_properties = defaultdict(lambda: {
333 "actions": [],
334 "properties": {},
335 "files": []
336 })
4b2d79ca 337
05d0d2ed
IB
338 for key in check_key_properties(config):
339 key_prop = config['key_properties'][key]
340
341 if not isinstance(key_prop, dict):
342 warn_print("key_property '{}' is not a hash, ignored"
343 .format(key))
344 continue
345
346 include_aliases(key_prop, aliases)
347 check_key_property(key_prop, key)
348
349 key_properties[key]["properties"] = key_prop
350
351 for mapped_key in check_mapped_keys(config):
352 for index, action in enumerate(check_mapped_key(
353 config['keys'], mapped_key)):
354 if not isinstance(action, dict) or\
355 not len(action) == 1 or\
356 not isinstance(list(action.values())[0] or {}, dict):
357 warn_print("action number {} of key '{}' is invalid, "
358 "ignored".format(index + 1, mapped_key))
359 continue
e5edd8b9 360
4b2d79ca
IB
361 action_name = list(action)[0]
362 action_args = {}
363 if action[action_name] is None:
05d0d2ed 364 action[action_name] = {}
4b2d79ca 365
05d0d2ed 366 include_aliases(action[action_name], aliases)
4b2d79ca
IB
367
368 for argument in action[action_name]:
369 if argument == 'file':
05d0d2ed 370 filename = str(action[action_name]['file'])
4b2d79ca 371 if filename not in seen_files:
05d0d2ed
IB
372 music_property = check_music_property(
373 music_properties[filename],
374 filename)
375
ab47d2a1
IB
376 if filename in self.open_files:
377 self.open_files[filename]\
378 .reload_properties(**music_property)
379
6c42e32d
IB
380 seen_files[filename] =\
381 self.open_files[filename]
ab47d2a1
IB
382 else:
383 seen_files[filename] = MusicFile(
384 filename, self, **music_property)
4b2d79ca
IB
385
386 if filename not in key_properties[mapped_key]['files']:
2e404903
IB
387 key_properties[mapped_key]['files'] \
388 .append(seen_files[filename])
4b2d79ca
IB
389
390 action_args['music'] = seen_files[filename]
4b2d79ca
IB
391 else:
392 action_args[argument] = action[action_name][argument]
393
2e404903
IB
394 key_properties[mapped_key]['actions'] \
395 .append([action_name, action_args])
4b2d79ca 396
29597680 397 return (key_properties, seen_files)
4b2d79ca
IB
398
399