@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,3 @@
1
+ pub mod interprete;
2
+ pub mod methods;
3
+ pub mod types;
@@ -0,0 +1,192 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ pub fn apply_defaults(synth_params: &mut HashMap<String, Value>) {
5
+ use devalang_types::Value as DV;
6
+ synth_params
7
+ .entry("attack".to_string())
8
+ .or_insert(DV::Number(1.0));
9
+ synth_params
10
+ .entry("decay".to_string())
11
+ .or_insert(DV::Number(50.0));
12
+ synth_params
13
+ .entry("sustain".to_string())
14
+ .or_insert(DV::Number(0.0));
15
+ synth_params
16
+ .entry("release".to_string())
17
+ .or_insert(DV::Number(120.0));
18
+ synth_params
19
+ .entry("glide".to_string())
20
+ .or_insert(DV::Boolean(false));
21
+ synth_params
22
+ .entry("slide".to_string())
23
+ .or_insert(DV::Boolean(false));
24
+ // arp defaults
25
+ synth_params
26
+ .entry("gate".to_string())
27
+ .or_insert(DV::Number(80.0));
28
+ synth_params
29
+ .entry("pattern".to_string())
30
+ .or_insert(DV::String("up".to_string()));
31
+ // per-step velocity shaping
32
+ synth_params
33
+ .entry("velocity_scale".to_string())
34
+ .or_insert(DV::Number(0.25));
35
+ }
36
+
37
+ // Helper to compute an arp step and schedule notes; this is small and returns the computed step
38
+ pub fn compute_arp_step(
39
+ duration_ms: f32,
40
+ note_count: usize,
41
+ synth_params: &HashMap<String, Value>,
42
+ ) -> f32 {
43
+ // New API: synth_params may contain "rate" (beats/duration string or number),
44
+ // "pattern" (up, down, updown), and "step" (ms or number)
45
+ use devalang_types::Duration as D;
46
+
47
+ // If rate is provided as number (notes across duration), use it
48
+ if let Some(Value::Number(rate)) = synth_params.get("rate") {
49
+ if *rate > 0.0 {
50
+ return duration_ms / *rate;
51
+ }
52
+ }
53
+
54
+ // If rate provided as Duration/Beat/String parse as musical fraction -> seconds
55
+ if let Some(v) = synth_params.get("rate") {
56
+ match v {
57
+ Value::Duration(d) => match d {
58
+ D::Number(sec) => {
59
+ if *sec > 0.0 {
60
+ return sec * 1000.0;
61
+ }
62
+ }
63
+ D::Beat(_) | D::Identifier(_) => {
64
+ // need bpm; fallback 120
65
+ let bpm = synth_params
66
+ .get("bpm")
67
+ .and_then(|bv| {
68
+ if let Value::Number(n) = bv {
69
+ Some(*n)
70
+ } else {
71
+ None
72
+ }
73
+ })
74
+ .unwrap_or(120.0);
75
+ if let Some(sec) =
76
+ crate::core::audio::engine::driver::duration_to_seconds(d, bpm)
77
+ {
78
+ if sec > 0.0 {
79
+ return sec * 1000.0;
80
+ }
81
+ }
82
+ }
83
+ _ => {}
84
+ },
85
+ Value::String(s) | Value::Identifier(s) => {
86
+ let bpm = synth_params
87
+ .get("bpm")
88
+ .and_then(|bv| {
89
+ if let Value::Number(n) = bv {
90
+ Some(*n)
91
+ } else {
92
+ None
93
+ }
94
+ })
95
+ .unwrap_or(120.0);
96
+ if let Some(sec) =
97
+ crate::core::audio::engine::driver::parse_fraction_to_seconds(s, bpm)
98
+ {
99
+ if sec > 0.0 {
100
+ return sec * 1000.0;
101
+ }
102
+ }
103
+ if let Ok(n) = s.parse::<f32>() {
104
+ if n > 0.0 {
105
+ return duration_ms / n;
106
+ }
107
+ }
108
+ }
109
+ _ => {}
110
+ }
111
+ }
112
+
113
+ // fallback to explicit step param
114
+ let default_step = (duration_ms / (note_count as f32)) * 0.5;
115
+ match synth_params.get("step") {
116
+ Some(Value::Number(n)) => *n,
117
+ Some(Value::String(s)) => s.parse::<f32>().unwrap_or(default_step),
118
+ Some(Value::Identifier(s)) => s.parse::<f32>().unwrap_or(default_step),
119
+ _ => default_step,
120
+ }
121
+ }
122
+
123
+ // Ensure arp defaults include an arp_step if provided as beats / number string
124
+ pub fn ensure_arp_step(synth_params: &mut HashMap<String, Value>, default_step: f32) {
125
+ use devalang_types::Value as DV;
126
+ if !synth_params.contains_key("arp_step") {
127
+ synth_params.insert("arp_step".to_string(), DV::Number(default_step));
128
+ }
129
+ }
130
+
131
+ // Prepare per-note scheduling & params for arp
132
+ pub fn prepare_note(
133
+ note_name: &str,
134
+ index: usize,
135
+ total: usize,
136
+ start_time_ms: f32,
137
+ duration_ms: f32,
138
+ amp: f32,
139
+ synth_params: &HashMap<String, Value>,
140
+ note_params: &HashMap<String, Value>,
141
+ _automation: &Option<HashMap<String, Value>>,
142
+ ) -> (f32, f32, f32, HashMap<String, Value>) {
143
+ // compute a slightly syncopated step: half-step + small swing
144
+ let mut step = compute_arp_step(duration_ms, total, synth_params);
145
+ // introduce simple swing based on index
146
+ let swing = match synth_params.get("swing") {
147
+ Some(Value::Number(n)) => *n,
148
+ _ => 0.05,
149
+ };
150
+ if index % 2 == 1 {
151
+ step *= 1.0 + swing;
152
+ }
153
+ let start = start_time_ms + (index as f32) * step;
154
+ // For now, arp doesn't alter amp or params beyond start offset
155
+ let amp_out = amp;
156
+ let mut params_out = note_params.clone();
157
+ // optionally expose arp-specific params into note params
158
+ if let Some(Value::Number(n)) = synth_params.get("gate") {
159
+ // normalize gate to 0.0-1.0 if >1 (percent)
160
+ let gate_val = if *n > 1.0 { (*n) / 100.0 } else { *n };
161
+ params_out
162
+ .entry("gate".to_string())
163
+ .or_insert(Value::Number(gate_val));
164
+ }
165
+ // simple velocity curve across arp steps (so later steps can be quieter/louder)
166
+ if let Some(Value::Number(vscale)) = synth_params.get("velocity_scale") {
167
+ let mult = 1.0 - ((index as f32) / (total as f32)) * (*vscale);
168
+ params_out.insert("velocity".to_string(), Value::Number(mult.max(0.0)));
169
+ }
170
+ // propagate pattern if present (engine or higher layer may use it)
171
+ if let Some(Value::String(p)) = synth_params.get("pattern") {
172
+ params_out.insert("pattern".to_string(), Value::String(p.clone()));
173
+ }
174
+ (start, note_to_freq(note_name), amp_out, params_out)
175
+ }
176
+
177
+ fn note_to_freq(note: &str) -> f32 {
178
+ let notes = [
179
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
180
+ ];
181
+
182
+ if note.len() < 2 || note.len() > 3 {
183
+ return 440.0;
184
+ }
185
+
186
+ let (name, octave_str) = note.split_at(note.len() - 1);
187
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
188
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
189
+ let midi_note = (octave + 1) * 12 + semitone;
190
+
191
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
192
+ }
@@ -0,0 +1,24 @@
1
+ pub mod arp;
2
+ pub mod pad;
3
+ pub mod pluck;
4
+ pub mod sub;
5
+
6
+ use devalang_types::Value;
7
+ use std::collections::HashMap;
8
+
9
+ pub fn apply_type(synth_params: &mut HashMap<String, Value>) {
10
+ if let Some(tval) = synth_params.get("type") {
11
+ let tname = match tval {
12
+ Value::String(s) => s.as_str(),
13
+ Value::Identifier(s) => s.as_str(),
14
+ _ => "",
15
+ };
16
+ match tname {
17
+ "pad" => pad::apply_defaults(synth_params),
18
+ "pluck" => pluck::apply_defaults(synth_params),
19
+ "arp" => arp::apply_defaults(synth_params),
20
+ "sub" => sub::apply_defaults(synth_params),
21
+ _ => {}
22
+ }
23
+ }
24
+ }
@@ -0,0 +1,116 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ pub fn apply_defaults(synth_params: &mut HashMap<String, Value>) {
5
+ use devalang_types::Value as DV;
6
+ // pad: long attack/release, high sustain (0.0-1.0)
7
+ synth_params
8
+ .entry("attack".to_string())
9
+ .or_insert(DV::Number(600.0));
10
+ synth_params
11
+ .entry("decay".to_string())
12
+ .or_insert(DV::Number(300.0));
13
+ synth_params
14
+ .entry("sustain".to_string())
15
+ .or_insert(DV::Number(0.8));
16
+ synth_params
17
+ .entry("release".to_string())
18
+ .or_insert(DV::Number(900.0));
19
+ // optional unison/voices parameters
20
+ synth_params
21
+ .entry("voices".to_string())
22
+ .or_insert(DV::Number(3.0));
23
+ synth_params
24
+ .entry("unison_detune".to_string())
25
+ .or_insert(DV::Number(15.0));
26
+ // slightly wider default chorus/depth for pads
27
+ if !synth_params.contains_key("effects") {
28
+ let mut ch: std::collections::HashMap<String, Value> = std::collections::HashMap::new();
29
+ ch.insert("type".to_string(), Value::String("chorus".to_string()));
30
+ ch.insert("depth".to_string(), Value::Number(0.12));
31
+ ch.insert("rate".to_string(), Value::Number(0.25));
32
+ synth_params.insert("effects".to_string(), DV::Array(vec![DV::Map(ch)]));
33
+ }
34
+ }
35
+
36
+ // Prepare per-note modifications for pad: longer release/attack, gentle detune per-voice
37
+ pub fn prepare_note(
38
+ _note_name: &str,
39
+ index: usize,
40
+ total: usize,
41
+ _start_time_ms: f32,
42
+ _duration_ms: f32,
43
+ amp: f32,
44
+ synth_params: &HashMap<String, Value>,
45
+ note_params: &HashMap<String, Value>,
46
+ _automation: &Option<HashMap<String, Value>>,
47
+ ) -> (f32, f32, f32, HashMap<String, Value>) {
48
+ let mut params_out = note_params.clone();
49
+ // inject pad envelope defaults if missing
50
+ params_out
51
+ .entry("attack".to_string())
52
+ .or_insert(Value::Number(600.0));
53
+ params_out
54
+ .entry("release".to_string())
55
+ .or_insert(Value::Number(900.0));
56
+ params_out
57
+ .entry("sustain".to_string())
58
+ .or_insert(Value::Number(0.8));
59
+
60
+ // propagate unison/voices into per-note params if synth provided
61
+ if let Some(Value::Number(v)) = synth_params.get("voices") {
62
+ params_out.insert("voices".to_string(), Value::Number(*v));
63
+ }
64
+ if let Some(Value::Number(d)) = synth_params.get("unison_detune") {
65
+ params_out.insert("unison_detune".to_string(), Value::Number(*d));
66
+ }
67
+
68
+ // gentle detune based on index and synth param "detune" if present
69
+ // gentle detune based on synth detune or unison settings
70
+ let detune_base = match synth_params.get("detune") {
71
+ Some(Value::Number(n)) => *n,
72
+ _ => 0.0,
73
+ };
74
+ let mut detune = detune_base * ((index as f32) / (total as f32));
75
+ // if unison used, spread voices around center
76
+ if let Some(Value::Number(_voices)) = synth_params.get("voices") {
77
+ if let Some(Value::Number(ud)) = synth_params.get("unison_detune") {
78
+ // spread by a fraction based on voice index
79
+ detune += (*ud) * (((index as f32) - (total as f32 - 1.0) / 2.0) / (total as f32));
80
+ }
81
+ }
82
+
83
+ params_out.insert("detune".to_string(), Value::Number(detune));
84
+
85
+ // compute frequency with detune in cents
86
+ let freq = note_to_freq(_note_name) * (2.0_f32).powf(detune / 1200.0);
87
+
88
+ // add a subtle chorus placeholder effect if none present (engine may ignore unknown effects)
89
+ use devalang_types::Value as DV;
90
+ if !params_out.contains_key("effects") {
91
+ let mut ch: std::collections::HashMap<String, Value> = std::collections::HashMap::new();
92
+ ch.insert("type".to_string(), Value::String("chorus".to_string()));
93
+ ch.insert("depth".to_string(), Value::Number(0.15));
94
+ ch.insert("rate".to_string(), Value::Number(0.25));
95
+ params_out.insert("effects".to_string(), DV::Array(vec![DV::Map(ch)]));
96
+ }
97
+
98
+ (0.0, freq, amp, params_out)
99
+ }
100
+
101
+ fn note_to_freq(note: &str) -> f32 {
102
+ let notes = [
103
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
104
+ ];
105
+
106
+ if note.len() < 2 || note.len() > 3 {
107
+ return 440.0;
108
+ }
109
+
110
+ let (name, octave_str) = note.split_at(note.len() - 1);
111
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
112
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
113
+ let midi_note = (octave + 1) * 12 + semitone;
114
+
115
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
116
+ }
@@ -0,0 +1,97 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ pub fn apply_defaults(synth_params: &mut HashMap<String, Value>) {
5
+ use devalang_types::Value as DV;
6
+ // pluck: very short attack, short decay, very low sustain, short release
7
+ synth_params
8
+ .entry("attack".to_string())
9
+ .or_insert(DV::Number(2.0));
10
+ synth_params
11
+ .entry("decay".to_string())
12
+ .or_insert(DV::Number(80.0));
13
+ synth_params
14
+ .entry("sustain".to_string())
15
+ .or_insert(DV::Number(0.0));
16
+ synth_params
17
+ .entry("release".to_string())
18
+ .or_insert(DV::Number(120.0));
19
+ synth_params
20
+ .entry("glide".to_string())
21
+ .or_insert(DV::Boolean(false));
22
+ synth_params
23
+ .entry("slide".to_string())
24
+ .or_insert(DV::Boolean(false));
25
+ // small transient helper for plucks
26
+ synth_params
27
+ .entry("pluck_click".to_string())
28
+ .or_insert(DV::Number(0.08));
29
+ synth_params
30
+ .entry("pluck_click_ms".to_string())
31
+ .or_insert(DV::Number(8.0));
32
+ }
33
+
34
+ // Prepare per-note modifications for pluck: short attack/release, optional amp curve per-index
35
+ pub fn prepare_note(
36
+ _note_name: &str,
37
+ index: usize,
38
+ _total: usize,
39
+ _start_time_ms: f32,
40
+ _duration_ms: f32,
41
+ amp: f32,
42
+ _synth_params: &HashMap<String, Value>,
43
+ note_params: &HashMap<String, Value>,
44
+ _automation: &Option<HashMap<String, Value>>,
45
+ ) -> (f32, f32, f32, HashMap<String, Value>) {
46
+ // pluck slightly reduces amp for later notes by a tiny fraction
47
+ let decay = 0.02 * (index as f32);
48
+ let amp_out = (amp * (1.0 - decay)).max(0.0);
49
+ let mut params_out = note_params.clone();
50
+ // ensure short attack & short release to sound plucky
51
+ params_out
52
+ .entry("attack".to_string())
53
+ .or_insert(Value::Number(1.0));
54
+ params_out
55
+ .entry("release".to_string())
56
+ .or_insert(Value::Number(100.0));
57
+ params_out
58
+ .entry("sustain".to_string())
59
+ .or_insert(Value::Number(0.0));
60
+ // add a small "click" transient and a light resonant filter to emphasize the pluck
61
+ use devalang_types::Value as DV;
62
+ let freq = note_to_freq(_note_name);
63
+ let mut filt: std::collections::HashMap<String, Value> = std::collections::HashMap::new();
64
+ filt.insert("type".to_string(), Value::String("bandpass".to_string()));
65
+ // center filter a bit above the note to give a bright pluck transient
66
+ // use more explicit filter parameter names requested by user
67
+ filt.insert(
68
+ "cutoff".to_string(),
69
+ Value::Number((freq * 2.0).min(10000.0)),
70
+ );
71
+ filt.insert("resonance".to_string(), Value::Number(6.0));
72
+ // only insert filters if not already present
73
+ if !params_out.contains_key("filters") {
74
+ params_out.insert("filters".to_string(), DV::Array(vec![DV::Map(filt)]));
75
+ }
76
+ // frequency unchanged here; caller computes freq
77
+ // stronger pitch drop for a plucky character
78
+ let freq_end = freq * 0.98;
79
+ (0.0, freq_end, amp_out, params_out)
80
+ }
81
+
82
+ fn note_to_freq(note: &str) -> f32 {
83
+ let notes = [
84
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
85
+ ];
86
+
87
+ if note.len() < 2 || note.len() > 3 {
88
+ return 440.0;
89
+ }
90
+
91
+ let (name, octave_str) = note.split_at(note.len() - 1);
92
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
93
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
94
+ let midi_note = (octave + 1) * 12 + semitone;
95
+
96
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
97
+ }
@@ -0,0 +1,100 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ pub fn apply_defaults(synth_params: &mut HashMap<String, Value>) {
5
+ use devalang_types::Value as DV;
6
+ // sub bass: strong sustain, lowpass emphasis
7
+ synth_params
8
+ .entry("attack".to_string())
9
+ .or_insert(DV::Number(10.0));
10
+ synth_params
11
+ .entry("decay".to_string())
12
+ .or_insert(DV::Number(80.0));
13
+ // sustain stored as 0.0-1.0 for consistency
14
+ synth_params
15
+ .entry("sustain".to_string())
16
+ .or_insert(DV::Number(1.0));
17
+ synth_params
18
+ .entry("release".to_string())
19
+ .or_insert(DV::Number(300.0));
20
+ synth_params
21
+ .entry("detune".to_string())
22
+ .or_insert(DV::Number(0.0));
23
+ synth_params
24
+ .entry("lowpass".to_string())
25
+ .or_insert(DV::Number(150.0));
26
+ // octave stacking and drive for extra body
27
+ synth_params
28
+ .entry("octaves".to_string())
29
+ .or_insert(DV::Number(1.0));
30
+ synth_params
31
+ .entry("drive".to_string())
32
+ .or_insert(DV::Number(0.0));
33
+ // encourage a lowpass default if none provided
34
+ synth_params
35
+ .entry("lowpass".to_string())
36
+ .or_insert(DV::Number(120.0));
37
+ }
38
+
39
+ // prepare_note for sub: shift note one or two octaves down and set stronger amp
40
+ pub fn prepare_note(
41
+ note_name: &str,
42
+ _index: usize,
43
+ _total: usize,
44
+ _start_time_ms: f32,
45
+ _duration_ms: f32,
46
+ amp: f32,
47
+ _synth_params: &HashMap<String, Value>,
48
+ note_params: &HashMap<String, Value>,
49
+ _automation: &Option<HashMap<String, Value>>,
50
+ ) -> (f32, f32, f32, HashMap<String, Value>) {
51
+ let mut params_out = note_params.clone();
52
+ params_out
53
+ .entry("attack".to_string())
54
+ .or_insert(Value::Number(5.0));
55
+ params_out
56
+ .entry("release".to_string())
57
+ .or_insert(Value::Number(200.0));
58
+ params_out
59
+ .entry("sustain".to_string())
60
+ .or_insert(Value::Number(1.0));
61
+
62
+ // compute deep-frequency one octave down by default
63
+ let freq = note_to_freq(note_name) * 0.5;
64
+ // allow stacking additional octaves for extra body
65
+ if let Some(Value::Number(o)) = _synth_params.get("octaves") {
66
+ let oct = (*o as i32).max(1);
67
+ if oct >= 2 {
68
+ // add lower harmonic by halving frequency again (engine may layer if octaves param is observed)
69
+ // we still return the primary frequency but advertise octaves in params
70
+ params_out.insert("octaves".to_string(), Value::Number(*o));
71
+ }
72
+ }
73
+
74
+ // boost amplitude a little for sub
75
+ let amp_out = (amp * 1.5).min(1.0);
76
+
77
+ // advertise drive if present so engine/effects can apply soft clipping
78
+ if let Some(Value::Number(d)) = _synth_params.get("drive") {
79
+ params_out.insert("drive".to_string(), Value::Number(*d));
80
+ }
81
+
82
+ (0.0, freq, amp_out, params_out)
83
+ }
84
+
85
+ fn note_to_freq(note: &str) -> f32 {
86
+ let notes = [
87
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
88
+ ];
89
+
90
+ if note.len() < 2 || note.len() > 3 {
91
+ return 440.0;
92
+ }
93
+
94
+ let (name, octave_str) = note.split_at(note.len() - 1);
95
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
96
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
97
+ let midi_note = (octave + 1) * 12 + semitone;
98
+
99
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
100
+ }
@@ -1,18 +1,16 @@
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
- }
1
+ use crate::core::parser::statement::{Statement, StatementKind};
2
+ use devalang_types::VariableTable;
3
+
4
+ // Store automation configuration into the variable table under a namespaced key
5
+ // Key: "<target>__automation" => Value::Map({ target, params })
6
+ pub fn interprete_automate_statement(
7
+ stmt: &Statement,
8
+ variable_table: &mut VariableTable,
9
+ ) -> Option<VariableTable> {
10
+ if let StatementKind::Automate { target } = &stmt.kind {
11
+ let key = format!("{}__automation", target);
12
+ variable_table.set(key, stmt.value.clone());
13
+ return Some(variable_table.clone());
14
+ }
15
+ None
16
+ }
@@ -3,8 +3,9 @@ use devalang_types::{Duration, Value};
3
3
  use crate::core::{
4
4
  audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
5
5
  parser::statement::{Statement, StatementKind},
6
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
6
+ store::global::GlobalStore,
7
7
  };
8
+ use devalang_types::store::{FunctionTable, VariableTable};
8
9
 
9
10
  pub fn interprete_call_statement(
10
11
  stmt: &Statement,
@@ -84,9 +85,9 @@ pub fn interprete_call_statement(
84
85
  // Build a variable table snapshot for resolution like triggers do
85
86
  // Preserve the full parent chain so lookups behave the same as runtime
86
87
  fn clone_with_parents(
87
- orig: &crate::core::store::variable::VariableTable,
88
- ) -> crate::core::store::variable::VariableTable {
89
- crate::core::store::variable::VariableTable {
88
+ orig: &devalang_types::VariableTable,
89
+ ) -> devalang_types::VariableTable {
90
+ devalang_types::VariableTable {
90
91
  variables: orig.variables.clone(),
91
92
  parent: orig
92
93
  .parent
@@ -6,8 +6,9 @@ use crate::core::{
6
6
  interpreter::driver::execute_audio_block,
7
7
  },
8
8
  parser::statement::Statement,
9
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
9
+ store::global::GlobalStore,
10
10
  };
11
+ use devalang_types::store::{FunctionTable, VariableTable};
11
12
 
12
13
  pub fn interprete_condition_statement(
13
14
  stmt: &Statement,
@@ -1,7 +1,5 @@
1
- use crate::core::{
2
- parser::statement::{Statement, StatementKind},
3
- store::function::{FunctionDef, FunctionTable},
4
- };
1
+ use crate::core::parser::statement::{Statement, StatementKind};
2
+ use devalang_types::store::{FunctionDef, FunctionTable};
5
3
 
6
4
  pub fn interprete_function_statement(
7
5
  stmt: &Statement,
@@ -1,8 +1,6 @@
1
- use crate::core::{
2
- parser::statement::{Statement, StatementKind},
3
- store::variable::VariableTable,
4
- };
1
+ use crate::core::parser::statement::{Statement, StatementKind};
5
2
  use devalang_types::Value;
3
+ use devalang_types::store::VariableTable;
6
4
 
7
5
  pub fn interprete_let_statement(
8
6
  stmt: &Statement,
@@ -1,9 +1,7 @@
1
1
  use devalang_types::Value;
2
2
 
3
- use crate::core::{
4
- parser::statement::{Statement, StatementKind},
5
- store::variable::VariableTable,
6
- };
3
+ use crate::core::parser::statement::{Statement, StatementKind};
4
+ use devalang_types::store::VariableTable;
7
5
 
8
6
  pub fn interprete_load_statement(
9
7
  stmt: &Statement,
@@ -3,8 +3,9 @@ use devalang_types::Value;
3
3
  use crate::core::{
4
4
  audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
5
5
  parser::statement::Statement,
6
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
6
+ store::global::GlobalStore,
7
7
  };
8
+ use devalang_types::store::{FunctionTable, VariableTable};
8
9
 
9
10
  pub fn interprete_loop_statement(
10
11
  stmt: &Statement,
@@ -0,0 +1,12 @@
1
+ pub mod arrow_call;
2
+ pub mod automate;
3
+ pub mod call;
4
+ pub mod condition;
5
+ pub mod function;
6
+ pub mod let_;
7
+ pub mod load;
8
+ pub mod loop_;
9
+ pub mod sleep;
10
+ pub mod spawn;
11
+ pub mod tempo;
12
+ pub mod trigger;