@devaloop/devalang 0.0.1-alpha.16-hotfix.3 → 0.0.1-alpha.18

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 (239) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +10 -10
  3. package/.github/workflows/ci.yml +0 -1
  4. package/Cargo.toml +18 -2
  5. package/README.md +82 -34
  6. package/docs/CHANGELOG.md +91 -0
  7. package/docs/ROADMAP.md +7 -4
  8. package/docs/TODO.md +1 -1
  9. package/examples/index.deva +55 -35
  10. package/examples/pattern.deva +5 -5
  11. package/out-tsc/bin/index.d.ts +2 -0
  12. package/out-tsc/core/functions/index.d.ts +37 -0
  13. package/out-tsc/core/functions/index.js +76 -0
  14. package/out-tsc/core/index.d.ts +6 -0
  15. package/out-tsc/core/index.js +22 -0
  16. package/out-tsc/core/types/index.d.ts +4 -0
  17. package/out-tsc/core/types/index.js +20 -0
  18. package/out-tsc/core/types/plugin.d.ts +18 -0
  19. package/out-tsc/core/types/plugin.js +2 -0
  20. package/out-tsc/core/types/result.d.ts +27 -0
  21. package/out-tsc/core/types/result.js +2 -0
  22. package/out-tsc/core/types/statement.d.ts +106 -0
  23. package/out-tsc/core/types/statement.js +2 -0
  24. package/out-tsc/core/types/value.d.ts +43 -0
  25. package/out-tsc/core/types/value.js +2 -0
  26. package/out-tsc/index.d.ts +7 -0
  27. package/out-tsc/index.js +41 -2
  28. package/out-tsc/pkg/devalang_core.d.ts +7 -0
  29. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  30. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  31. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  32. package/out-tsc/scripts/postinstall.d.ts +1 -0
  33. package/out-tsc/scripts/postinstall.js +33 -23
  34. package/out-tsc/scripts/version/bump.d.ts +1 -0
  35. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  36. package/out-tsc/scripts/version/index.d.ts +1 -0
  37. package/out-tsc/scripts/version/sync.d.ts +1 -0
  38. package/package.json +16 -4
  39. package/project-version.json +3 -3
  40. package/rust/cli/bank/api.rs +122 -0
  41. package/rust/cli/bank/commands.rs +275 -0
  42. package/rust/cli/bank/mod.rs +29 -0
  43. package/rust/cli/build/commands.rs +107 -0
  44. package/rust/cli/build/mod.rs +2 -0
  45. package/rust/cli/build/process.rs +146 -0
  46. package/rust/cli/{check.rs → check/mod.rs} +18 -31
  47. package/rust/cli/discover/commands.rs +253 -0
  48. package/rust/cli/discover/config.rs +111 -0
  49. package/rust/cli/discover/fs.rs +19 -0
  50. package/rust/cli/discover/install.rs +103 -0
  51. package/rust/cli/discover/metadata.rs +48 -0
  52. package/rust/cli/discover/mod.rs +5 -0
  53. package/rust/cli/{init.rs → init/commands.rs} +88 -87
  54. package/rust/cli/init/mod.rs +1 -0
  55. package/rust/cli/install/addon.rs +126 -0
  56. package/rust/cli/install/bank.rs +53 -0
  57. package/rust/cli/{install.rs → install/commands.rs} +9 -9
  58. package/rust/{installer → cli/install}/mod.rs +2 -3
  59. package/rust/cli/install/plugin.rs +61 -0
  60. package/rust/cli/{login.rs → login/commands.rs} +8 -11
  61. package/rust/cli/login/mod.rs +1 -0
  62. package/rust/cli/mod.rs +2 -2
  63. package/rust/cli/{driver.rs → parser.rs} +7 -2
  64. package/rust/cli/play/commands.rs +324 -0
  65. package/rust/cli/play/io.rs +17 -0
  66. package/rust/cli/play/mod.rs +5 -0
  67. package/rust/cli/play/process.rs +150 -0
  68. package/rust/cli/play/realtime.rs +91 -0
  69. package/rust/cli/play/utils.rs +23 -0
  70. package/rust/cli/{telemetry.rs → telemetry/commands.rs} +4 -4
  71. package/rust/cli/telemetry/event_creator.rs +80 -0
  72. package/rust/cli/telemetry/mod.rs +3 -0
  73. package/rust/cli/telemetry/send.rs +51 -0
  74. package/rust/cli/{template.rs → template/commands.rs} +1 -1
  75. package/rust/cli/template/mod.rs +1 -0
  76. package/rust/cli/{update.rs → update/commands.rs} +6 -6
  77. package/rust/cli/update/mod.rs +1 -0
  78. package/rust/config/driver.rs +57 -72
  79. package/rust/config/mod.rs +1 -2
  80. package/rust/config/ops.rs +26 -0
  81. package/rust/config/settings.rs +40 -42
  82. package/rust/core/audio/engine/helpers.rs +158 -0
  83. package/rust/core/audio/engine/mod.rs +7 -0
  84. package/rust/core/audio/engine/sample.rs +359 -0
  85. package/rust/core/audio/engine/synth.rs +325 -0
  86. package/rust/core/audio/evaluator.rs +68 -27
  87. package/rust/core/audio/interpreter/arrow_call.rs +113 -33
  88. package/rust/core/audio/interpreter/call.rs +232 -56
  89. package/rust/core/audio/interpreter/condition.rs +3 -2
  90. package/rust/core/audio/interpreter/driver.rs +206 -151
  91. package/rust/core/audio/interpreter/let_.rs +1 -1
  92. package/rust/core/audio/interpreter/load.rs +2 -1
  93. package/rust/core/audio/interpreter/loop_.rs +7 -6
  94. package/rust/core/audio/interpreter/sleep.rs +2 -1
  95. package/rust/core/audio/interpreter/spawn.rs +186 -54
  96. package/rust/core/audio/interpreter/tempo.rs +31 -10
  97. package/rust/core/audio/interpreter/trigger.rs +2 -2
  98. package/rust/core/audio/loader/trigger.rs +4 -7
  99. package/rust/core/audio/player.rs +6 -0
  100. package/rust/core/audio/renderer.rs +5 -7
  101. package/rust/core/audio/special/env.rs +3 -1
  102. package/rust/core/audio/special/math.rs +26 -6
  103. package/rust/core/audio/special/modulator.rs +2 -2
  104. package/rust/core/builder/mod.rs +9 -3
  105. package/rust/core/debugger/lexer.rs +1 -1
  106. package/rust/core/debugger/mod.rs +6 -0
  107. package/rust/core/debugger/module.rs +4 -4
  108. package/rust/core/debugger/preprocessor.rs +1 -1
  109. package/rust/core/debugger/store.rs +2 -2
  110. package/rust/core/error/mod.rs +189 -0
  111. package/rust/core/lexer/driver.rs +61 -0
  112. package/rust/core/lexer/handler/arrow.rs +1 -1
  113. package/rust/core/lexer/handler/at.rs +1 -1
  114. package/rust/core/lexer/handler/brace.rs +2 -2
  115. package/rust/core/lexer/handler/colon.rs +1 -1
  116. package/rust/core/lexer/handler/comment.rs +1 -1
  117. package/rust/core/lexer/handler/dot.rs +1 -1
  118. package/rust/core/lexer/handler/driver.rs +1 -1
  119. package/rust/core/lexer/handler/identifier.rs +4 -3
  120. package/rust/core/lexer/handler/mod.rs +1 -2
  121. package/rust/core/lexer/handler/number.rs +1 -1
  122. package/rust/core/lexer/handler/operator.rs +1 -1
  123. package/rust/core/lexer/handler/parenthesis.rs +2 -2
  124. package/rust/core/lexer/handler/slash.rs +1 -1
  125. package/rust/core/lexer/handler/string.rs +1 -1
  126. package/rust/core/lexer/mod.rs +1 -52
  127. package/rust/core/lexer/token.rs +91 -97
  128. package/rust/core/mod.rs +0 -1
  129. package/rust/core/parser/driver.rs +78 -22
  130. package/rust/core/parser/handler/arrow_call.rs +28 -8
  131. package/rust/core/parser/handler/at.rs +55 -21
  132. package/rust/core/parser/handler/bank.rs +14 -4
  133. package/rust/core/parser/handler/condition.rs +6 -3
  134. package/rust/core/parser/handler/dot.rs +5 -3
  135. package/rust/core/parser/handler/identifier/automate.rs +13 -16
  136. package/rust/core/parser/handler/identifier/call.rs +4 -4
  137. package/rust/core/parser/handler/identifier/emit.rs +9 -5
  138. package/rust/core/parser/handler/identifier/function.rs +20 -7
  139. package/rust/core/parser/handler/identifier/group.rs +11 -7
  140. package/rust/core/parser/handler/identifier/let_.rs +24 -9
  141. package/rust/core/parser/handler/identifier/mod.rs +6 -5
  142. package/rust/core/parser/handler/identifier/on.rs +16 -7
  143. package/rust/core/parser/handler/identifier/print.rs +6 -9
  144. package/rust/core/parser/handler/identifier/sleep.rs +12 -5
  145. package/rust/core/parser/handler/identifier/spawn.rs +4 -4
  146. package/rust/core/parser/handler/identifier/synth.rs +79 -9
  147. package/rust/core/parser/handler/loop_.rs +38 -13
  148. package/rust/core/parser/handler/mod.rs +1 -0
  149. package/rust/core/parser/handler/pattern.rs +74 -0
  150. package/rust/core/parser/handler/tempo.rs +9 -5
  151. package/rust/core/parser/mod.rs +0 -1
  152. package/rust/core/parser/statement.rs +6 -137
  153. package/rust/core/plugin/loader.rs +41 -27
  154. package/rust/core/plugin/runner.rs +68 -17
  155. package/rust/core/preprocessor/loader.rs +181 -99
  156. package/rust/core/preprocessor/processor.rs +9 -9
  157. package/rust/core/preprocessor/resolver/bank.rs +6 -8
  158. package/rust/core/preprocessor/resolver/call.rs +47 -23
  159. package/rust/core/preprocessor/resolver/condition.rs +6 -8
  160. package/rust/core/preprocessor/resolver/driver.rs +28 -28
  161. package/rust/core/preprocessor/resolver/function.rs +6 -6
  162. package/rust/core/preprocessor/resolver/group.rs +6 -8
  163. package/rust/core/preprocessor/resolver/loop_.rs +8 -10
  164. package/rust/core/preprocessor/resolver/mod.rs +1 -0
  165. package/rust/core/preprocessor/resolver/pattern.rs +75 -0
  166. package/rust/core/preprocessor/resolver/spawn.rs +45 -22
  167. package/rust/core/preprocessor/resolver/synth.rs +6 -8
  168. package/rust/core/preprocessor/resolver/tempo.rs +6 -8
  169. package/rust/core/preprocessor/resolver/trigger.rs +22 -19
  170. package/rust/core/preprocessor/resolver/value.rs +99 -4
  171. package/rust/core/store/export.rs +28 -28
  172. package/rust/core/store/function.rs +6 -0
  173. package/rust/core/store/global.rs +7 -1
  174. package/rust/core/store/import.rs +28 -28
  175. package/rust/core/store/variable.rs +16 -2
  176. package/rust/core/utils/mod.rs +0 -1
  177. package/rust/lib.rs +102 -9
  178. package/rust/main.rs +159 -45
  179. package/rust/types/Cargo.toml +11 -0
  180. package/rust/types/src/addons.rs +55 -0
  181. package/rust/types/src/ast.rs +202 -0
  182. package/rust/types/src/config.rs +74 -0
  183. package/rust/types/src/lib.rs +12 -0
  184. package/rust/types/src/telemetry.rs +85 -0
  185. package/rust/utils/Cargo.toml +26 -0
  186. package/rust/utils/{error.rs → src/error.rs} +186 -200
  187. package/rust/utils/src/file.rs +94 -0
  188. package/rust/utils/src/first_usage.rs +97 -0
  189. package/rust/utils/{mod.rs → src/lib.rs} +1 -1
  190. package/rust/utils/{logger.rs → src/logger.rs} +17 -12
  191. package/rust/utils/src/path.rs +88 -0
  192. package/rust/utils/src/signature.rs +41 -0
  193. package/rust/utils/{spinner.rs → src/spinner.rs} +3 -5
  194. package/rust/utils/src/version.rs +27 -0
  195. package/rust/utils/{watcher.rs → src/watcher.rs} +13 -1
  196. package/rust/web/cdn.rs +34 -0
  197. package/templates/minimal/README.md +98 -54
  198. package/templates/welcome/README.md +98 -54
  199. package/templates/welcome/src/index.deva +56 -8
  200. package/templates/welcome/src/variables.deva +2 -4
  201. package/tests/rust/TODO.md +0 -0
  202. package/tests/typescript/index.spec.ts +136 -0
  203. package/tests/typescript/playhead.spec.ts +36 -0
  204. package/tests/typescript/render_e2e.spec.ts +77 -0
  205. package/tsconfig.json +1 -1
  206. package/typescript/core/functions/index.ts +83 -0
  207. package/typescript/core/index.ts +6 -0
  208. package/typescript/core/types/index.ts +4 -0
  209. package/typescript/core/types/plugin.ts +19 -0
  210. package/typescript/core/types/result.ts +29 -0
  211. package/typescript/core/types/statement.ts +47 -0
  212. package/typescript/core/types/value.ts +29 -0
  213. package/typescript/index.ts +7 -2
  214. package/typescript/pkg/devalang_core.d.ts +4 -0
  215. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  216. package/rust/cli/bank.rs +0 -462
  217. package/rust/cli/build.rs +0 -252
  218. package/rust/cli/play.rs +0 -1123
  219. package/rust/common/cdn.rs +0 -5
  220. package/rust/config/loader.rs +0 -165
  221. package/rust/config/stats.rs +0 -257
  222. package/rust/core/audio/engine.rs +0 -696
  223. package/rust/core/shared/bank.rs +0 -21
  224. package/rust/core/shared/duration.rs +0 -9
  225. package/rust/core/shared/mod.rs +0 -3
  226. package/rust/core/shared/value.rs +0 -35
  227. package/rust/core/utils/validation.rs +0 -35
  228. package/rust/installer/addon.rs +0 -84
  229. package/rust/installer/bank.rs +0 -62
  230. package/rust/installer/plugin.rs +0 -54
  231. package/rust/installer/utils.rs +0 -56
  232. package/rust/utils/file.rs +0 -38
  233. package/rust/utils/first_usage.rs +0 -83
  234. package/rust/utils/signature.rs +0 -19
  235. package/rust/utils/telemetry.rs +0 -292
  236. package/rust/utils/version.rs +0 -15
  237. /package/rust/{common → web}/api.rs +0 -0
  238. /package/rust/{common → web}/mod.rs +0 -0
  239. /package/rust/{common → web}/sso.rs +0 -0
@@ -0,0 +1,359 @@
1
+ use crate::core::{store::variable::VariableTable, utils::path::normalize_path};
2
+ use devalang_types::Value;
3
+ use rodio::{Decoder, Source};
4
+ use std::{collections::HashMap, fs::File, io::BufReader, path::Path};
5
+
6
+ const SAMPLE_RATE: u32 = 44100;
7
+ const CHANNELS: u16 = 2;
8
+
9
+ impl super::synth::AudioEngine {
10
+ pub fn insert_sample(
11
+ &mut self,
12
+ filepath: &str,
13
+ time_secs: f32,
14
+ dur_sec: f32,
15
+ effects: Option<HashMap<String, Value>>,
16
+ variable_table: &VariableTable,
17
+ ) {
18
+ if filepath.is_empty() {
19
+ eprintln!("❌ Empty file path provided for audio sample.");
20
+ return;
21
+ }
22
+
23
+ let module_root = Path::new(&self.module_name);
24
+ let root = match devalang_utils::path::get_project_root() {
25
+ Ok(p) => p,
26
+ Err(_) => std::env::current_dir().unwrap_or_else(|_| std::path::PathBuf::from(".")),
27
+ };
28
+ let resolved_path: String;
29
+
30
+ let mut var_path = filepath.to_string();
31
+ if let Some(Value::String(variable_path)) = variable_table.variables.get(filepath) {
32
+ var_path = variable_path.clone();
33
+ } else if let Some(Value::Sample(sample_path)) = variable_table.variables.get(filepath) {
34
+ var_path = sample_path.clone();
35
+ }
36
+
37
+ if var_path.starts_with("devalang://") {
38
+ let path_after_protocol = var_path.replace("devalang://", "");
39
+ let parts: Vec<&str> = path_after_protocol.split('/').collect();
40
+
41
+ if parts.len() < 3 {
42
+ eprintln!(
43
+ "❌ Invalid devalang:// path format. Expected devalang://<type>/<author>.<bank>/<entity>"
44
+ );
45
+ return;
46
+ }
47
+
48
+ let obj_type = parts[0];
49
+ let bank_name = parts[1];
50
+ // Rejoin the remainder as the entity path so bank entries can contain
51
+ // nested paths like "subdir/sample.wav" or plain names.
52
+ let entity_name = parts[2..].join("/");
53
+
54
+ let deva_dir = match devalang_utils::path::get_deva_dir() {
55
+ Ok(dir) => dir,
56
+ Err(e) => {
57
+ eprintln!("❌ {}", e);
58
+ return;
59
+ }
60
+ };
61
+ let subdir = match obj_type {
62
+ "bank" => "banks",
63
+ "plugin" => "plugins",
64
+ "preset" => "presets",
65
+ "template" => "templates",
66
+ other => other,
67
+ };
68
+
69
+ // Determine the bank audio base directory. Prefer an optional
70
+ // `audioPath` declared in the bank's bank.toml (supports keys
71
+ // `audioPath` or `audio_path`). If absent, fall back to `audio/`.
72
+ let mut audio_dir = deva_dir.join(subdir).join(bank_name).join("audio");
73
+ // Try to read bank.toml to get audioPath
74
+ let bank_toml = deva_dir.join(subdir).join(bank_name).join("bank.toml");
75
+ if bank_toml.exists() {
76
+ if let Ok(content) = std::fs::read_to_string(&bank_toml) {
77
+ if let Ok(parsed) = toml::from_str::<toml::Value>(&content) {
78
+ if let Some(ap) = parsed
79
+ .get("audioPath")
80
+ .or_else(|| parsed.get("audio_path"))
81
+ .and_then(|v| v.as_str())
82
+ {
83
+ // normalize separators
84
+ let ap_norm = ap.replace("\\", "/");
85
+ audio_dir = deva_dir.join(subdir).join(bank_name).join(ap_norm);
86
+ }
87
+ }
88
+ }
89
+ }
90
+ // Force looking into the computed audio_dir. If the entity_name
91
+ // already contains an extension (e.g. .wav/.mp3) or a nested path,
92
+ // preserve it as-is. Otherwise, try with a .wav extension.
93
+ let bank_base = audio_dir;
94
+ let candidate = bank_base.join(&entity_name);
95
+
96
+ if candidate.exists() {
97
+ resolved_path = candidate.to_string_lossy().to_string();
98
+ } else {
99
+ // Detect whether the provided entity already includes an extension.
100
+ let has_extension = std::path::Path::new(&entity_name).extension().is_some();
101
+
102
+ if !has_extension {
103
+ // Try appending .wav as a fallback for shorthand names without extension
104
+ let wav_candidate = bank_base.join(format!("{}.wav", entity_name));
105
+ if wav_candidate.exists() {
106
+ resolved_path = wav_candidate.to_string_lossy().to_string();
107
+ } else {
108
+ // Last resort: use the legacy location (no audio/), also with .wav
109
+ resolved_path = deva_dir
110
+ .join(subdir)
111
+ .join(bank_name)
112
+ .join(format!("{}.wav", entity_name))
113
+ .to_string_lossy()
114
+ .to_string();
115
+ }
116
+ } else {
117
+ // If an extension was specified, don't append .wav; try legacy location
118
+ let legacy_candidate = deva_dir.join(subdir).join(bank_name).join(&entity_name);
119
+
120
+ if legacy_candidate.exists() {
121
+ resolved_path = legacy_candidate.to_string_lossy().to_string();
122
+ } else {
123
+ // No file found; fall back to the audio candidate path (even if missing)
124
+ resolved_path = candidate.to_string_lossy().to_string();
125
+ }
126
+ }
127
+ }
128
+ } else {
129
+ let entry_dir = module_root.parent().unwrap_or(&root);
130
+ let absolute_path = root.join(entry_dir).join(&var_path);
131
+
132
+ resolved_path = normalize_path(absolute_path.to_string_lossy().to_string());
133
+ }
134
+
135
+ if !Path::new(&resolved_path).exists() {
136
+ eprintln!("❌ Unknown trigger or missing audio file: {}", filepath);
137
+ return;
138
+ }
139
+
140
+ let file = match File::open(&resolved_path) {
141
+ Ok(f) => BufReader::new(f),
142
+ Err(e) => {
143
+ eprintln!("❌ Failed to open audio file {}: {}", resolved_path, e);
144
+ return;
145
+ }
146
+ };
147
+
148
+ let decoder = match Decoder::new(file) {
149
+ Ok(d) => d,
150
+ Err(e) => {
151
+ eprintln!("❌ Failed to decode audio file {}: {}", resolved_path, e);
152
+ return;
153
+ }
154
+ };
155
+
156
+ // Read frames from decoder and convert to mono if needed.
157
+ let max_frames = (dur_sec * (SAMPLE_RATE as f32)) as usize;
158
+ let dec_channels = decoder.channels() as usize;
159
+ let max_raw_samples = max_frames.saturating_mul(dec_channels.max(1));
160
+ let raw_samples: Vec<i16> = decoder.convert_samples().take(max_raw_samples).collect();
161
+
162
+ // Convert interleaved channels to mono by averaging channels per frame.
163
+ // Apply a small RMS-preserving scale so mono level is similar to mixed stereo.
164
+ let actual_frames = if dec_channels > 0 { raw_samples.len() / dec_channels } else { 0 };
165
+ let mut samples: Vec<i16> = Vec::with_capacity(actual_frames);
166
+ let rms_scale = (dec_channels as f32).sqrt();
167
+ for frame in 0..actual_frames {
168
+ let mut sum: i32 = 0;
169
+ for ch in 0..dec_channels {
170
+ sum += raw_samples[frame * dec_channels + ch] as i32;
171
+ }
172
+ if dec_channels > 0 {
173
+ let avg = (sum / (dec_channels as i32)) as f32;
174
+ let scaled = (avg * rms_scale).clamp(i16::MIN as f32, i16::MAX as f32) as i16;
175
+ samples.push(scaled);
176
+ } else {
177
+ samples.push(0);
178
+ }
179
+ }
180
+
181
+ if samples.is_empty() {
182
+ eprintln!("❌ No samples read from {}", resolved_path);
183
+ return;
184
+ }
185
+
186
+ let offset = (time_secs * (SAMPLE_RATE as f32) * (CHANNELS as f32)) as usize;
187
+ let required_len = offset + samples.len() * (CHANNELS as usize);
188
+ if self.buffer.len() < required_len {
189
+ self.buffer.resize(required_len, 0);
190
+ }
191
+
192
+ if let Some(effects_map) = effects {
193
+ self.pad_samples(&samples, time_secs, Some(effects_map));
194
+ } else {
195
+ self.pad_samples(&samples, time_secs, None);
196
+ }
197
+ }
198
+
199
+ fn pad_samples(
200
+ &mut self,
201
+ samples: &[i16],
202
+ time_secs: f32,
203
+ effects_map: Option<HashMap<String, Value>>,
204
+ ) {
205
+ let offset = (time_secs * (SAMPLE_RATE as f32) * (CHANNELS as f32)) as usize;
206
+ let total_samples = samples.len();
207
+
208
+ let mut gain = 1.0;
209
+ let mut pan = 0.0;
210
+ let mut fade_in = 0.0;
211
+ let mut fade_out = 0.0;
212
+ let mut pitch = 1.0;
213
+ let mut drive = 0.0;
214
+ let mut reverb = 0.0;
215
+ let mut delay = 0.0; // delay time in seconds
216
+ let delay_feedback = 0.35; // default feedback
217
+
218
+ if let Some(map) = &effects_map {
219
+ for (key, val) in map {
220
+ match (key.as_str(), val) {
221
+ ("gain", Value::Number(v)) => {
222
+ gain = *v;
223
+ }
224
+ ("pan", Value::Number(v)) => {
225
+ pan = *v;
226
+ }
227
+ ("fadeIn", Value::Number(v)) => {
228
+ fade_in = *v;
229
+ }
230
+ ("fadeOut", Value::Number(v)) => {
231
+ fade_out = *v;
232
+ }
233
+ ("pitch", Value::Number(v)) => {
234
+ pitch = *v;
235
+ }
236
+ ("drive", Value::Number(v)) => {
237
+ drive = *v;
238
+ }
239
+ ("reverb", Value::Number(v)) => {
240
+ reverb = *v;
241
+ }
242
+ ("delay", Value::Number(v)) => {
243
+ delay = *v;
244
+ }
245
+ _ => eprintln!("⚠️ Unknown or invalid effect '{}'", key),
246
+ }
247
+ }
248
+ }
249
+
250
+ let fade_in_samples = (fade_in * (SAMPLE_RATE as f32)) as usize;
251
+ let fade_out_samples = (fade_out * (SAMPLE_RATE as f32)) as usize;
252
+
253
+ // If no fade specified, apply a tiny default fade (2 ms) when sample boundaries are non-zero
254
+ let default_boundary_fade_ms = 1.0_f32; // 1 ms
255
+ let default_fade_samples = (default_boundary_fade_ms * (SAMPLE_RATE as f32)) as usize;
256
+ let mut effective_fade_in = fade_in_samples;
257
+ let mut effective_fade_out = fade_out_samples;
258
+ if effective_fade_in == 0 {
259
+ if let Some(&first) = samples.first() {
260
+ if first.abs() > 64 { // increased threshold to detect only strong abrupt starts
261
+ effective_fade_in = default_fade_samples.max(1);
262
+ }
263
+ }
264
+ }
265
+ if effective_fade_out == 0 {
266
+ if let Some(&last) = samples.last() {
267
+ if last.abs() > 64 { // increased threshold to detect only strong abrupt ends
268
+ effective_fade_out = default_fade_samples.max(1);
269
+ }
270
+ }
271
+ }
272
+
273
+ // Ensure fades do not exceed half the sample length to avoid silencing short samples
274
+ if total_samples > 0 {
275
+ let cap = total_samples / 2;
276
+ if effective_fade_in > cap {
277
+ effective_fade_in = cap.max(1);
278
+ }
279
+ if effective_fade_out > cap {
280
+ effective_fade_out = cap.max(1);
281
+ }
282
+ }
283
+
284
+ let delay_samples = if delay > 0.0 {
285
+ (delay * (SAMPLE_RATE as f32)) as usize
286
+ } else {
287
+ 0
288
+ };
289
+ let mut delay_buffer: Vec<f32> = vec![0.0; total_samples + delay_samples];
290
+
291
+ for i in 0..total_samples {
292
+ let pitch_index = if pitch != 1.0 {
293
+ ((i as f32) / pitch) as usize
294
+ } else {
295
+ i
296
+ };
297
+
298
+ let mut adjusted = if pitch_index < total_samples {
299
+ samples[pitch_index] as f32
300
+ } else {
301
+ 0.0
302
+ };
303
+
304
+ adjusted *= gain;
305
+
306
+ if effective_fade_in > 0 && i < effective_fade_in {
307
+ if effective_fade_in == 1 {
308
+ adjusted *= 0.0;
309
+ } else {
310
+ adjusted *= (i as f32) / (effective_fade_in as f32);
311
+ }
312
+ }
313
+ if effective_fade_out > 0 && i >= total_samples.saturating_sub(effective_fade_out) {
314
+ if effective_fade_out == 1 {
315
+ adjusted *= 0.0;
316
+ } else {
317
+ adjusted *= ((total_samples - 1 - i) as f32) / ((effective_fade_out - 1) as f32);
318
+ }
319
+ }
320
+
321
+ if drive > 0.0 {
322
+ let normalized = adjusted / (i16::MAX as f32);
323
+ let pre_gain = (10f32).powf(drive / 20.0);
324
+ let driven = (normalized * pre_gain).tanh();
325
+ adjusted = driven * (i16::MAX as f32);
326
+ }
327
+
328
+ if delay_samples > 0 && i >= delay_samples {
329
+ let echo = delay_buffer[i - delay_samples] * delay_feedback;
330
+ adjusted += echo;
331
+ }
332
+ if delay_samples > 0 {
333
+ delay_buffer[i] = adjusted;
334
+ }
335
+
336
+ if reverb > 0.0 {
337
+ let reverb_delay = (0.03 * (SAMPLE_RATE as f32)) as usize;
338
+ if i >= reverb_delay {
339
+ adjusted += (self.buffer[offset + i - reverb_delay] as f32) * reverb;
340
+ }
341
+ }
342
+
343
+ let adjusted_sample = adjusted.round().clamp(i16::MIN as f32, i16::MAX as f32) as i16;
344
+
345
+ let (left_gain, right_gain) = crate::core::audio::engine::helpers::pan_gains(pan);
346
+
347
+ let left = ((adjusted_sample as f32) * left_gain) as i16;
348
+ let right = ((adjusted_sample as f32) * right_gain) as i16;
349
+
350
+ let left_pos = offset + i * 2;
351
+ let right_pos = left_pos + 1;
352
+
353
+ if right_pos < self.buffer.len() {
354
+ self.buffer[left_pos] = self.buffer[left_pos].saturating_add(left);
355
+ self.buffer[right_pos] = self.buffer[right_pos].saturating_add(right);
356
+ }
357
+ }
358
+ }
359
+ }
@@ -0,0 +1,325 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ // Sample rate and channel constants used throughout the engine.
5
+ const SAMPLE_RATE: u32 = 44100;
6
+ const CHANNELS: u16 = 2;
7
+
8
+ /// AudioEngine holds the generated interleaved stereo buffer and
9
+ /// provides simple utilities to mix/merge buffers and export WAV files.
10
+ ///
11
+ /// Notes:
12
+ /// - Buffer is interleaved stereo (L,R,L,R...).
13
+ /// - Methods are synchronous and operate on in-memory buffers.
14
+ #[derive(Debug, Clone, PartialEq)]
15
+ pub struct AudioEngine {
16
+ /// Master volume multiplier (not automatically applied by helpers).
17
+ pub volume: f32,
18
+ /// Interleaved i16 PCM buffer.
19
+ pub buffer: Vec<i16>,
20
+ /// Logical module name used for error traces/diagnostics.
21
+ pub module_name: String,
22
+ /// Simple diagnostic counter for inserted notes.
23
+ pub note_count: usize,
24
+ }
25
+
26
+ impl AudioEngine {
27
+ pub fn new(module_name: String) -> Self {
28
+ AudioEngine {
29
+ volume: 1.0,
30
+ buffer: vec![],
31
+ module_name,
32
+ note_count: 0,
33
+ }
34
+ }
35
+
36
+ pub fn get_buffer(&self) -> &[i16] {
37
+ &self.buffer
38
+ }
39
+
40
+ pub fn get_normalized_buffer(&self) -> Vec<f32> {
41
+ self.buffer.iter().map(|&s| (s as f32) / 32768.0).collect()
42
+ }
43
+
44
+ pub fn mix(&mut self, other: &AudioEngine) {
45
+ let max_len = self.buffer.len().max(other.buffer.len());
46
+ self.buffer.resize(max_len, 0);
47
+
48
+ for (i, &sample) in other.buffer.iter().enumerate() {
49
+ self.buffer[i] = self.buffer[i].saturating_add(sample);
50
+ }
51
+ }
52
+
53
+ pub fn merge_with(&mut self, other: AudioEngine) {
54
+ // If the other buffer is empty, simply return without warning (common for spawns that produced nothing)
55
+ if other.buffer.is_empty() {
56
+ return;
57
+ }
58
+
59
+ // If the other buffer is present but contains only zeros, warn and skip merge
60
+ if other.buffer.iter().all(|&s| s == 0) {
61
+ eprintln!("⚠️ Skipping merge: other buffer is silent");
62
+ return;
63
+ }
64
+
65
+ if self.buffer.iter().all(|&s| s == 0) {
66
+ self.buffer = other.buffer;
67
+ return;
68
+ }
69
+
70
+ self.mix(&other);
71
+ }
72
+
73
+ pub fn set_duration(&mut self, duration_secs: f32) {
74
+ let total_samples = (duration_secs * (SAMPLE_RATE as f32) * (CHANNELS as f32)) as usize;
75
+
76
+ if self.buffer.len() < total_samples {
77
+ self.buffer.resize(total_samples, 0);
78
+ }
79
+ }
80
+
81
+ pub fn generate_wav_file(&mut self, output_dir: &String) -> Result<(), String> {
82
+ if self.buffer.len() % (CHANNELS as usize) != 0 {
83
+ self.buffer.push(0);
84
+ println!("Completed buffer to respect stereo format.");
85
+ }
86
+
87
+ let spec = hound::WavSpec {
88
+ channels: CHANNELS,
89
+ sample_rate: SAMPLE_RATE,
90
+ bits_per_sample: 16,
91
+ sample_format: hound::SampleFormat::Int,
92
+ };
93
+
94
+ let mut writer = hound::WavWriter::create(output_dir, spec)
95
+ .map_err(|e| format!("Error creating WAV file: {}", e))?;
96
+
97
+ for sample in &self.buffer {
98
+ writer
99
+ .write_sample(*sample)
100
+ .map_err(|e| format!("Error writing sample: {:?}", e))?;
101
+ }
102
+
103
+ writer
104
+ .finalize()
105
+ .map_err(|e| format!("Error finalizing WAV: {:?}", e))?;
106
+
107
+ Ok(())
108
+ }
109
+
110
+ // Insert note moved here from original engine.rs
111
+ pub fn insert_note(
112
+ &mut self,
113
+ waveform: String,
114
+ freq: f32,
115
+ amp: f32,
116
+ start_time_ms: f32,
117
+ duration_ms: f32,
118
+ synth_params: HashMap<String, Value>,
119
+ note_params: HashMap<String, Value>,
120
+ automation: Option<HashMap<String, Value>>,
121
+ ) {
122
+ // Keep internal logic; helpers called from helpers module
123
+ let attack = self.extract_f32(&synth_params, "attack").unwrap_or(0.0);
124
+ let decay = self.extract_f32(&synth_params, "decay").unwrap_or(0.0);
125
+ let sustain = self.extract_f32(&synth_params, "sustain").unwrap_or(1.0);
126
+ let release = self.extract_f32(&synth_params, "release").unwrap_or(0.0);
127
+ let attack_s = if attack > 10.0 {
128
+ attack / 1000.0
129
+ } else {
130
+ attack
131
+ };
132
+ let decay_s = if decay > 10.0 { decay / 1000.0 } else { decay };
133
+ let release_s = if release > 10.0 {
134
+ release / 1000.0
135
+ } else {
136
+ release
137
+ };
138
+ let sustain_level = if sustain > 1.0 {
139
+ (sustain / 100.0).clamp(0.0, 1.0)
140
+ } else {
141
+ sustain.clamp(0.0, 1.0)
142
+ };
143
+
144
+ let duration_ms = self
145
+ .extract_f32(&note_params, "duration")
146
+ .unwrap_or(duration_ms);
147
+ let velocity = self.extract_f32(&note_params, "velocity").unwrap_or(1.0);
148
+ let glide = self.extract_boolean(&note_params, "glide").unwrap_or(false);
149
+ let slide = self.extract_boolean(&note_params, "slide").unwrap_or(false);
150
+
151
+ let _amplitude = (i16::MAX as f32) * amp.clamp(0.0, 1.0) * velocity.clamp(0.0, 1.0);
152
+
153
+ let freq_start = freq;
154
+ let mut freq_end = freq;
155
+ let amp_start = amp * velocity.clamp(0.0, 1.0);
156
+ let mut amp_end = amp_start;
157
+
158
+ if glide {
159
+ if let Some(Value::Number(target_freq)) = note_params.get("target_freq") {
160
+ freq_end = *target_freq;
161
+ } else {
162
+ freq_end = freq * 1.5;
163
+ }
164
+ }
165
+
166
+ if slide {
167
+ if let Some(Value::Number(target_amp)) = note_params.get("target_amp") {
168
+ amp_end = *target_amp * velocity.clamp(0.0, 1.0);
169
+ } else {
170
+ amp_end = amp_start * 0.5;
171
+ }
172
+ }
173
+
174
+ let sample_rate = SAMPLE_RATE as f32;
175
+ let channels = CHANNELS as usize;
176
+
177
+ let total_samples = ((duration_ms / 1000.0) * sample_rate) as usize;
178
+ let start_sample = ((start_time_ms / 1000.0) * sample_rate) as usize;
179
+
180
+ let (volume_env, pan_env, pitch_env) =
181
+ crate::core::audio::engine::helpers::env_maps_from_automation(&automation);
182
+
183
+ let mut stereo_samples: Vec<i16> = Vec::with_capacity(total_samples * 2);
184
+ let fade_len = (sample_rate * 0.01) as usize; // 10 ms fade
185
+
186
+ let attack_samples = (attack_s * sample_rate) as usize;
187
+ let decay_samples = (decay_s * sample_rate) as usize;
188
+ let release_samples = (release_s * sample_rate) as usize;
189
+ let sustain_samples = if total_samples > attack_samples + decay_samples + release_samples {
190
+ total_samples - attack_samples - decay_samples - release_samples
191
+ } else {
192
+ 0
193
+ };
194
+
195
+ for i in 0..total_samples {
196
+ let t = ((start_sample + i) as f32) / sample_rate;
197
+
198
+ // Glide
199
+ let current_freq = if glide {
200
+ freq_start + ((freq_end - freq_start) * (i as f32)) / (total_samples as f32)
201
+ } else {
202
+ freq
203
+ };
204
+
205
+ // Pitch automation (in semitones), applied as frequency multiplier
206
+ let pitch_semi = crate::core::audio::engine::helpers::eval_env_map(
207
+ &pitch_env,
208
+ (i as f32) / (total_samples as f32),
209
+ 0.0,
210
+ );
211
+ let current_freq = current_freq * (2.0_f32).powf(pitch_semi / 12.0);
212
+
213
+ // Slide
214
+ let current_amp = if slide {
215
+ amp_start + ((amp_end - amp_start) * (i as f32)) / (total_samples as f32)
216
+ } else {
217
+ amp_start
218
+ };
219
+
220
+ let mut value =
221
+ crate::core::audio::engine::helpers::oscillator_sample(&waveform, current_freq, t);
222
+
223
+ // ADSR envelope
224
+ let envelope = crate::core::audio::engine::helpers::adsr_envelope_value(
225
+ i,
226
+ attack_samples,
227
+ decay_samples,
228
+ sustain_samples,
229
+ release_samples,
230
+ sustain_level,
231
+ );
232
+
233
+ // Fade in/out
234
+ if fade_len > 0 && i < fade_len {
235
+ if fade_len == 1 {
236
+ value *= 0.0;
237
+ } else {
238
+ value *= (i as f32) / (fade_len as f32);
239
+ }
240
+ } else if fade_len > 0 && i >= total_samples.saturating_sub(fade_len) {
241
+ if fade_len == 1 {
242
+ value *= 0.0;
243
+ } else {
244
+ // ensure last sample becomes exactly zero to avoid clicks
245
+ value *= ((total_samples - 1 - i) as f32) / ((fade_len - 1) as f32);
246
+ }
247
+ }
248
+
249
+ value *= envelope;
250
+ let mut sample_val = value * (i16::MAX as f32) * current_amp;
251
+
252
+ let vol_mul = crate::core::audio::engine::helpers::eval_env_map(
253
+ &volume_env,
254
+ (i as f32) / (total_samples as f32),
255
+ 1.0,
256
+ )
257
+ .clamp(0.0, 10.0);
258
+ sample_val *= vol_mul;
259
+
260
+ let pan_val = crate::core::audio::engine::helpers::eval_env_map(
261
+ &pan_env,
262
+ (i as f32) / (total_samples as f32),
263
+ 0.0,
264
+ )
265
+ .clamp(-1.0, 1.0);
266
+ let (left_gain, right_gain) = crate::core::audio::engine::helpers::pan_gains(pan_val);
267
+
268
+ let left = (sample_val * left_gain)
269
+ .round()
270
+ .clamp(i16::MIN as f32, i16::MAX as f32) as i16;
271
+ let right = (sample_val * right_gain)
272
+ .round()
273
+ .clamp(i16::MIN as f32, i16::MAX as f32) as i16;
274
+
275
+ stereo_samples.push(left);
276
+ stereo_samples.push(right);
277
+ }
278
+
279
+ // Increment note counter for diagnostics
280
+ self.note_count = self.note_count.saturating_add(1);
281
+
282
+ crate::core::audio::engine::helpers::mix_stereo_samples_into_buffer(
283
+ self,
284
+ start_sample,
285
+ channels,
286
+ &stereo_samples,
287
+ );
288
+ }
289
+
290
+ // helper extraction functions left in this struct for now
291
+ fn extract_f32(&self, map: &HashMap<String, Value>, key: &str) -> Option<f32> {
292
+ match map.get(key) {
293
+ Some(Value::Number(n)) => Some(*n),
294
+ Some(Value::String(s)) => s.parse::<f32>().ok(),
295
+ Some(Value::Boolean(b)) => Some(if *b { 1.0 } else { 0.0 }),
296
+ _ => None,
297
+ }
298
+ }
299
+
300
+ fn extract_boolean(&self, map: &HashMap<String, Value>, key: &str) -> Option<bool> {
301
+ match map.get(key) {
302
+ Some(Value::Boolean(b)) => Some(*b),
303
+ Some(Value::Number(n)) => Some(*n != 0.0),
304
+ Some(Value::Identifier(s)) => {
305
+ if s == "true" {
306
+ Some(true)
307
+ } else if s == "false" {
308
+ Some(false)
309
+ } else {
310
+ None
311
+ }
312
+ }
313
+ Some(Value::String(s)) => {
314
+ if s == "true" {
315
+ Some(true)
316
+ } else if s == "false" {
317
+ Some(false)
318
+ } else {
319
+ None
320
+ }
321
+ }
322
+ _ => None,
323
+ }
324
+ }
325
+ }