diff options
-rw-r--r-- | helpers/__init__.py | 44 | ||||
-rw-r--r-- | helpers/mapping.py | 7 | ||||
-rw-r--r-- | helpers/mixer.py | 27 | ||||
-rw-r--r-- | helpers/music_file.py | 5 |
4 files changed, 59 insertions, 24 deletions
diff --git a/helpers/__init__.py b/helpers/__init__.py index 4b9529d..807aa44 100644 --- a/helpers/__init__.py +++ b/helpers/__init__.py | |||
@@ -6,11 +6,7 @@ import math | |||
6 | import sounddevice as sd | 6 | import sounddevice as sd |
7 | 7 | ||
8 | class Config: | 8 | class Config: |
9 | def __init__(self, **kwargs): | 9 | pass |
10 | for arg in kwargs: | ||
11 | setattr(self, arg, kwargs[arg]) | ||
12 | |||
13 | config = Config(yml_file="config.yml") | ||
14 | 10 | ||
15 | def path(): | 11 | def path(): |
16 | if getattr(sys, 'frozen', False): | 12 | if getattr(sys, 'frozen', False): |
@@ -29,11 +25,37 @@ def parse_args(): | |||
29 | 25 | ||
30 | sys.argv.extend(kivy_args) | 26 | sys.argv.extend(kivy_args) |
31 | 27 | ||
32 | parser = argparse.ArgumentParser(description="A Music Sampler application.") | 28 | parser = argparse.ArgumentParser( |
29 | description="A Music Sampler application.", | ||
30 | formatter_class=argparse.ArgumentDefaultsHelpFormatter) | ||
33 | parser.add_argument("-c", "--config", | 31 | parser.add_argument("-c", "--config", |
34 | default="config.yml", | 32 | default="config.yml", |
35 | required=False, | 33 | required=False, |
36 | help="Config file to load") | 34 | help="Config file to load") |
35 | parser.add_argument("-l", "--latency", | ||
36 | default="high", | ||
37 | required=False, | ||
38 | help="Latency: low, high or number of seconds") | ||
39 | parser.add_argument("-b", "--blocksize", | ||
40 | default=0, | ||
41 | type=int, | ||
42 | required=False, | ||
43 | help="Blocksize: If not 0, the numbe of frames to take at each step for the mixer") | ||
44 | parser.add_argument("-f", "--frame-rate", | ||
45 | default=44100, | ||
46 | type=int, | ||
47 | required=False, | ||
48 | help="Frame rate to play the musics") | ||
49 | parser.add_argument("-x", "--channels", | ||
50 | default=2, | ||
51 | type=int, | ||
52 | required=False, | ||
53 | help="Number of channels to use") | ||
54 | parser.add_argument("-s", "--sample-width", | ||
55 | default=2, | ||
56 | type=int, | ||
57 | required=False, | ||
58 | help="Sample width (number of bytes for each frame)") | ||
37 | parser.add_argument("-V", "--version", | 59 | parser.add_argument("-V", "--version", |
38 | action="version", | 60 | action="version", |
39 | help="Displays the current version and exits. Only use in bundled package", | 61 | help="Displays the current version and exits. Only use in bundled package", |
@@ -52,7 +74,12 @@ def parse_args(): | |||
52 | help="Kivy arguments. All arguments after this are interpreted by Kivy. Pass \"-- --help\" to get Kivy's usage.") | 74 | help="Kivy arguments. All arguments after this are interpreted by Kivy. Pass \"-- --help\" to get Kivy's usage.") |
53 | args = parser.parse_args(argv) | 75 | args = parser.parse_args(argv) |
54 | 76 | ||
55 | config.yml_file = args.config | 77 | Config.yml_file = args.config |
78 | Config.latency = args.latency | ||
79 | Config.blocksize = args.blocksize | ||
80 | Config.frame_rate = args.frame_rate | ||
81 | Config.channels = args.channels | ||
82 | Config.sample_width = args.sample_width | ||
56 | 83 | ||
57 | class SelectDeviceAction(argparse.Action): | 84 | class SelectDeviceAction(argparse.Action): |
58 | def __call__(self, parser, namespace, values, option_string=None): | 85 | def __call__(self, parser, namespace, values, option_string=None): |
@@ -71,9 +98,6 @@ def show_version(): | |||
71 | else: | 98 | else: |
72 | return "option '-v' can only be used in bundled package" | 99 | return "option '-v' can only be used in bundled package" |
73 | 100 | ||
74 | def yml_file(): | ||
75 | return config.yml_file | ||
76 | |||
77 | def duration_to_min_sec(duration): | 101 | def duration_to_min_sec(duration): |
78 | minutes = int(duration / 60) | 102 | minutes = int(duration / 60) |
79 | seconds = int(duration) % 60 | 103 | seconds = int(duration) % 60 |
diff --git a/helpers/mapping.py b/helpers/mapping.py index 60c7691..8e0265c 100644 --- a/helpers/mapping.py +++ b/helpers/mapping.py | |||
@@ -9,7 +9,7 @@ import sys | |||
9 | 9 | ||
10 | from .music_file import * | 10 | from .music_file import * |
11 | from .mixer import Mixer | 11 | from .mixer import Mixer |
12 | from . import yml_file,gain | 12 | from . import Config, gain |
13 | 13 | ||
14 | class Mapping(RelativeLayout): | 14 | class Mapping(RelativeLayout): |
15 | expected_keys = NumericProperty(0) | 15 | expected_keys = NumericProperty(0) |
@@ -17,6 +17,7 @@ class Mapping(RelativeLayout): | |||
17 | ready_color = ListProperty([1, 165/255, 0, 1]) | 17 | ready_color = ListProperty([1, 165/255, 0, 1]) |
18 | 18 | ||
19 | def __init__(self, **kwargs): | 19 | def __init__(self, **kwargs): |
20 | self.mixer = Mixer() | ||
20 | self.key_config, self.open_files = self.parse_config() | 21 | self.key_config, self.open_files = self.parse_config() |
21 | super(Mapping, self).__init__(**kwargs) | 22 | super(Mapping, self).__init__(**kwargs) |
22 | self._keyboard = Window.request_keyboard(self._keyboard_closed, self) | 23 | self._keyboard = Window.request_keyboard(self._keyboard_closed, self) |
@@ -24,8 +25,6 @@ class Mapping(RelativeLayout): | |||
24 | self.running = [] | 25 | self.running = [] |
25 | Clock.schedule_interval(self.not_all_keys_ready, 1) | 26 | Clock.schedule_interval(self.not_all_keys_ready, 1) |
26 | 27 | ||
27 | self.mixer = Mixer() | ||
28 | |||
29 | @property | 28 | @property |
30 | def master_gain(self): | 29 | def master_gain(self): |
31 | return gain(self.master_volume) | 30 | return gain(self.master_volume) |
@@ -83,7 +82,7 @@ class Mapping(RelativeLayout): | |||
83 | self.running.remove((key, start_time)) | 82 | self.running.remove((key, start_time)) |
84 | 83 | ||
85 | def parse_config(self): | 84 | def parse_config(self): |
86 | stream = open(yml_file(), "r") | 85 | stream = open(Config.yml_file, "r") |
87 | config = yaml.load(stream) | 86 | config = yaml.load(stream) |
88 | stream.close() | 87 | stream.close() |
89 | 88 | ||
diff --git a/helpers/mixer.py b/helpers/mixer.py index 9e8179a..d08520a 100644 --- a/helpers/mixer.py +++ b/helpers/mixer.py | |||
@@ -2,17 +2,28 @@ import sounddevice as sd | |||
2 | import audioop | 2 | import audioop |
3 | import time | 3 | import time |
4 | 4 | ||
5 | frame_rate = 44100 | 5 | from . import Config |
6 | channels = 2 | 6 | |
7 | sample_width = 2 | 7 | sample_width = Config.sample_width |
8 | def sample_width_to_dtype(sample_width): | ||
9 | if sample_width == 1 or sample_width == 2 or sample_width == 4: | ||
10 | return 'int' + str(8*sample_width) | ||
11 | else: | ||
12 | raise "Unknown sample width" | ||
13 | |||
14 | def _latency(latency): | ||
15 | if latency == "high" or latency == "low": | ||
16 | return latency | ||
17 | else: | ||
18 | return float(latency) | ||
8 | 19 | ||
9 | class Mixer: | 20 | class Mixer: |
10 | def __init__(self): | 21 | def __init__(self): |
11 | self.stream = sd.RawOutputStream(samplerate=frame_rate, | 22 | self.stream = sd.RawOutputStream(samplerate=Config.frame_rate, |
12 | channels=channels, | 23 | channels=Config.channels, |
13 | dtype='int' + str(8*sample_width), # FIXME: ? | 24 | dtype=sample_width_to_dtype(Config.sample_width), |
14 | latency="high", | 25 | latency=_latency(Config.latency), |
15 | blocksize=5000, | 26 | blocksize=Config.blocksize, |
16 | callback=self.play_callback, | 27 | callback=self.play_callback, |
17 | ) | 28 | ) |
18 | self.open_files = [] | 29 | self.open_files = [] |
diff --git a/helpers/music_file.py b/helpers/music_file.py index 6da547b..f1aa341 100644 --- a/helpers/music_file.py +++ b/helpers/music_file.py | |||
@@ -6,7 +6,7 @@ from transitions.extensions import HierarchicalMachine as Machine | |||
6 | import os.path | 6 | import os.path |
7 | 7 | ||
8 | from .lock import Lock | 8 | from .lock import Lock |
9 | from . import gain | 9 | from . import Config, gain |
10 | 10 | ||
11 | file_lock = Lock("file") | 11 | file_lock = Lock("file") |
12 | 12 | ||
@@ -48,7 +48,7 @@ class MusicFile(Machine): | |||
48 | try: | 48 | try: |
49 | print("Loading « {} »".format(self.name)) | 49 | print("Loading « {} »".format(self.name)) |
50 | db_gain = gain(self.volume_factor * 100) | 50 | db_gain = gain(self.volume_factor * 100) |
51 | self.audio_segment = pydub.AudioSegment.from_file(self.filename).set_frame_rate(44100).set_channels(2).set_sample_width(2).apply_gain(db_gain) | 51 | self.audio_segment = pydub.AudioSegment.from_file(self.filename).set_frame_rate(Config.frame_rate).set_channels(Config.channels).set_sample_width(Config.sample_width).apply_gain(db_gain) |
52 | self.audio_segment_frame_width = self.audio_segment.frame_width | 52 | self.audio_segment_frame_width = self.audio_segment.frame_width |
53 | self.sound_duration = self.audio_segment.duration_seconds | 53 | self.sound_duration = self.audio_segment.duration_seconds |
54 | except Exception as e: | 54 | except Exception as e: |
@@ -121,6 +121,7 @@ class MusicFile(Machine): | |||
121 | data += new_data | 121 | data += new_data |
122 | nb_frames += new_nb_frames | 122 | nb_frames += new_nb_frames |
123 | elif nb_frames == 0: | 123 | elif nb_frames == 0: |
124 | # FIXME: too slow | ||
124 | threading.Thread(name = "MSFinishedCallback", target=self.finished_callback).start() | 125 | threading.Thread(name = "MSFinishedCallback", target=self.finished_callback).start() |
125 | 126 | ||
126 | return data.ljust(out_data_length, b'\0') | 127 | return data.ljust(out_data_length, b'\0') |