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

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 (220) hide show
  1. package/.devalang +9 -10
  2. package/Cargo.toml +5 -4
  3. package/README.md +7 -5
  4. package/docs/CHANGELOG.md +42 -0
  5. package/docs/ROADMAP.md +5 -1
  6. package/docs/TODO.md +3 -14
  7. package/examples/bus.deva +10 -0
  8. package/examples/effect.deva +2 -0
  9. package/examples/filter.deva +11 -0
  10. package/examples/lfo.deva +9 -0
  11. package/examples/synth.deva +11 -1
  12. package/examples/synth_types.deva +17 -0
  13. package/out-tsc/core/functions/index.d.ts +5 -0
  14. package/out-tsc/core/functions/index.js +11 -0
  15. package/out-tsc/pkg/devalang_core.d.ts +2 -0
  16. package/out-tsc/pkg/devalang_core.js +17 -2
  17. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -7
  18. package/package.json +1 -1
  19. package/project-version.json +3 -3
  20. package/rust/cli/bank/api.rs +122 -122
  21. package/rust/cli/bank/commands.rs +33 -2
  22. package/rust/cli/bank/mod.rs +29 -29
  23. package/rust/cli/build/commands.rs +53 -3
  24. package/rust/cli/build/mod.rs +2 -2
  25. package/rust/cli/build/process.rs +26 -7
  26. package/rust/cli/check/mod.rs +2 -2
  27. package/rust/cli/discover/commands.rs +253 -253
  28. package/rust/cli/discover/config.rs +111 -111
  29. package/rust/cli/discover/fs.rs +19 -19
  30. package/rust/cli/discover/install.rs +103 -103
  31. package/rust/cli/discover/metadata.rs +48 -48
  32. package/rust/cli/discover/mod.rs +5 -5
  33. package/rust/cli/install/addon.rs +118 -118
  34. package/rust/cli/install/bank.rs +22 -3
  35. package/rust/cli/install/commands.rs +35 -35
  36. package/rust/cli/install/mod.rs +4 -4
  37. package/rust/cli/install/plugin.rs +80 -61
  38. package/rust/cli/login/commands.rs +124 -124
  39. package/rust/cli/mod.rs +12 -12
  40. package/rust/cli/parser.rs +46 -1
  41. package/rust/cli/play/commands.rs +71 -20
  42. package/rust/cli/play/mod.rs +5 -5
  43. package/rust/cli/play/process.rs +14 -5
  44. package/rust/cli/play/realtime.rs +91 -91
  45. package/rust/cli/telemetry/commands.rs +22 -22
  46. package/rust/cli/telemetry/event_creator.rs +80 -80
  47. package/rust/cli/telemetry/mod.rs +3 -3
  48. package/rust/cli/telemetry/send.rs +51 -51
  49. package/rust/cli/template/commands.rs +69 -69
  50. package/rust/config/driver.rs +112 -103
  51. package/rust/config/mod.rs +3 -3
  52. package/rust/config/ops.rs +26 -26
  53. package/rust/config/settings.rs +101 -101
  54. package/rust/core/audio/engine/driver.rs +220 -0
  55. package/rust/core/audio/engine/export.rs +169 -0
  56. package/rust/core/audio/engine/helpers.rs +178 -170
  57. package/rust/core/audio/engine/mod.rs +51 -2
  58. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  59. package/rust/core/audio/engine/notes/mod.rs +44 -0
  60. package/rust/core/audio/engine/notes/params.rs +294 -0
  61. package/rust/core/audio/engine/sample/insert.rs +199 -0
  62. package/rust/core/audio/engine/sample/mod.rs +40 -0
  63. package/rust/core/audio/engine/sample/padding.rs +170 -0
  64. package/rust/core/audio/evaluator/condition.rs +61 -0
  65. package/rust/core/audio/evaluator/mod.rs +9 -0
  66. package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +1 -159
  67. package/rust/core/audio/evaluator/rhs.rs +16 -0
  68. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  69. package/rust/core/audio/interpreter/driver.rs +55 -23
  70. package/rust/core/audio/interpreter/mod.rs +1 -13
  71. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  72. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  73. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  74. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  75. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  76. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  77. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  78. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  79. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  80. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  81. package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +16 -18
  82. package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +5 -4
  83. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +2 -1
  84. package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +2 -4
  85. package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +2 -4
  86. package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +2 -4
  87. package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +2 -1
  88. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  89. package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
  90. package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +3 -2
  91. package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
  92. package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +1 -1
  93. package/rust/core/audio/loader/trigger.rs +2 -1
  94. package/rust/core/audio/mod.rs +6 -7
  95. package/rust/core/audio/player.rs +70 -70
  96. package/rust/core/audio/special/easing.rs +189 -189
  97. package/rust/core/audio/special/env.rs +45 -45
  98. package/rust/core/audio/special/math.rs +134 -134
  99. package/rust/core/audio/special/mod.rs +9 -9
  100. package/rust/core/audio/special/modulator.rs +143 -143
  101. package/rust/core/builder/mod.rs +45 -2
  102. package/rust/core/debugger/lexer.rs +27 -27
  103. package/rust/core/debugger/{module.rs → logs.rs} +3 -6
  104. package/rust/core/debugger/mod.rs +30 -30
  105. package/rust/core/debugger/preprocessor.rs +27 -27
  106. package/rust/core/debugger/store.rs +2 -4
  107. package/rust/core/error/mod.rs +269 -269
  108. package/rust/core/lexer/driver.rs +59 -61
  109. package/rust/core/lexer/handler/arrow.rs +82 -82
  110. package/rust/core/lexer/handler/at.rs +21 -21
  111. package/rust/core/lexer/handler/brace.rs +41 -41
  112. package/rust/core/lexer/handler/colon.rs +21 -21
  113. package/rust/core/lexer/handler/comment.rs +30 -30
  114. package/rust/core/lexer/handler/dot.rs +21 -21
  115. package/rust/core/lexer/handler/driver.rs +337 -337
  116. package/rust/core/lexer/handler/identifier.rs +47 -47
  117. package/rust/core/lexer/handler/indent.rs +66 -66
  118. package/rust/core/lexer/handler/mod.rs +15 -15
  119. package/rust/core/lexer/handler/newline.rs +23 -23
  120. package/rust/core/lexer/handler/number.rs +31 -31
  121. package/rust/core/lexer/handler/operator.rs +46 -46
  122. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  123. package/rust/core/lexer/handler/slash.rs +21 -21
  124. package/rust/core/lexer/handler/string.rs +63 -63
  125. package/rust/core/lexer/mod.rs +3 -3
  126. package/rust/core/mod.rs +0 -1
  127. package/rust/core/parser/driver/block.rs +111 -0
  128. package/rust/core/parser/driver/cursor.rs +82 -0
  129. package/rust/core/parser/driver/driver_impl.rs +139 -0
  130. package/rust/core/parser/driver/mod.rs +6 -0
  131. package/rust/core/parser/driver/parse_array.rs +120 -0
  132. package/rust/core/parser/driver/parse_map.rs +223 -0
  133. package/rust/core/parser/driver/parser.rs +160 -0
  134. package/rust/core/parser/handler/arrow_call.rs +28 -4
  135. package/rust/core/parser/handler/at.rs +279 -279
  136. package/rust/core/parser/handler/bank.rs +104 -104
  137. package/rust/core/parser/handler/condition.rs +83 -83
  138. package/rust/core/parser/handler/dot.rs +148 -148
  139. package/rust/core/parser/handler/identifier/automate.rs +254 -254
  140. package/rust/core/parser/handler/identifier/call.rs +91 -91
  141. package/rust/core/parser/handler/identifier/emit.rs +70 -70
  142. package/rust/core/parser/handler/identifier/function.rs +113 -113
  143. package/rust/core/parser/handler/identifier/group.rs +89 -89
  144. package/rust/core/parser/handler/identifier/let_.rs +173 -173
  145. package/rust/core/parser/handler/identifier/mod.rs +55 -55
  146. package/rust/core/parser/handler/identifier/on.rs +107 -107
  147. package/rust/core/parser/handler/identifier/print.rs +49 -49
  148. package/rust/core/parser/handler/identifier/sleep.rs +96 -43
  149. package/rust/core/parser/handler/identifier/spawn.rs +91 -91
  150. package/rust/core/parser/handler/identifier/synth.rs +135 -135
  151. package/rust/core/parser/handler/loop_.rs +194 -194
  152. package/rust/core/parser/handler/mod.rs +9 -9
  153. package/rust/core/parser/handler/pattern.rs +1 -1
  154. package/rust/core/parser/handler/tempo.rs +105 -57
  155. package/rust/core/parser/statement.rs +10 -11
  156. package/rust/core/plugin/loader.rs +1 -1
  157. package/rust/core/plugin/mod.rs +2 -2
  158. package/rust/core/plugin/runner/mod.rs +11 -0
  159. package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +297 -347
  160. package/rust/core/plugin/runner/wasm32.rs +43 -0
  161. package/rust/core/preprocessor/loader/inject.rs +278 -0
  162. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  163. package/rust/core/preprocessor/loader/mod.rs +235 -0
  164. package/rust/core/preprocessor/module.rs +2 -7
  165. package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +6 -13
  166. package/rust/core/preprocessor/processor/mod.rs +1 -0
  167. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  168. package/rust/core/preprocessor/resolver/call.rs +124 -124
  169. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  170. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  171. package/rust/core/preprocessor/resolver/function.rs +2 -2
  172. package/rust/core/preprocessor/resolver/group.rs +46 -18
  173. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  174. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  175. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  176. package/rust/core/preprocessor/resolver/pattern.rs +83 -83
  177. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  178. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  179. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  180. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  181. package/rust/core/preprocessor/resolver/value.rs +176 -176
  182. package/rust/core/store/global.rs +2 -6
  183. package/rust/core/store/mod.rs +1 -5
  184. package/rust/lib.rs +18 -3
  185. package/rust/main.rs +27 -3
  186. package/rust/types/Cargo.toml +1 -1
  187. package/rust/types/src/addons.rs +55 -55
  188. package/rust/types/src/config.rs +84 -74
  189. package/rust/types/src/lib.rs +15 -12
  190. package/rust/types/src/plugin.rs +20 -0
  191. package/rust/types/src/store.rs +139 -0
  192. package/rust/types/src/telemetry.rs +85 -85
  193. package/rust/utils/Cargo.toml +2 -2
  194. package/rust/utils/src/file.rs +94 -94
  195. package/rust/utils/src/first_usage.rs +97 -97
  196. package/rust/utils/src/lib.rs +9 -9
  197. package/rust/utils/src/logger.rs +200 -200
  198. package/rust/utils/src/path.rs +129 -88
  199. package/rust/utils/src/signature.rs +41 -41
  200. package/rust/utils/src/spinner.rs +20 -20
  201. package/rust/utils/src/version.rs +27 -27
  202. package/rust/utils/src/watcher.rs +46 -46
  203. package/rust/web/api.rs +5 -5
  204. package/rust/web/cdn.rs +34 -34
  205. package/rust/web/mod.rs +3 -3
  206. package/tests/integration.rs +21 -21
  207. package/typescript/core/functions/index.ts +11 -0
  208. package/typescript/pkg/devalang_core.ts +20 -4
  209. package/rust/core/audio/engine/sample.rs +0 -366
  210. package/rust/core/audio/engine/synth.rs +0 -325
  211. package/rust/core/audio/interpreter/arrow_call.rs +0 -311
  212. package/rust/core/audio/renderer.rs +0 -54
  213. package/rust/core/parser/driver.rs +0 -584
  214. package/rust/core/preprocessor/loader.rs +0 -637
  215. package/rust/core/store/export.rs +0 -28
  216. package/rust/core/store/function.rs +0 -40
  217. package/rust/core/store/import.rs +0 -28
  218. package/rust/core/store/variable.rs +0 -51
  219. package/rust/core/utils/mod.rs +0 -1
  220. package/rust/core/utils/path.rs +0 -37
@@ -0,0 +1,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
+ 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
+ }
@@ -0,0 +1,2 @@
1
+ pub mod chord;
2
+ pub mod note;
@@ -0,0 +1,316 @@
1
+ use crate::core::{audio::engine::AudioEngine, plugin::runner::WasmPluginRunner};
2
+ use devalang_types::Value;
3
+ use devalang_utils::logger::{LogLevel, Logger};
4
+ use std::collections::HashMap;
5
+
6
+ pub fn interprete_note_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
+ let filtered_args: Vec<_> = args
23
+ .iter()
24
+ .filter(|arg| !matches!(arg, Value::Unknown))
25
+ .collect();
26
+
27
+ let Some(Value::Identifier(note_name)) = filtered_args.first().map(|v| (*v).clone()) else {
28
+ println!(
29
+ "❌ Invalid or missing argument for 'note' method on '{}'.",
30
+ target
31
+ );
32
+ return (*max_end_time, cursor_copy);
33
+ };
34
+
35
+ let mut note_params = HashMap::new();
36
+ if let Some(arg1) = filtered_args.get(1) {
37
+ if let Value::Map(map) = (*arg1).clone() {
38
+ for (key, value) in map {
39
+ note_params.insert(key, value);
40
+ }
41
+ }
42
+ }
43
+
44
+ // Note parameters and calculations
45
+ let amp_note = extract_f32(&note_params, "amp", base_bpm).unwrap_or(amp);
46
+ let duration_ms =
47
+ extract_f32(&note_params, "duration", base_bpm).unwrap_or(base_duration * 1000.0);
48
+
49
+ let duration_secs = duration_ms / 1000.0;
50
+ let final_freq = note_to_freq(&note_name);
51
+ let start_time = cursor_copy;
52
+ let end_time = start_time + duration_secs;
53
+
54
+ // Fetch automation map if present:
55
+ // - Global (per-synth): key "<target>__automation" => map with key "params"
56
+ // - Per-note: note parameter "automate" => map
57
+ let auto_key = format!("{}__automation", target);
58
+ let synth_automation = match variable_table.get(&auto_key) {
59
+ Some(Value::Map(map)) => match map.get("params") {
60
+ Some(Value::Map(p)) => Some(p.clone()),
61
+ _ => None,
62
+ },
63
+ _ => None,
64
+ };
65
+
66
+ let note_automation = match note_params.get("automate") {
67
+ Some(Value::Map(m)) => Some(m.clone()),
68
+ _ => None,
69
+ };
70
+
71
+ // Merge: per-note overrides synth automation per key (volume/pan/pitch)
72
+ let automation = match (synth_automation, note_automation) {
73
+ (Some(mut a), Some(n)) => {
74
+ for (k, v) in n {
75
+ a.insert(k, v);
76
+ }
77
+ Some(a)
78
+ }
79
+ (None, Some(n)) => Some(n),
80
+ (Some(a), None) => Some(a),
81
+ _ => None,
82
+ };
83
+
84
+ // If waveform references a plugin alias (e.g., alias.synth), use the WASM plugin runner
85
+ if waveform_str.contains('.') && waveform_str.ends_with(".synth") {
86
+ let alias = waveform_str.split('.').next().unwrap_or("");
87
+ if let Some(Value::String(uri)) = variable_table.get(alias) {
88
+ if let Some(id) = uri.strip_prefix("devalang://plugin/") {
89
+ let mut parts = id.split('.');
90
+ let author = parts.next().unwrap_or("");
91
+ let name = parts.next().unwrap_or("");
92
+ let key = format!("{}:{}", author, name);
93
+ if let Some((_info, wasm_bytes)) = global_store.plugins.get(&key) {
94
+ // Prepare buffer (stereo f32)
95
+ let sample_rate = 44100.0_f32;
96
+ let total_samples = ((duration_ms / 1000.0) * sample_rate) as usize;
97
+ let channels = 2usize;
98
+ let start_index = ((start_time * sample_rate) as usize) * channels;
99
+ let required_len = start_index + total_samples * channels;
100
+ if audio_engine.buffer.len() < required_len {
101
+ audio_engine.buffer.resize(required_len, 0);
102
+ }
103
+ let mut fbuf = vec![0.0f32; total_samples * channels];
104
+ let runner = WasmPluginRunner::new();
105
+ let mut params_num: std::collections::HashMap<String, f32> =
106
+ std::collections::HashMap::new();
107
+ let mut params_str: std::collections::HashMap<String, String> =
108
+ std::collections::HashMap::new();
109
+ for (k, v) in synth_params.iter() {
110
+ match v {
111
+ Value::Number(n) => {
112
+ params_num.insert(k.clone(), *n);
113
+ }
114
+ Value::String(s) => {
115
+ params_str.insert(k.clone(), s.clone());
116
+ }
117
+ Value::Identifier(s) => {
118
+ params_str.insert(k.clone(), s.clone());
119
+ }
120
+ _ => {}
121
+ }
122
+ }
123
+ let _ = runner.render_note_with_params_in_place(
124
+ wasm_bytes,
125
+ &mut fbuf,
126
+ None,
127
+ final_freq,
128
+ amp_note,
129
+ duration_ms as i32,
130
+ 44100,
131
+ 2,
132
+ &params_num,
133
+ Some(&params_str),
134
+ );
135
+ for (i, sample) in fbuf.iter().enumerate().take(total_samples * channels) {
136
+ let s = (sample.clamp(-1.0, 1.0) * (i16::MAX as f32)) as i16;
137
+ let idx = start_index + i;
138
+ audio_engine.buffer[idx] = audio_engine.buffer[idx].saturating_add(s);
139
+ }
140
+ } else {
141
+ let logger = Logger::new();
142
+ logger.log_message(
143
+ LogLevel::Warning,
144
+ &format!(
145
+ "Plugin bytes not found for key '{}' (alias '{}').",
146
+ key, alias
147
+ ),
148
+ );
149
+ }
150
+ } else {
151
+ let logger = Logger::new();
152
+ logger.log_message(
153
+ LogLevel::Warning,
154
+ &format!("Invalid plugin URI in alias '{}': {}", alias, uri),
155
+ );
156
+ }
157
+ } else {
158
+ let logger = Logger::new();
159
+ logger.log_message(
160
+ LogLevel::Warning,
161
+ &format!("Plugin alias '{}' not found in variable table.", alias),
162
+ );
163
+ }
164
+ } else {
165
+ // Allow types to adjust per-note scheduling/params
166
+ let start_ms = start_time * 1000.0;
167
+ let mut final_amp = amp_note;
168
+ let mut final_note_params = note_params.clone();
169
+
170
+ let mut handled = false;
171
+ if let Some(tval) = synth_params.get("type") {
172
+ let tname = match tval {
173
+ Value::String(s) => s.as_str(),
174
+ Value::Identifier(s) => s.as_str(),
175
+ _ => "",
176
+ };
177
+ match tname {
178
+ "arp" => {
179
+ // compute a step (ms) from synth params (rate/step). compute_arp_step
180
+ // will interpret `rate` as number of notes across the provided duration
181
+ let step_ms = crate::core::audio::interpreter::statements::arrow_call::types::arp::
182
+ compute_arp_step(duration_ms, 1, &synth_params);
183
+ let steps = if step_ms > 0.0 {
184
+ ((duration_ms / step_ms).ceil() as usize).max(1)
185
+ } else {
186
+ 1usize
187
+ };
188
+
189
+ // For each arp step, call prepare_note to get per-step params and schedule it
190
+ for idx in 0..steps {
191
+ let (start_abs_ms, freq_step, amp_out, params_out) =
192
+ crate::core::audio::interpreter::statements::arrow_call::types::arp::prepare_note(
193
+ &note_name,
194
+ idx,
195
+ steps,
196
+ start_ms,
197
+ duration_ms,
198
+ amp_note,
199
+ &synth_params,
200
+ &final_note_params,
201
+ &automation,
202
+ );
203
+
204
+ // sub-note duration: default to step_ms so arp steps are audible and sequenced
205
+ let sub_duration_ms = if step_ms > 0.0 { step_ms } else { duration_ms };
206
+
207
+ audio_engine.insert_note(
208
+ waveform_str.to_string(),
209
+ freq_step,
210
+ amp_out,
211
+ start_abs_ms,
212
+ sub_duration_ms,
213
+ synth_params.clone(),
214
+ params_out.clone(),
215
+ automation.clone(),
216
+ );
217
+ }
218
+
219
+ // mark handled to avoid the unconditional insert below
220
+ handled = true;
221
+ }
222
+ "pluck" => {
223
+ let (_s, _f, amp_out, params_out) =
224
+ crate::core::audio::interpreter::statements::arrow_call::types::pluck::prepare_note(
225
+ &note_name,
226
+ 0,
227
+ 1,
228
+ start_ms,
229
+ duration_ms,
230
+ amp_note,
231
+ &synth_params,
232
+ &final_note_params,
233
+ &automation,
234
+ );
235
+ final_amp = amp_out;
236
+ final_note_params = params_out;
237
+ }
238
+ "pad" => {
239
+ let (_s, _f, amp_out, params_out) =
240
+ crate::core::audio::interpreter::statements::arrow_call::types::pad::prepare_note(
241
+ &note_name,
242
+ 0,
243
+ 1,
244
+ start_ms,
245
+ duration_ms,
246
+ amp_note,
247
+ &synth_params,
248
+ &final_note_params,
249
+ &automation,
250
+ );
251
+ final_amp = amp_out;
252
+ final_note_params = params_out;
253
+ }
254
+ _ => {}
255
+ }
256
+ }
257
+
258
+ if !handled {
259
+ audio_engine.insert_note(
260
+ waveform_str.to_string(),
261
+ final_freq,
262
+ final_amp,
263
+ start_ms,
264
+ duration_ms,
265
+ synth_params.clone(),
266
+ final_note_params,
267
+ automation,
268
+ );
269
+ }
270
+ }
271
+
272
+ *max_end_time = (*max_end_time).max(end_time);
273
+
274
+ if update_cursor {
275
+ if let Some(c) = cursor_time.as_mut() {
276
+ **c = end_time;
277
+ }
278
+ }
279
+
280
+ return (*max_end_time, end_time);
281
+ }
282
+
283
+ fn extract_f32(map: &HashMap<String, Value>, key: &str, base_bpm: f32) -> Option<f32> {
284
+ map.get(key).and_then(|v| match v {
285
+ Value::Number(n) => Some(*n),
286
+ Value::Beat(beat_str) => {
287
+ let parts: Vec<&str> = beat_str.split('/').collect();
288
+ if parts.len() == 2 {
289
+ let numerator = parts[0].parse::<f32>().ok()?;
290
+ let denominator = parts[1].parse::<f32>().ok()?;
291
+
292
+ Some((numerator / denominator) * ((60.0 / base_bpm) * 1000.0))
293
+ } else {
294
+ None
295
+ }
296
+ }
297
+ _ => None,
298
+ })
299
+ }
300
+
301
+ fn note_to_freq(note: &str) -> f32 {
302
+ let notes = [
303
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
304
+ ];
305
+
306
+ if note.len() < 2 || note.len() > 3 {
307
+ return 440.0;
308
+ }
309
+
310
+ let (name, octave_str) = note.split_at(note.len() - 1);
311
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
312
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
313
+ let midi_note = (octave + 1) * 12 + semitone;
314
+
315
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
316
+ }