@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,384 +1,398 @@
1
- use crate::core::audio::engine::AudioEngine;
2
- use devalang_types::Value;
3
- use devalang_utils::logger::{LogLevel, Logger};
4
- use std::collections::HashMap;
5
-
6
- pub fn interprete_chord_method(
7
- args: &Vec<devalang_types::Value>,
8
- target: &str,
9
- audio_engine: &mut AudioEngine,
10
- variable_table: &devalang_types::VariableTable,
11
- _global_store: &crate::core::store::global::GlobalStore,
12
- waveform_str: &str,
13
- synth_params: &HashMap<String, devalang_types::Value>,
14
- amp: f32,
15
- base_bpm: f32,
16
- base_duration: f32,
17
- max_end_time: &mut f32,
18
- mut cursor_time: Option<&mut f32>,
19
- cursor_copy: f32,
20
- update_cursor: bool,
21
- ) -> (f32, f32) {
22
- // Filter unknown args
23
- let filtered_args: Vec<_> = args
24
- .iter()
25
- .filter(|arg| !matches!(arg, Value::Unknown))
26
- .collect();
27
-
28
- // Expect at least one note identifier, up to 4 notes, optionally last arg is a map of params
29
- if filtered_args.is_empty() {
30
- let logger = Logger::new();
31
- logger.log_message(
32
- LogLevel::Error,
33
- &format!("Invalid or missing arguments for 'chord' on '{}'.", target),
34
- );
35
- return (*max_end_time, cursor_copy);
36
- }
37
-
38
- // Collect note names (first N args that are Identifier or String)
39
- let mut note_names: Vec<String> = Vec::new();
40
- let mut note_params: HashMap<String, Value> = HashMap::new();
41
-
42
- for (_i, arg) in filtered_args.iter().enumerate().take(5) {
43
- match (*arg).clone() {
44
- Value::Identifier(s) | Value::String(s) => {
45
- if note_names.len() < 4 {
46
- note_names.push(s);
47
- }
48
- }
49
- Value::Map(m) => {
50
- // treat as chord-level params (duration, glide, etc.)
51
- for (k, v) in m {
52
- note_params.insert(k, v);
53
- }
54
- }
55
- _ => {}
56
- }
57
- }
58
-
59
- if note_names.is_empty() {
60
- let logger = Logger::new();
61
- logger.log_message(
62
- LogLevel::Error,
63
- &format!("No valid notes found for 'chord' on '{}'.", target),
64
- );
65
- return (*max_end_time, cursor_copy);
66
- }
67
-
68
- // duration & amp for chord
69
- let amp_note = extract_f32(&note_params, "amp", base_bpm).unwrap_or(amp);
70
- let duration_ms =
71
- extract_f32(&note_params, "duration", base_bpm).unwrap_or(base_duration * 1000.0);
72
- let duration_secs = duration_ms / 1000.0;
73
-
74
- let start_time = cursor_copy;
75
- let end_time = start_time + duration_secs;
76
-
77
- // Expand shorthand chord notation like C#min, Dmaj, Amin7 into individual note names
78
- note_names = expand_chord_shorthands(note_names);
79
-
80
- // Prepare automation merge similar to note method
81
- let auto_key = format!("{}__automation", target);
82
- let synth_automation = match variable_table.get(&auto_key) {
83
- Some(Value::Map(map)) => match map.get("params") {
84
- Some(Value::Map(p)) => Some(p.clone()),
85
- _ => None,
86
- },
87
- _ => None,
88
- };
89
-
90
- let chord_automation = match note_params.get("automate") {
91
- Some(Value::Map(m)) => Some(m.clone()),
92
- _ => None,
93
- };
94
-
95
- let automation = match (synth_automation, chord_automation) {
96
- (Some(mut a), Some(n)) => {
97
- for (k, v) in n {
98
- a.insert(k, v);
99
- }
100
- Some(a)
101
- }
102
- (None, Some(n)) => Some(n),
103
- (Some(a), None) => Some(a),
104
- _ => None,
105
- };
106
-
107
- // For each note, let the selected type compute per-note scheduling and parameters
108
- for (i, note_name) in note_names.iter().enumerate() {
109
- // default: start at chord start
110
- let start_ms_default = start_time * 1000.0;
111
- let mut note_amp = amp_note;
112
- let mut note_params_clone = note_params.clone();
113
- let mut note_start_ms = start_ms_default;
114
- let mut note_freq = note_to_freq(note_name);
115
-
116
- if let Some(tval) = synth_params.get("type") {
117
- let tname = match tval {
118
- Value::String(s) => s.as_str(),
119
- Value::Identifier(s) => s.as_str(),
120
- _ => "",
121
- };
122
- match tname {
123
- "arp" => {
124
- let (s_ms, _f, amp_out, params_out) =
125
- crate::core::audio::interpreter::statements::arrow_call::types::arp::prepare_note(
126
- note_name,
127
- i,
128
- note_names.len(),
129
- start_ms_default,
130
- duration_ms,
131
- amp_note,
132
- &synth_params,
133
- &note_params_clone,
134
- &automation,
135
- );
136
- note_start_ms = s_ms;
137
- note_freq = _f;
138
- note_amp = amp_out;
139
- note_params_clone = params_out;
140
- }
141
- "pluck" => {
142
- let (s_ms, _f, amp_out, params_out) =
143
- crate::core::audio::interpreter::statements::arrow_call::types::pluck::prepare_note(
144
- note_name,
145
- i,
146
- note_names.len(),
147
- start_ms_default,
148
- duration_ms,
149
- amp_note,
150
- &synth_params,
151
- &note_params_clone,
152
- &automation,
153
- );
154
- // pluck.prepare_note returns start offset (s_ms) relative to default; if zero use default
155
- if s_ms > 0.0 {
156
- note_start_ms = s_ms
157
- }
158
- note_amp = amp_out;
159
- note_params_clone = params_out;
160
- }
161
- "pad" => {
162
- let (s_ms, _f, amp_out, params_out) =
163
- crate::core::audio::interpreter::statements::arrow_call::types::pad::prepare_note(
164
- note_name,
165
- i,
166
- note_names.len(),
167
- start_ms_default,
168
- duration_ms,
169
- amp_note,
170
- &synth_params,
171
- &note_params_clone,
172
- &automation,
173
- );
174
- if s_ms > 0.0 {
175
- note_start_ms = s_ms
176
- }
177
- note_amp = amp_out;
178
- note_params_clone = params_out;
179
- }
180
- _ => {}
181
- }
182
- }
183
-
184
- audio_engine.insert_note(
185
- waveform_str.to_string(),
186
- note_freq,
187
- note_amp,
188
- note_start_ms,
189
- duration_ms,
190
- synth_params.clone(),
191
- note_params_clone,
192
- automation.clone(),
193
- );
194
-
195
- let note_end = (note_start_ms / 1000.0) + (duration_ms / 1000.0);
196
- *max_end_time = (*max_end_time).max(note_end);
197
- }
198
-
199
- *max_end_time = (*max_end_time).max(end_time);
200
-
201
- if update_cursor {
202
- if let Some(c) = cursor_time.as_mut() {
203
- **c = end_time;
204
- }
205
- }
206
-
207
- (*max_end_time, end_time)
208
- }
209
-
210
- fn note_to_freq(note: &str) -> f32 {
211
- let notes = [
212
- "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
213
- ];
214
-
215
- if note.len() < 2 || note.len() > 3 {
216
- return 440.0;
217
- }
218
-
219
- let (name, octave_str) = note.split_at(note.len() - 1);
220
- let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
221
- let octave = octave_str.parse::<i32>().unwrap_or(4);
222
- let midi_note = (octave + 1) * 12 + semitone;
223
-
224
- 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
225
- }
226
-
227
- fn extract_f32(map: &HashMap<String, Value>, key: &str, base_bpm: f32) -> Option<f32> {
228
- map.get(key).and_then(|v| match v {
229
- Value::Number(n) => Some(*n),
230
- Value::Beat(beat_str) => {
231
- let parts: Vec<&str> = beat_str.split('/').collect();
232
- if parts.len() == 2 {
233
- let numerator = parts[0].parse::<f32>().ok()?;
234
- let denominator = parts[1].parse::<f32>().ok()?;
235
-
236
- Some((numerator / denominator) * ((60.0 / base_bpm) * 1000.0))
237
- } else {
238
- None
239
- }
240
- }
241
- _ => None,
242
- })
243
- }
244
-
245
- // Expand shorthand chords into constituent note names (strings with octave if applicable)
246
- fn expand_chord_shorthands(names: Vec<String>) -> Vec<String> {
247
- let mut out: Vec<String> = Vec::new();
248
- for name in names.into_iter() {
249
- if let Some(parts) = parse_chord_shorthand(&name) {
250
- // parts contains (root_semitone, octave, chord_type)
251
- let (root_semitone, octave, chord_type) = parts;
252
- let root_midi = (octave + 1) * 12 + root_semitone as i32;
253
- match chord_type.as_str() {
254
- "min" | "m" => {
255
- let third = root_midi + 3;
256
- let fifth = root_midi + 7;
257
- out.push(midi_to_note(root_midi));
258
- out.push(midi_to_note(third));
259
- out.push(midi_to_note(fifth));
260
- }
261
- "7" => {
262
- let third = root_midi + 4;
263
- let fifth = root_midi + 7;
264
- let seventh = root_midi + 10;
265
- out.push(midi_to_note(root_midi));
266
- out.push(midi_to_note(third));
267
- out.push(midi_to_note(fifth));
268
- out.push(midi_to_note(seventh));
269
- }
270
- _ => {
271
- // default to major triad
272
- let third = root_midi + 4;
273
- let fifth = root_midi + 7;
274
- out.push(midi_to_note(root_midi));
275
- out.push(midi_to_note(third));
276
- out.push(midi_to_note(fifth));
277
- }
278
- }
279
- } else {
280
- out.push(name);
281
- }
282
- }
283
- out
284
- }
285
-
286
- fn parse_chord_shorthand(s: &str) -> Option<(u8, i32, String)> {
287
- // examples: C#min, Amin, Dmaj7, Ebm
288
- let s = s.trim();
289
- if s.is_empty() {
290
- return None;
291
- }
292
- let mut chars = s.chars().peekable();
293
- let first = chars.next()?;
294
- let letter = first.to_ascii_uppercase();
295
- if !matches!(letter, 'A'..='G') {
296
- return None;
297
- }
298
- let mut name = String::new();
299
- name.push(letter);
300
- // accidental
301
- if let Some(&c) = chars.peek() {
302
- if c == '#' || c == 'b' {
303
- name.push(c);
304
- chars.next();
305
- }
306
- }
307
-
308
- let rest: String = chars.collect();
309
- // extract trailing digits as octave
310
- let mut octave: i32 = 4; // default
311
- let mut type_part = rest.as_str();
312
- // if ends with digit(s)
313
- if !rest.is_empty() {
314
- let mut digits = String::new();
315
- for ch in rest.chars().rev() {
316
- if ch.is_ascii_digit() {
317
- digits.insert(0, ch);
318
- } else {
319
- break;
320
- }
321
- }
322
- if !digits.is_empty() {
323
- if let Ok(o) = digits.parse::<i32>() {
324
- octave = o;
325
- // remove digits from end
326
- let split_at = rest.len() - digits.len();
327
- type_part = &rest[..split_at];
328
- }
329
- }
330
- }
331
-
332
- let chord_type = type_part.to_ascii_lowercase();
333
- // compute semitone index
334
- let semitone = match name.as_str() {
335
- "C" => 0,
336
- "C#" => 1,
337
- "DB" => 1,
338
- "D" => 2,
339
- "D#" => 3,
340
- "EB" => 3,
341
- "E" => 4,
342
- "F" => 5,
343
- "F#" => 6,
344
- "GB" => 6,
345
- "G" => 7,
346
- "G#" => 8,
347
- "AB" => 8,
348
- "A" => 9,
349
- "A#" => 10,
350
- "BB" => 10,
351
- "B" => 11,
352
- _ => return None,
353
- };
354
-
355
- // normalize chord_type: if empty => major
356
- let chord_type = if chord_type.is_empty() {
357
- "maj".to_string()
358
- } else {
359
- chord_type
360
- };
361
-
362
- Some((semitone as u8, octave, chord_type))
363
- }
364
-
365
- fn midi_to_note(m: i32) -> String {
366
- let semitone = (m % 12 + 12) % 12;
367
- let octave = (m / 12) - 1;
368
- let name = match semitone {
369
- 0 => "C",
370
- 1 => "C#",
371
- 2 => "D",
372
- 3 => "D#",
373
- 4 => "E",
374
- 5 => "F",
375
- 6 => "F#",
376
- 7 => "G",
377
- 8 => "G#",
378
- 9 => "A",
379
- 10 => "A#",
380
- 11 => "B",
381
- _ => "C",
382
- };
383
- format!("{}{}", name, octave)
384
- }
1
+ use crate::core::audio::engine::AudioEngine;
2
+ use devalang_types::Value;
3
+ use devalang_utils::logger::{LogLevel, Logger};
4
+ use std::collections::HashMap;
5
+
6
+ pub fn interprete_chord_method(
7
+ args: &Vec<devalang_types::Value>,
8
+ target: &str,
9
+ audio_engine: &mut AudioEngine,
10
+ variable_table: &devalang_types::VariableTable,
11
+ _global_store: &crate::core::store::global::GlobalStore,
12
+ waveform_str: &str,
13
+ synth_params: &HashMap<String, devalang_types::Value>,
14
+ amp: f32,
15
+ base_bpm: f32,
16
+ base_duration: f32,
17
+ max_end_time: &mut f32,
18
+ mut cursor_time: Option<&mut f32>,
19
+ cursor_copy: f32,
20
+ update_cursor: bool,
21
+ ) -> (f32, f32) {
22
+ // Filter unknown args
23
+ let filtered_args: Vec<_> = args
24
+ .iter()
25
+ .filter(|arg| !matches!(arg, Value::Unknown))
26
+ .collect();
27
+
28
+ // Expect at least one note identifier, up to 4 notes, optionally last arg is a map of params
29
+ if filtered_args.is_empty() {
30
+ let logger = Logger::new();
31
+ logger.log_message(
32
+ LogLevel::Error,
33
+ &format!("Invalid or missing arguments for 'chord' on '{}'.", target),
34
+ );
35
+ return (*max_end_time, cursor_copy);
36
+ }
37
+
38
+ // Collect note names (first N args that are Identifier or String)
39
+ let mut note_names: Vec<String> = Vec::new();
40
+ let mut note_params: HashMap<String, Value> = HashMap::new();
41
+
42
+ for (_i, arg) in filtered_args.iter().enumerate().take(5) {
43
+ match (*arg).clone() {
44
+ Value::Identifier(s) | Value::String(s) => {
45
+ if note_names.len() < 4 {
46
+ note_names.push(s);
47
+ }
48
+ }
49
+ Value::Map(m) => {
50
+ // treat as chord-level params (duration, glide, etc.)
51
+ for (k, v) in m {
52
+ note_params.insert(k, v);
53
+ }
54
+ }
55
+ _ => {}
56
+ }
57
+ }
58
+
59
+ if note_names.is_empty() {
60
+ let logger = Logger::new();
61
+ logger.log_message(
62
+ LogLevel::Error,
63
+ &format!("No valid notes found for 'chord' on '{}'.", target),
64
+ );
65
+ return (*max_end_time, cursor_copy);
66
+ }
67
+
68
+ // duration & amp for chord
69
+ let amp_note = extract_f32(&note_params, "amp", base_bpm).unwrap_or(amp);
70
+ let duration_ms =
71
+ extract_f32(&note_params, "duration", base_bpm).unwrap_or(base_duration * 1000.0);
72
+ let duration_secs = duration_ms / 1000.0;
73
+
74
+ let start_time = cursor_copy;
75
+ let end_time = start_time + duration_secs;
76
+
77
+ // Expand shorthand chord notation like C#min, Dmaj, Amin7 into individual note names
78
+ note_names = expand_chord_shorthands(note_names);
79
+
80
+ // Prepare automation merge similar to note method
81
+ let auto_key = format!("{}__automation", target);
82
+ let synth_automation = match variable_table.get(&auto_key) {
83
+ Some(Value::Map(map)) => match map.get("params") {
84
+ Some(Value::Map(p)) => Some(p.clone()),
85
+ _ => None,
86
+ },
87
+ _ => None,
88
+ };
89
+
90
+ let chord_automation = match note_params.get("automate") {
91
+ Some(Value::Map(m)) => Some(m.clone()),
92
+ _ => None,
93
+ };
94
+
95
+ let automation = match (synth_automation, chord_automation) {
96
+ (Some(mut a), Some(n)) => {
97
+ for (k, v) in n {
98
+ a.insert(k, v);
99
+ }
100
+ Some(a)
101
+ }
102
+ (None, Some(n)) => Some(n),
103
+ (Some(a), None) => Some(a),
104
+ _ => None,
105
+ };
106
+
107
+ // For each note, let the selected type compute per-note scheduling and parameters
108
+ for (i, note_name) in note_names.iter().enumerate() {
109
+ // default: start at chord start
110
+ let start_ms_default = start_time * 1000.0;
111
+ let mut note_amp = amp_note;
112
+ let mut note_params_clone = note_params.clone();
113
+ let mut note_start_ms = start_ms_default;
114
+ let mut note_freq = note_to_freq(note_name);
115
+
116
+ if let Some(tval) = synth_params.get("type") {
117
+ let tname = match tval {
118
+ Value::String(s) => s.as_str(),
119
+ Value::Identifier(s) => s.as_str(),
120
+ _ => "",
121
+ };
122
+ match tname {
123
+ "arp" => {
124
+ let (s_ms, _f, amp_out, params_out) =
125
+ crate::core::audio::interpreter::statements::arrow_call::types::arp::prepare_note(
126
+ note_name,
127
+ i,
128
+ note_names.len(),
129
+ start_ms_default,
130
+ duration_ms,
131
+ amp_note,
132
+ &synth_params,
133
+ &note_params_clone,
134
+ &automation,
135
+ );
136
+ note_start_ms = s_ms;
137
+ note_freq = _f;
138
+ note_amp = amp_out;
139
+ note_params_clone = params_out;
140
+ }
141
+ "pluck" => {
142
+ let (s_ms, _f, amp_out, params_out) =
143
+ crate::core::audio::interpreter::statements::arrow_call::types::pluck::prepare_note(
144
+ note_name,
145
+ i,
146
+ note_names.len(),
147
+ start_ms_default,
148
+ duration_ms,
149
+ amp_note,
150
+ &synth_params,
151
+ &note_params_clone,
152
+ &automation,
153
+ );
154
+ // pluck.prepare_note returns start offset (s_ms) relative to default; if zero use default
155
+ if s_ms > 0.0 {
156
+ note_start_ms = s_ms
157
+ }
158
+ note_amp = amp_out;
159
+ note_params_clone = params_out;
160
+ }
161
+ "pad" => {
162
+ let (s_ms, _f, amp_out, params_out) =
163
+ crate::core::audio::interpreter::statements::arrow_call::types::pad::prepare_note(
164
+ note_name,
165
+ i,
166
+ note_names.len(),
167
+ start_ms_default,
168
+ duration_ms,
169
+ amp_note,
170
+ &synth_params,
171
+ &note_params_clone,
172
+ &automation,
173
+ );
174
+ if s_ms > 0.0 {
175
+ note_start_ms = s_ms
176
+ }
177
+ note_amp = amp_out;
178
+ note_params_clone = params_out;
179
+ }
180
+ _ => {}
181
+ }
182
+ }
183
+
184
+ let ranges = audio_engine.insert_note(
185
+ Some(target.to_string()),
186
+ waveform_str.to_string(),
187
+ note_freq,
188
+ note_amp,
189
+ note_start_ms,
190
+ duration_ms,
191
+ synth_params.clone(),
192
+ note_params_clone.clone(),
193
+ automation.clone(),
194
+ );
195
+
196
+ // Apply simple per-note effects if provided under note_params_clone.effects (map)
197
+ if let Some(Value::Map(eff_map)) = note_params_clone.get("effects") {
198
+ for (_start, _len) in ranges.iter() {
199
+ crate::core::audio::interpreter::statements::arrow_call::methods::effects::apply_effect_chain(
200
+ "echo",
201
+ &vec![Value::Map(eff_map.clone())],
202
+ target,
203
+ audio_engine,
204
+ variable_table,
205
+ );
206
+ }
207
+ }
208
+
209
+ let note_end = (note_start_ms / 1000.0) + (duration_ms / 1000.0);
210
+ *max_end_time = (*max_end_time).max(note_end);
211
+ }
212
+
213
+ *max_end_time = (*max_end_time).max(end_time);
214
+
215
+ if update_cursor {
216
+ if let Some(c) = cursor_time.as_mut() {
217
+ **c = end_time;
218
+ }
219
+ }
220
+
221
+ (*max_end_time, end_time)
222
+ }
223
+
224
+ fn note_to_freq(note: &str) -> f32 {
225
+ let notes = [
226
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
227
+ ];
228
+
229
+ if note.len() < 2 || note.len() > 3 {
230
+ return 440.0;
231
+ }
232
+
233
+ let (name, octave_str) = note.split_at(note.len() - 1);
234
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
235
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
236
+ let midi_note = (octave + 1) * 12 + semitone;
237
+
238
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
239
+ }
240
+
241
+ fn extract_f32(map: &HashMap<String, Value>, key: &str, base_bpm: f32) -> Option<f32> {
242
+ map.get(key).and_then(|v| match v {
243
+ Value::Number(n) => Some(*n),
244
+ Value::Beat(beat_str) => {
245
+ let parts: Vec<&str> = beat_str.split('/').collect();
246
+ if parts.len() == 2 {
247
+ let numerator = parts[0].parse::<f32>().ok()?;
248
+ let denominator = parts[1].parse::<f32>().ok()?;
249
+
250
+ Some((numerator / denominator) * ((60.0 / base_bpm) * 1000.0))
251
+ } else {
252
+ None
253
+ }
254
+ }
255
+ _ => None,
256
+ })
257
+ }
258
+
259
+ // Expand shorthand chords into constituent note names (strings with octave if applicable)
260
+ fn expand_chord_shorthands(names: Vec<String>) -> Vec<String> {
261
+ let mut out: Vec<String> = Vec::new();
262
+ for name in names.into_iter() {
263
+ if let Some(parts) = parse_chord_shorthand(&name) {
264
+ // parts contains (root_semitone, octave, chord_type)
265
+ let (root_semitone, octave, chord_type) = parts;
266
+ let root_midi = (octave + 1) * 12 + root_semitone as i32;
267
+ match chord_type.as_str() {
268
+ "min" | "m" => {
269
+ let third = root_midi + 3;
270
+ let fifth = root_midi + 7;
271
+ out.push(midi_to_note(root_midi));
272
+ out.push(midi_to_note(third));
273
+ out.push(midi_to_note(fifth));
274
+ }
275
+ "7" => {
276
+ let third = root_midi + 4;
277
+ let fifth = root_midi + 7;
278
+ let seventh = root_midi + 10;
279
+ out.push(midi_to_note(root_midi));
280
+ out.push(midi_to_note(third));
281
+ out.push(midi_to_note(fifth));
282
+ out.push(midi_to_note(seventh));
283
+ }
284
+ _ => {
285
+ // default to major triad
286
+ let third = root_midi + 4;
287
+ let fifth = root_midi + 7;
288
+ out.push(midi_to_note(root_midi));
289
+ out.push(midi_to_note(third));
290
+ out.push(midi_to_note(fifth));
291
+ }
292
+ }
293
+ } else {
294
+ out.push(name);
295
+ }
296
+ }
297
+ out
298
+ }
299
+
300
+ fn parse_chord_shorthand(s: &str) -> Option<(u8, i32, String)> {
301
+ // examples: C#min, Amin, Dmaj7, Ebm
302
+ let s = s.trim();
303
+ if s.is_empty() {
304
+ return None;
305
+ }
306
+ let mut chars = s.chars().peekable();
307
+ let first = chars.next()?;
308
+ let letter = first.to_ascii_uppercase();
309
+ if !matches!(letter, 'A'..='G') {
310
+ return None;
311
+ }
312
+ let mut name = String::new();
313
+ name.push(letter);
314
+ // accidental
315
+ if let Some(&c) = chars.peek() {
316
+ if c == '#' || c == 'b' {
317
+ name.push(c);
318
+ chars.next();
319
+ }
320
+ }
321
+
322
+ let rest: String = chars.collect();
323
+ // extract trailing digits as octave
324
+ let mut octave: i32 = 4; // default
325
+ let mut type_part = rest.as_str();
326
+ // if ends with digit(s)
327
+ if !rest.is_empty() {
328
+ let mut digits = String::new();
329
+ for ch in rest.chars().rev() {
330
+ if ch.is_ascii_digit() {
331
+ digits.insert(0, ch);
332
+ } else {
333
+ break;
334
+ }
335
+ }
336
+ if !digits.is_empty() {
337
+ if let Ok(o) = digits.parse::<i32>() {
338
+ octave = o;
339
+ // remove digits from end
340
+ let split_at = rest.len() - digits.len();
341
+ type_part = &rest[..split_at];
342
+ }
343
+ }
344
+ }
345
+
346
+ let chord_type = type_part.to_ascii_lowercase();
347
+ // compute semitone index
348
+ let semitone = match name.as_str() {
349
+ "C" => 0,
350
+ "C#" => 1,
351
+ "DB" => 1,
352
+ "D" => 2,
353
+ "D#" => 3,
354
+ "EB" => 3,
355
+ "E" => 4,
356
+ "F" => 5,
357
+ "F#" => 6,
358
+ "GB" => 6,
359
+ "G" => 7,
360
+ "G#" => 8,
361
+ "AB" => 8,
362
+ "A" => 9,
363
+ "A#" => 10,
364
+ "BB" => 10,
365
+ "B" => 11,
366
+ _ => return None,
367
+ };
368
+
369
+ // normalize chord_type: if empty => major
370
+ let chord_type = if chord_type.is_empty() {
371
+ "maj".to_string()
372
+ } else {
373
+ chord_type
374
+ };
375
+
376
+ Some((semitone as u8, octave, chord_type))
377
+ }
378
+
379
+ fn midi_to_note(m: i32) -> String {
380
+ let semitone = (m % 12 + 12) % 12;
381
+ let octave = (m / 12) - 1;
382
+ let name = match semitone {
383
+ 0 => "C",
384
+ 1 => "C#",
385
+ 2 => "D",
386
+ 3 => "D#",
387
+ 4 => "E",
388
+ 5 => "F",
389
+ 6 => "F#",
390
+ 7 => "G",
391
+ 8 => "G#",
392
+ 9 => "A",
393
+ 10 => "A#",
394
+ 11 => "B",
395
+ _ => "C",
396
+ };
397
+ format!("{}{}", name, octave)
398
+ }