@devaloop/devalang 0.0.1-alpha.9 → 0.0.1-beta.1

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