]>
Commit | Line | Data |
---|---|---|
1 | from kivy.uix.relativelayout import RelativeLayout | |
2 | from kivy.properties import NumericProperty, ListProperty | |
3 | from kivy.core.window import Window | |
4 | from kivy.clock import Clock | |
5 | ||
6 | import threading | |
7 | import pygame | |
8 | import yaml | |
9 | import sys | |
10 | ||
11 | from .lock import * | |
12 | from .music_file import * | |
13 | from . import yml_file | |
14 | ||
15 | class Mapping(RelativeLayout): | |
16 | expected_keys = NumericProperty(0) | |
17 | ready_color = ListProperty([1, 165/255, 0, 1]) | |
18 | ||
19 | def __init__(self, **kwargs): | |
20 | self.key_config, self.channel_number, self.open_files = self.parse_config() | |
21 | super(Mapping, self).__init__(**kwargs) | |
22 | self._keyboard = Window.request_keyboard(self._keyboard_closed, self) | |
23 | self._keyboard.bind(on_key_down=self._on_keyboard_down) | |
24 | self.running = [] | |
25 | Clock.schedule_interval(self.not_all_keys_ready, 1) | |
26 | ||
27 | ||
28 | pygame.mixer.init(frequency = 44100) | |
29 | pygame.mixer.set_num_channels(self.channel_number) | |
30 | ||
31 | def _keyboard_closed(self): | |
32 | self._keyboard.unbind(on_key_down=self._on_keyboard_down) | |
33 | self._keyboard = None | |
34 | ||
35 | def _on_keyboard_down(self, keyboard, keycode, text, modifiers): | |
36 | key = self.find_by_key_code(keycode) | |
37 | if len(modifiers) == 0 and key is not None: | |
38 | threading.Thread(name = "MSKeyAction", target=key.do_actions).start() | |
39 | elif 'ctrl' in modifiers and (keycode[0] == 113 or keycode[0] == '99'): | |
40 | for thread in threading.enumerate(): | |
41 | if thread.getName()[0:2] != "MS": | |
42 | continue | |
43 | thread.join() | |
44 | ||
45 | pygame.quit() | |
46 | sys.exit() | |
47 | return True | |
48 | ||
49 | def find_by_key_code(self, key_code): | |
50 | if "Key_" + str(key_code[0]) in self.ids: | |
51 | return self.ids["Key_" + str(key_code[0])] | |
52 | return None | |
53 | ||
54 | def find_by_unicode(self, key_sym): | |
55 | for key in self.children: | |
56 | if not type(key).__name__ == "Key": | |
57 | continue | |
58 | print(key.key_sym, key_sym) | |
59 | if key.key_sym == key_sym: | |
60 | print("found") | |
61 | return key | |
62 | return None | |
63 | ||
64 | def not_all_keys_ready(self, dt): | |
65 | for key in self.children: | |
66 | if not type(key).__name__ == "Key": | |
67 | continue | |
68 | if not key.is_key_ready: | |
69 | return True | |
70 | self.ready_color = [0, 1, 0, 1] | |
71 | return False | |
72 | ||
73 | def stop_all_running(self): | |
74 | self.running = [] | |
75 | ||
76 | def start_running(self, key, start_time): | |
77 | self.running.append((key, start_time)) | |
78 | ||
79 | def keep_running(self, key, start_time): | |
80 | return (key, start_time) in self.running | |
81 | ||
82 | def finished_running(self, key, start_time): | |
83 | if (key, start_time) in self.running: | |
84 | self.running.remove((key, start_time)) | |
85 | ||
86 | def parse_config(self): | |
87 | stream = open(yml_file(), "r") | |
88 | config = yaml.load(stream) | |
89 | stream.close() | |
90 | ||
91 | aliases = config['aliases'] | |
92 | seen_files = {} | |
93 | ||
94 | file_lock = Lock("file") | |
95 | ||
96 | channel_id = 0 | |
97 | ||
98 | key_properties = {} | |
99 | ||
100 | for key in config['key_properties']: | |
101 | if key not in key_properties: | |
102 | key_prop = config['key_properties'][key] | |
103 | if 'include' in key_prop: | |
104 | included = key_prop['include'] | |
105 | del(key_prop['include']) | |
106 | ||
107 | if isinstance(included, str): | |
108 | key_prop.update(aliases[included], **key_prop) | |
109 | else: | |
110 | for included_ in included: | |
111 | key_prop.update(aliases[included_], **key_prop) | |
112 | ||
113 | key_properties[key] = { | |
114 | "actions": [], | |
115 | "properties": key_prop, | |
116 | "files": [] | |
117 | } | |
118 | ||
119 | for mapped_key in config['keys']: | |
120 | if mapped_key not in key_properties: | |
121 | key_properties[mapped_key] = { | |
122 | "actions": [], | |
123 | "properties": {}, | |
124 | "files": [] | |
125 | } | |
126 | for action in config['keys'][mapped_key]: | |
127 | action_name = list(action)[0] | |
128 | action_args = {} | |
129 | if action[action_name] is None: | |
130 | action[action_name] = [] | |
131 | ||
132 | if 'include' in action[action_name]: | |
133 | included = action[action_name]['include'] | |
134 | del(action[action_name]['include']) | |
135 | ||
136 | if isinstance(included, str): | |
137 | action[action_name].update(aliases[included], **action[action_name]) | |
138 | else: | |
139 | for included_ in included: | |
140 | action[action_name].update(aliases[included_], **action[action_name]) | |
141 | ||
142 | for argument in action[action_name]: | |
143 | if argument == 'file': | |
144 | filename = action[action_name]['file'] | |
145 | if filename not in seen_files: | |
146 | if filename in config['music_properties']: | |
147 | seen_files[filename] = MusicFile( | |
148 | filename, | |
149 | file_lock, | |
150 | channel_id, | |
151 | **config['music_properties'][filename]) | |
152 | else: | |
153 | seen_files[filename] = MusicFile( | |
154 | filename, | |
155 | file_lock, | |
156 | channel_id) | |
157 | channel_id = channel_id + 1 | |
158 | ||
159 | if filename not in key_properties[mapped_key]['files']: | |
160 | key_properties[mapped_key]['files'].append(seen_files[filename]) | |
161 | ||
162 | action_args['music'] = seen_files[filename] | |
163 | ||
164 | else: | |
165 | action_args[argument] = action[action_name][argument] | |
166 | ||
167 | key_properties[mapped_key]['actions'].append([action_name, action_args]) | |
168 | ||
169 | return (key_properties, channel_id + 1, seen_files) | |
170 | ||
171 |