@devaloop/devalang 0.0.1-beta.2 → 0.0.1-beta.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (159) hide show
  1. package/Cargo.toml +84 -81
  2. package/README.md +3 -2
  3. package/docs/CHANGELOG.md +41 -0
  4. package/docs/ROADMAP.md +3 -3
  5. package/examples/chain.deva +19 -0
  6. package/examples/plugin.deva +10 -10
  7. package/examples/routing.deva +23 -0
  8. package/out-tsc/bin/project-version.json +6 -0
  9. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
  10. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  11. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  12. package/package.json +23 -10
  13. package/project-version.json +3 -3
  14. package/rust/bindings/Cargo.toml +9 -0
  15. package/rust/bindings/src/lib.rs +86 -0
  16. package/rust/cli/addon/commands.rs +35 -0
  17. package/rust/cli/addon/download.rs +234 -0
  18. package/rust/cli/addon/install.rs +33 -0
  19. package/rust/cli/addon/list.rs +224 -0
  20. package/rust/cli/addon/metadata.rs +124 -0
  21. package/rust/cli/addon/mod.rs +8 -0
  22. package/rust/cli/addon/remove.rs +271 -0
  23. package/rust/cli/addon/update.rs +305 -0
  24. package/rust/cli/{install/addon.rs → addon/utils.rs} +109 -118
  25. package/rust/cli/build/commands.rs +153 -153
  26. package/rust/cli/build/process.rs +165 -165
  27. package/rust/cli/check/mod.rs +208 -208
  28. package/rust/cli/discover/commands.rs +275 -253
  29. package/rust/cli/discover/config.rs +109 -111
  30. package/rust/cli/discover/fs.rs +19 -19
  31. package/rust/cli/discover/install.rs +214 -103
  32. package/rust/cli/discover/metadata.rs +48 -48
  33. package/rust/cli/discover/mod.rs +5 -5
  34. package/rust/cli/me/commands.rs +52 -0
  35. package/rust/cli/me/mod.rs +1 -0
  36. package/rust/cli/mod.rs +12 -12
  37. package/rust/cli/parser.rs +30 -69
  38. package/rust/cli/play/commands.rs +375 -375
  39. package/rust/cli/play/process.rs +159 -159
  40. package/rust/core/audio/engine/driver.rs +19 -2
  41. package/rust/core/audio/engine/export.rs +169 -169
  42. package/rust/core/audio/engine/mod.rs +56 -56
  43. package/rust/core/audio/engine/notes/dsp.rs +88 -85
  44. package/rust/core/audio/engine/notes/mod.rs +53 -44
  45. package/rust/core/audio/engine/notes/params.rs +294 -294
  46. package/rust/core/audio/engine/sample/insert.rs +148 -47
  47. package/rust/core/audio/engine/sample/mod.rs +40 -40
  48. package/rust/core/audio/engine/sample/padding.rs +170 -170
  49. package/rust/core/audio/evaluator/condition.rs +61 -61
  50. package/rust/core/audio/evaluator/numeric.rs +152 -152
  51. package/rust/core/audio/evaluator/rhs.rs +16 -16
  52. package/rust/core/audio/evaluator/string_expr.rs +94 -94
  53. package/rust/core/audio/interpreter/driver.rs +574 -574
  54. package/rust/core/audio/interpreter/mod.rs +2 -2
  55. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
  56. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
  57. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  58. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
  59. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
  60. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
  61. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
  62. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
  63. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
  64. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
  65. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
  66. package/rust/core/audio/interpreter/statements/automate.rs +16 -16
  67. package/rust/core/audio/interpreter/statements/call.rs +31 -1
  68. package/rust/core/audio/interpreter/statements/condition.rs +72 -72
  69. package/rust/core/audio/interpreter/statements/function.rs +24 -24
  70. package/rust/core/audio/interpreter/statements/let_.rs +36 -36
  71. package/rust/core/audio/interpreter/statements/load.rs +17 -17
  72. package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
  73. package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
  74. package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
  75. package/rust/core/audio/loader/trigger.rs +98 -98
  76. package/rust/core/audio/player.rs +70 -70
  77. package/rust/core/audio/special/mod.rs +9 -9
  78. package/rust/core/builder/mod.rs +129 -129
  79. package/rust/core/debugger/lexer.rs +27 -27
  80. package/rust/core/debugger/logs.rs +52 -52
  81. package/rust/core/debugger/preprocessor.rs +27 -27
  82. package/rust/core/debugger/store.rs +38 -38
  83. package/rust/core/lexer/driver.rs +59 -59
  84. package/rust/core/lexer/handler/arrow.rs +82 -82
  85. package/rust/core/lexer/handler/at.rs +21 -21
  86. package/rust/core/lexer/handler/brace.rs +41 -41
  87. package/rust/core/lexer/handler/colon.rs +21 -21
  88. package/rust/core/lexer/handler/comment.rs +30 -30
  89. package/rust/core/lexer/handler/dot.rs +21 -21
  90. package/rust/core/lexer/handler/driver.rs +337 -337
  91. package/rust/core/lexer/handler/identifier.rs +47 -47
  92. package/rust/core/lexer/handler/indent.rs +66 -66
  93. package/rust/core/lexer/handler/mod.rs +15 -15
  94. package/rust/core/lexer/handler/newline.rs +23 -23
  95. package/rust/core/lexer/handler/number.rs +31 -31
  96. package/rust/core/lexer/handler/operator.rs +46 -46
  97. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  98. package/rust/core/lexer/handler/slash.rs +21 -21
  99. package/rust/core/lexer/handler/string.rs +63 -63
  100. package/rust/core/lexer/mod.rs +3 -3
  101. package/rust/core/mod.rs +9 -9
  102. package/rust/core/parser/driver/block.rs +111 -111
  103. package/rust/core/parser/driver/cursor.rs +82 -82
  104. package/rust/core/parser/driver/driver_impl.rs +21 -1
  105. package/rust/core/parser/driver/mod.rs +6 -6
  106. package/rust/core/parser/driver/parse_array.rs +120 -120
  107. package/rust/core/parser/driver/parse_map.rs +247 -223
  108. package/rust/core/parser/driver/parser.rs +160 -160
  109. package/rust/core/parser/handler/arrow_call.rs +65 -14
  110. package/rust/core/parser/handler/identifier/synth.rs +171 -135
  111. package/rust/core/parser/handler/mod.rs +9 -9
  112. package/rust/core/parser/handler/pattern.rs +24 -1
  113. package/rust/core/plugin/loader.rs +137 -137
  114. package/rust/core/plugin/mod.rs +2 -2
  115. package/rust/core/plugin/runner/non_wasm.rs +481 -297
  116. package/rust/core/plugin/runner/wasm32.rs +1 -0
  117. package/rust/core/preprocessor/loader/inject.rs +313 -278
  118. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
  119. package/rust/core/preprocessor/loader/mod.rs +235 -235
  120. package/rust/core/preprocessor/module.rs +55 -55
  121. package/rust/core/preprocessor/processor/handlers.rs +107 -107
  122. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  123. package/rust/core/preprocessor/resolver/call.rs +124 -124
  124. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  125. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  126. package/rust/core/preprocessor/resolver/function.rs +69 -69
  127. package/rust/core/preprocessor/resolver/group.rs +122 -122
  128. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  129. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  130. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  131. package/rust/core/preprocessor/resolver/pattern.rs +95 -83
  132. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  133. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  134. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  135. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  136. package/rust/core/preprocessor/resolver/value.rs +176 -176
  137. package/rust/core/store/global.rs +57 -57
  138. package/rust/lib.rs +323 -323
  139. package/rust/macros/Cargo.toml +14 -0
  140. package/rust/macros/src/lib.rs +52 -0
  141. package/rust/main.rs +311 -142
  142. package/rust/types/Cargo.toml +1 -1
  143. package/rust/types/src/addons.rs +3 -1
  144. package/rust/types/src/config.rs +1 -3
  145. package/rust/utils/Cargo.toml +5 -2
  146. package/rust/utils/src/file.rs +397 -14
  147. package/rust/utils/src/path.rs +31 -2
  148. package/rust/utils/src/version.rs +38 -7
  149. package/rust/web/auth.rs +5 -0
  150. package/rust/web/forge.rs +5 -0
  151. package/rust/web/mod.rs +5 -3
  152. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  153. package/rust/cli/bank/api.rs +0 -122
  154. package/rust/cli/bank/commands.rs +0 -306
  155. package/rust/cli/bank/mod.rs +0 -29
  156. package/rust/cli/install/bank.rs +0 -72
  157. package/rust/cli/install/commands.rs +0 -35
  158. package/rust/cli/install/mod.rs +0 -4
  159. package/rust/cli/install/plugin.rs +0 -80
@@ -1,294 +1,294 @@
1
- use devalang_types::Value;
2
- use std::collections::HashMap;
3
-
4
- pub struct FilterSpec {
5
- pub kind: String,
6
- pub cutoff: f32,
7
- }
8
-
9
- pub struct FilterState {
10
- pub prev_l: f32,
11
- pub prev_r: f32,
12
- pub prev_in_l: f32,
13
- pub prev_in_r: f32,
14
- pub prev_out_l: f32,
15
- pub prev_out_r: f32,
16
- }
17
-
18
- pub struct NoteSetup {
19
- pub sample_rate: f32,
20
- pub channels: usize,
21
- pub total_samples: usize,
22
- pub start_sample: usize,
23
- pub attack_samples: usize,
24
- pub decay_samples: usize,
25
- pub release_samples: usize,
26
- pub sustain_level: f32,
27
- pub pluck_click: f32,
28
- pub pluck_click_samples: usize,
29
- pub drive: f32,
30
- pub filters: Vec<FilterSpec>,
31
- pub filter_states: Vec<FilterState>,
32
- pub lfo_rate: f32,
33
- pub lfo_depth: f32,
34
- pub lfo_target: Option<String>,
35
- pub voices: usize,
36
- pub unison_detune: f32,
37
- pub volume_env: HashMap<String, Value>,
38
- pub pan_env: HashMap<String, Value>,
39
- pub pitch_env: HashMap<String, Value>,
40
- }
41
-
42
- pub fn build_note_setup(
43
- engine: &mut crate::core::audio::engine::AudioEngine,
44
- _waveform: &str,
45
- freq: f32,
46
- amp: f32,
47
- start_time_ms: f32,
48
- mut duration_ms: f32,
49
- synth_params: &HashMap<String, Value>,
50
- note_params: &HashMap<String, Value>,
51
- automation: &Option<HashMap<String, Value>>,
52
- ) -> NoteSetup {
53
- use crate::core::audio::engine::helpers;
54
-
55
- // Extract ADSR and normalize units
56
- let attack = engine.extract_f32(synth_params, "attack").unwrap_or(0.0);
57
- let decay = engine.extract_f32(synth_params, "decay").unwrap_or(0.0);
58
- let sustain = engine.extract_f32(synth_params, "sustain").unwrap_or(1.0);
59
- let release = engine.extract_f32(synth_params, "release").unwrap_or(0.0);
60
- let _sustain_level = if sustain > 1.0 {
61
- (sustain / 100.0).clamp(0.0, 1.0)
62
- } else {
63
- sustain.clamp(0.0, 1.0)
64
- };
65
-
66
- if let Some(g) = engine.extract_f32(note_params, "gate") {
67
- if g > 0.0 && g <= 1.0 {
68
- duration_ms = duration_ms * g;
69
- } else if g > 1.0 {
70
- duration_ms = duration_ms * (g / 100.0);
71
- }
72
- } else if let Some(g) = engine.extract_f32(synth_params, "gate") {
73
- if g > 0.0 && g <= 1.0 {
74
- duration_ms = duration_ms * g;
75
- } else if g > 1.0 {
76
- duration_ms = duration_ms * (g / 100.0);
77
- }
78
- }
79
-
80
- let velocity = engine.extract_f32(note_params, "velocity").unwrap_or(1.0);
81
-
82
- let detune_cents = engine
83
- .extract_f32(note_params, "detune")
84
- .or(engine.extract_f32(synth_params, "detune"))
85
- .unwrap_or(0.0);
86
-
87
- let _lowpass_cut = engine
88
- .extract_f32(note_params, "lowpass")
89
- .or(engine.extract_f32(synth_params, "lowpass"))
90
- .unwrap_or(0.0);
91
-
92
- let _amplitude = (i16::MAX as f32) * amp.clamp(0.0, 1.0) * velocity.clamp(0.0, 1.0);
93
-
94
- let _freq_start = freq;
95
- let mut _freq_end = freq;
96
- let _amp_start = amp * velocity.clamp(0.0, 1.0);
97
- let mut _amp_end = _amp_start;
98
-
99
- let glide = engine
100
- .extract_boolean(note_params, "glide")
101
- .unwrap_or(false);
102
- let slide = engine
103
- .extract_boolean(note_params, "slide")
104
- .unwrap_or(false);
105
- if glide {
106
- if let Some(Value::Number(target_freq)) = note_params.get("target_freq") {
107
- _freq_end = *target_freq;
108
- } else {
109
- _freq_end = freq * 1.5;
110
- }
111
- }
112
- if slide {
113
- if let Some(Value::Number(target_amp)) = note_params.get("target_amp") {
114
- _amp_end = *target_amp * velocity.clamp(0.0, 1.0);
115
- } else {
116
- _amp_end = _amp_start * 0.5;
117
- }
118
- }
119
-
120
- let sample_rate = engine.sample_rate as f32;
121
- let channels = engine.channels as usize;
122
-
123
- let total_samples = ((duration_ms / 1000.0) * sample_rate) as usize;
124
- let start_sample = ((start_time_ms / 1000.0) * sample_rate) as usize;
125
-
126
- // MIDI event
127
- let midi_note_f = 69.0 + 12.0 * (_freq_start / 440.0).log2();
128
- let midi_note = midi_note_f.round().clamp(0.0, 127.0) as u8;
129
- let midi_vel = (velocity.clamp(0.0, 1.0) * 127.0).round().clamp(0.0, 127.0) as u8;
130
- engine
131
- .midi_events
132
- .push(crate::core::audio::engine::driver::MidiNoteEvent {
133
- key: midi_note,
134
- vel: midi_vel,
135
- start_ms: start_time_ms as u32,
136
- duration_ms: duration_ms as u32,
137
- channel: 0,
138
- });
139
-
140
- let _detune_factor = (2.0_f32).powf(detune_cents / 1200.0);
141
-
142
- let (_volume_env, _pan_env, _pitch_env) = helpers::env_maps_from_automation(automation);
143
-
144
- let attack_s = if attack > 10.0 {
145
- attack / 1000.0
146
- } else {
147
- attack
148
- };
149
- let decay_s = if decay > 10.0 { decay / 1000.0 } else { decay };
150
- let release_s = if release > 10.0 {
151
- release / 1000.0
152
- } else {
153
- release
154
- };
155
- let sustain_level = _sustain_level;
156
-
157
- let attack_samples = (attack_s * sample_rate) as usize;
158
- let decay_samples = (decay_s * sample_rate) as usize;
159
- let release_samples = (release_s * sample_rate) as usize;
160
-
161
- // optional pluck click
162
- let pluck_click = engine
163
- .extract_f32(note_params, "pluck_click")
164
- .or(engine.extract_f32(synth_params, "pluck_click"))
165
- .unwrap_or(0.0);
166
- let pluck_click_ms = engine
167
- .extract_f32(note_params, "pluck_click_ms")
168
- .or(engine.extract_f32(synth_params, "pluck_click_ms"))
169
- .unwrap_or(10.0);
170
- let pluck_click_samples = ((pluck_click_ms / 1000.0) * sample_rate) as usize;
171
-
172
- let drive = engine
173
- .extract_f32(note_params, "drive")
174
- .or(engine.extract_f32(synth_params, "drive"))
175
- .unwrap_or(0.0);
176
-
177
- // parse filter specs
178
- let mut raw_filters: Vec<HashMap<String, Value>> = Vec::new();
179
- if let Some(Value::Array(arr)) = synth_params.get("filters") {
180
- for v in arr {
181
- if let Value::Map(m) = v {
182
- raw_filters.push(m.clone());
183
- }
184
- }
185
- }
186
- if let Some(Value::Array(arr)) = note_params.get("filters") {
187
- for v in arr {
188
- if let Value::Map(m) = v {
189
- raw_filters.push(m.clone());
190
- }
191
- }
192
- }
193
-
194
- let mut filters: Vec<FilterSpec> = Vec::new();
195
- let mut filter_states: Vec<FilterState> = Vec::new();
196
- for rf in raw_filters.into_iter() {
197
- let kind = rf
198
- .get("type")
199
- .and_then(|v| match v {
200
- Value::String(s) => Some(s.clone()),
201
- Value::Identifier(s) => Some(s.clone()),
202
- _ => None,
203
- })
204
- .unwrap_or_else(|| "lowpass".to_string());
205
- let cutoff = rf
206
- .get("cutoff")
207
- .and_then(|v| match v {
208
- Value::Number(n) => Some(*n),
209
- Value::String(s) => s.parse::<f32>().ok(),
210
- _ => None,
211
- })
212
- .unwrap_or(1000.0);
213
- filters.push(FilterSpec {
214
- kind: kind.to_lowercase(),
215
- cutoff,
216
- });
217
- filter_states.push(FilterState {
218
- prev_l: 0.0,
219
- prev_r: 0.0,
220
- prev_in_l: 0.0,
221
- prev_in_r: 0.0,
222
- prev_out_l: 0.0,
223
- prev_out_r: 0.0,
224
- });
225
- }
226
-
227
- // LFO parsing (from synth or note) - simplified: prefer note params over synth params
228
- let mut lfo_rate = 0.0f32;
229
- let mut lfo_depth = 0.0f32;
230
- let mut lfo_target: Option<String> = None;
231
- if let Some(Value::Map(m)) = synth_params.get("lfo") {
232
- if let Some(Value::Number(r)) = m.get("rate") {
233
- lfo_rate = *r;
234
- }
235
- if let Some(Value::Number(d)) = m.get("depth") {
236
- lfo_depth = *d;
237
- }
238
- if let Some(Value::String(t)) = m.get("target") {
239
- lfo_target = Some(t.clone());
240
- }
241
- }
242
- if let Some(Value::Map(m)) = note_params.get("lfo") {
243
- if let Some(Value::Number(r)) = m.get("rate") {
244
- lfo_rate = *r;
245
- }
246
- if let Some(Value::Number(d)) = m.get("depth") {
247
- lfo_depth = *d;
248
- }
249
- if let Some(Value::String(t)) = m.get("target") {
250
- lfo_target = Some(t.clone());
251
- }
252
- }
253
-
254
- let voices = engine
255
- .extract_f32(note_params, "voices")
256
- .or(engine.extract_f32(synth_params, "voices"))
257
- .unwrap_or(1.0)
258
- .max(1.0)
259
- .round() as usize;
260
- let unison_detune = engine
261
- .extract_f32(note_params, "unison_detune")
262
- .or(engine.extract_f32(synth_params, "unison_detune"))
263
- .unwrap_or(0.0);
264
-
265
- let (volume_env, pan_env, pitch_env) = (
266
- helpers::env_map_to_hash(&_volume_env),
267
- helpers::env_map_to_hash(&_pan_env),
268
- helpers::env_map_to_hash(&_pitch_env),
269
- );
270
-
271
- NoteSetup {
272
- sample_rate,
273
- channels,
274
- total_samples,
275
- start_sample,
276
- attack_samples,
277
- decay_samples,
278
- release_samples,
279
- sustain_level,
280
- pluck_click,
281
- pluck_click_samples,
282
- drive,
283
- filters,
284
- filter_states,
285
- lfo_rate,
286
- lfo_depth,
287
- lfo_target,
288
- voices,
289
- unison_detune,
290
- volume_env,
291
- pan_env,
292
- pitch_env,
293
- }
294
- }
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ pub struct FilterSpec {
5
+ pub kind: String,
6
+ pub cutoff: f32,
7
+ }
8
+
9
+ pub struct FilterState {
10
+ pub prev_l: f32,
11
+ pub prev_r: f32,
12
+ pub prev_in_l: f32,
13
+ pub prev_in_r: f32,
14
+ pub prev_out_l: f32,
15
+ pub prev_out_r: f32,
16
+ }
17
+
18
+ pub struct NoteSetup {
19
+ pub sample_rate: f32,
20
+ pub channels: usize,
21
+ pub total_samples: usize,
22
+ pub start_sample: usize,
23
+ pub attack_samples: usize,
24
+ pub decay_samples: usize,
25
+ pub release_samples: usize,
26
+ pub sustain_level: f32,
27
+ pub pluck_click: f32,
28
+ pub pluck_click_samples: usize,
29
+ pub drive: f32,
30
+ pub filters: Vec<FilterSpec>,
31
+ pub filter_states: Vec<FilterState>,
32
+ pub lfo_rate: f32,
33
+ pub lfo_depth: f32,
34
+ pub lfo_target: Option<String>,
35
+ pub voices: usize,
36
+ pub unison_detune: f32,
37
+ pub volume_env: HashMap<String, Value>,
38
+ pub pan_env: HashMap<String, Value>,
39
+ pub pitch_env: HashMap<String, Value>,
40
+ }
41
+
42
+ pub fn build_note_setup(
43
+ engine: &mut crate::core::audio::engine::AudioEngine,
44
+ _waveform: &str,
45
+ freq: f32,
46
+ amp: f32,
47
+ start_time_ms: f32,
48
+ mut duration_ms: f32,
49
+ synth_params: &HashMap<String, Value>,
50
+ note_params: &HashMap<String, Value>,
51
+ automation: &Option<HashMap<String, Value>>,
52
+ ) -> NoteSetup {
53
+ use crate::core::audio::engine::helpers;
54
+
55
+ // Extract ADSR and normalize units
56
+ let attack = engine.extract_f32(synth_params, "attack").unwrap_or(0.0);
57
+ let decay = engine.extract_f32(synth_params, "decay").unwrap_or(0.0);
58
+ let sustain = engine.extract_f32(synth_params, "sustain").unwrap_or(1.0);
59
+ let release = engine.extract_f32(synth_params, "release").unwrap_or(0.0);
60
+ let _sustain_level = if sustain > 1.0 {
61
+ (sustain / 100.0).clamp(0.0, 1.0)
62
+ } else {
63
+ sustain.clamp(0.0, 1.0)
64
+ };
65
+
66
+ if let Some(g) = engine.extract_f32(note_params, "gate") {
67
+ if g > 0.0 && g <= 1.0 {
68
+ duration_ms = duration_ms * g;
69
+ } else if g > 1.0 {
70
+ duration_ms = duration_ms * (g / 100.0);
71
+ }
72
+ } else if let Some(g) = engine.extract_f32(synth_params, "gate") {
73
+ if g > 0.0 && g <= 1.0 {
74
+ duration_ms = duration_ms * g;
75
+ } else if g > 1.0 {
76
+ duration_ms = duration_ms * (g / 100.0);
77
+ }
78
+ }
79
+
80
+ let velocity = engine.extract_f32(note_params, "velocity").unwrap_or(1.0);
81
+
82
+ let detune_cents = engine
83
+ .extract_f32(note_params, "detune")
84
+ .or(engine.extract_f32(synth_params, "detune"))
85
+ .unwrap_or(0.0);
86
+
87
+ let _lowpass_cut = engine
88
+ .extract_f32(note_params, "lowpass")
89
+ .or(engine.extract_f32(synth_params, "lowpass"))
90
+ .unwrap_or(0.0);
91
+
92
+ let _amplitude = (i16::MAX as f32) * amp.clamp(0.0, 1.0) * velocity.clamp(0.0, 1.0);
93
+
94
+ let _freq_start = freq;
95
+ let mut _freq_end = freq;
96
+ let _amp_start = amp * velocity.clamp(0.0, 1.0);
97
+ let mut _amp_end = _amp_start;
98
+
99
+ let glide = engine
100
+ .extract_boolean(note_params, "glide")
101
+ .unwrap_or(false);
102
+ let slide = engine
103
+ .extract_boolean(note_params, "slide")
104
+ .unwrap_or(false);
105
+ if glide {
106
+ if let Some(Value::Number(target_freq)) = note_params.get("target_freq") {
107
+ _freq_end = *target_freq;
108
+ } else {
109
+ _freq_end = freq * 1.5;
110
+ }
111
+ }
112
+ if slide {
113
+ if let Some(Value::Number(target_amp)) = note_params.get("target_amp") {
114
+ _amp_end = *target_amp * velocity.clamp(0.0, 1.0);
115
+ } else {
116
+ _amp_end = _amp_start * 0.5;
117
+ }
118
+ }
119
+
120
+ let sample_rate = engine.sample_rate as f32;
121
+ let channels = engine.channels as usize;
122
+
123
+ let total_samples = ((duration_ms / 1000.0) * sample_rate) as usize;
124
+ let start_sample = ((start_time_ms / 1000.0) * sample_rate) as usize;
125
+
126
+ // MIDI event
127
+ let midi_note_f = 69.0 + 12.0 * (_freq_start / 440.0).log2();
128
+ let midi_note = midi_note_f.round().clamp(0.0, 127.0) as u8;
129
+ let midi_vel = (velocity.clamp(0.0, 1.0) * 127.0).round().clamp(0.0, 127.0) as u8;
130
+ engine
131
+ .midi_events
132
+ .push(crate::core::audio::engine::driver::MidiNoteEvent {
133
+ key: midi_note,
134
+ vel: midi_vel,
135
+ start_ms: start_time_ms as u32,
136
+ duration_ms: duration_ms as u32,
137
+ channel: 0,
138
+ });
139
+
140
+ let _detune_factor = (2.0_f32).powf(detune_cents / 1200.0);
141
+
142
+ let (_volume_env, _pan_env, _pitch_env) = helpers::env_maps_from_automation(automation);
143
+
144
+ let attack_s = if attack > 10.0 {
145
+ attack / 1000.0
146
+ } else {
147
+ attack
148
+ };
149
+ let decay_s = if decay > 10.0 { decay / 1000.0 } else { decay };
150
+ let release_s = if release > 10.0 {
151
+ release / 1000.0
152
+ } else {
153
+ release
154
+ };
155
+ let sustain_level = _sustain_level;
156
+
157
+ let attack_samples = (attack_s * sample_rate) as usize;
158
+ let decay_samples = (decay_s * sample_rate) as usize;
159
+ let release_samples = (release_s * sample_rate) as usize;
160
+
161
+ // optional pluck click
162
+ let pluck_click = engine
163
+ .extract_f32(note_params, "pluck_click")
164
+ .or(engine.extract_f32(synth_params, "pluck_click"))
165
+ .unwrap_or(0.0);
166
+ let pluck_click_ms = engine
167
+ .extract_f32(note_params, "pluck_click_ms")
168
+ .or(engine.extract_f32(synth_params, "pluck_click_ms"))
169
+ .unwrap_or(10.0);
170
+ let pluck_click_samples = ((pluck_click_ms / 1000.0) * sample_rate) as usize;
171
+
172
+ let drive = engine
173
+ .extract_f32(note_params, "drive")
174
+ .or(engine.extract_f32(synth_params, "drive"))
175
+ .unwrap_or(0.0);
176
+
177
+ // parse filter specs
178
+ let mut raw_filters: Vec<HashMap<String, Value>> = Vec::new();
179
+ if let Some(Value::Array(arr)) = synth_params.get("filters") {
180
+ for v in arr {
181
+ if let Value::Map(m) = v {
182
+ raw_filters.push(m.clone());
183
+ }
184
+ }
185
+ }
186
+ if let Some(Value::Array(arr)) = note_params.get("filters") {
187
+ for v in arr {
188
+ if let Value::Map(m) = v {
189
+ raw_filters.push(m.clone());
190
+ }
191
+ }
192
+ }
193
+
194
+ let mut filters: Vec<FilterSpec> = Vec::new();
195
+ let mut filter_states: Vec<FilterState> = Vec::new();
196
+ for rf in raw_filters.into_iter() {
197
+ let kind = rf
198
+ .get("type")
199
+ .and_then(|v| match v {
200
+ Value::String(s) => Some(s.clone()),
201
+ Value::Identifier(s) => Some(s.clone()),
202
+ _ => None,
203
+ })
204
+ .unwrap_or_else(|| "lowpass".to_string());
205
+ let cutoff = rf
206
+ .get("cutoff")
207
+ .and_then(|v| match v {
208
+ Value::Number(n) => Some(*n),
209
+ Value::String(s) => s.parse::<f32>().ok(),
210
+ _ => None,
211
+ })
212
+ .unwrap_or(1000.0);
213
+ filters.push(FilterSpec {
214
+ kind: kind.to_lowercase(),
215
+ cutoff,
216
+ });
217
+ filter_states.push(FilterState {
218
+ prev_l: 0.0,
219
+ prev_r: 0.0,
220
+ prev_in_l: 0.0,
221
+ prev_in_r: 0.0,
222
+ prev_out_l: 0.0,
223
+ prev_out_r: 0.0,
224
+ });
225
+ }
226
+
227
+ // LFO parsing (from synth or note) - simplified: prefer note params over synth params
228
+ let mut lfo_rate = 0.0f32;
229
+ let mut lfo_depth = 0.0f32;
230
+ let mut lfo_target: Option<String> = None;
231
+ if let Some(Value::Map(m)) = synth_params.get("lfo") {
232
+ if let Some(Value::Number(r)) = m.get("rate") {
233
+ lfo_rate = *r;
234
+ }
235
+ if let Some(Value::Number(d)) = m.get("depth") {
236
+ lfo_depth = *d;
237
+ }
238
+ if let Some(Value::String(t)) = m.get("target") {
239
+ lfo_target = Some(t.clone());
240
+ }
241
+ }
242
+ if let Some(Value::Map(m)) = note_params.get("lfo") {
243
+ if let Some(Value::Number(r)) = m.get("rate") {
244
+ lfo_rate = *r;
245
+ }
246
+ if let Some(Value::Number(d)) = m.get("depth") {
247
+ lfo_depth = *d;
248
+ }
249
+ if let Some(Value::String(t)) = m.get("target") {
250
+ lfo_target = Some(t.clone());
251
+ }
252
+ }
253
+
254
+ let voices = engine
255
+ .extract_f32(note_params, "voices")
256
+ .or(engine.extract_f32(synth_params, "voices"))
257
+ .unwrap_or(1.0)
258
+ .max(1.0)
259
+ .round() as usize;
260
+ let unison_detune = engine
261
+ .extract_f32(note_params, "unison_detune")
262
+ .or(engine.extract_f32(synth_params, "unison_detune"))
263
+ .unwrap_or(0.0);
264
+
265
+ let (volume_env, pan_env, pitch_env) = (
266
+ helpers::env_map_to_hash(&_volume_env),
267
+ helpers::env_map_to_hash(&_pan_env),
268
+ helpers::env_map_to_hash(&_pitch_env),
269
+ );
270
+
271
+ NoteSetup {
272
+ sample_rate,
273
+ channels,
274
+ total_samples,
275
+ start_sample,
276
+ attack_samples,
277
+ decay_samples,
278
+ release_samples,
279
+ sustain_level,
280
+ pluck_click,
281
+ pluck_click_samples,
282
+ drive,
283
+ filters,
284
+ filter_states,
285
+ lfo_rate,
286
+ lfo_depth,
287
+ lfo_target,
288
+ voices,
289
+ unison_detune,
290
+ volume_env,
291
+ pan_env,
292
+ pitch_env,
293
+ }
294
+ }