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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (207) hide show
  1. package/.devalang +9 -10
  2. package/Cargo.toml +84 -80
  3. package/README.md +10 -7
  4. package/docs/CHANGELOG.md +83 -0
  5. package/docs/ROADMAP.md +6 -2
  6. package/docs/TODO.md +3 -14
  7. package/examples/bus.deva +10 -0
  8. package/examples/chain.deva +19 -0
  9. package/examples/effect.deva +2 -0
  10. package/examples/filter.deva +11 -0
  11. package/examples/lfo.deva +9 -0
  12. package/examples/plugin.deva +10 -10
  13. package/examples/routing.deva +23 -0
  14. package/examples/synth.deva +11 -1
  15. package/examples/synth_types.deva +17 -0
  16. package/out-tsc/bin/project-version.json +6 -0
  17. package/out-tsc/core/functions/index.d.ts +5 -0
  18. package/out-tsc/core/functions/index.js +11 -0
  19. package/out-tsc/pkg/devalang_core.d.ts +2 -0
  20. package/out-tsc/pkg/devalang_core.js +17 -2
  21. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +1 -0
  22. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  23. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  24. package/package.json +23 -10
  25. package/project-version.json +3 -3
  26. package/rust/bindings/Cargo.toml +9 -0
  27. package/rust/bindings/src/lib.rs +86 -0
  28. package/rust/cli/addon/commands.rs +35 -0
  29. package/rust/cli/addon/download.rs +234 -0
  30. package/rust/cli/addon/install.rs +33 -0
  31. package/rust/cli/addon/list.rs +224 -0
  32. package/rust/cli/addon/metadata.rs +124 -0
  33. package/rust/cli/addon/mod.rs +8 -0
  34. package/rust/cli/addon/remove.rs +271 -0
  35. package/rust/cli/addon/update.rs +305 -0
  36. package/rust/cli/{install/addon.rs → addon/utils.rs} +34 -43
  37. package/rust/cli/build/commands.rs +153 -103
  38. package/rust/cli/build/mod.rs +2 -2
  39. package/rust/cli/build/process.rs +165 -146
  40. package/rust/cli/check/mod.rs +208 -208
  41. package/rust/cli/discover/commands.rs +53 -31
  42. package/rust/cli/discover/config.rs +2 -4
  43. package/rust/cli/discover/install.rs +139 -28
  44. package/rust/cli/discover/metadata.rs +3 -3
  45. package/rust/cli/login/commands.rs +124 -124
  46. package/rust/cli/me/commands.rs +52 -0
  47. package/rust/cli/me/mod.rs +1 -0
  48. package/rust/cli/mod.rs +2 -2
  49. package/rust/cli/parser.rs +76 -70
  50. package/rust/cli/play/commands.rs +375 -324
  51. package/rust/cli/play/mod.rs +5 -5
  52. package/rust/cli/play/process.rs +159 -150
  53. package/rust/cli/play/realtime.rs +91 -91
  54. package/rust/cli/telemetry/commands.rs +22 -22
  55. package/rust/cli/telemetry/event_creator.rs +80 -80
  56. package/rust/cli/telemetry/mod.rs +3 -3
  57. package/rust/cli/telemetry/send.rs +51 -51
  58. package/rust/cli/template/commands.rs +69 -69
  59. package/rust/config/driver.rs +112 -103
  60. package/rust/config/mod.rs +3 -3
  61. package/rust/config/ops.rs +26 -26
  62. package/rust/config/settings.rs +101 -101
  63. package/rust/core/audio/engine/driver.rs +237 -0
  64. package/rust/core/audio/engine/export.rs +169 -0
  65. package/rust/core/audio/engine/helpers.rs +178 -170
  66. package/rust/core/audio/engine/mod.rs +56 -7
  67. package/rust/core/audio/engine/notes/dsp.rs +88 -0
  68. package/rust/core/audio/engine/notes/mod.rs +53 -0
  69. package/rust/core/audio/engine/notes/params.rs +294 -0
  70. package/rust/core/audio/engine/sample/insert.rs +300 -0
  71. package/rust/core/audio/engine/sample/mod.rs +40 -0
  72. package/rust/core/audio/engine/sample/padding.rs +170 -0
  73. package/rust/core/audio/evaluator/condition.rs +61 -0
  74. package/rust/core/audio/evaluator/mod.rs +9 -0
  75. package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +152 -310
  76. package/rust/core/audio/evaluator/rhs.rs +16 -0
  77. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  78. package/rust/core/audio/interpreter/driver.rs +574 -542
  79. package/rust/core/audio/interpreter/mod.rs +2 -14
  80. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +179 -0
  81. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -0
  82. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  83. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +3 -0
  84. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +371 -0
  85. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  86. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  87. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  88. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  89. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  90. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  91. package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +2 -4
  92. package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +36 -5
  93. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -71
  94. package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +24 -26
  95. package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +36 -38
  96. package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +17 -19
  97. package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +115 -114
  98. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  99. package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
  100. package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +54 -4
  101. package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
  102. package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +242 -239
  103. package/rust/core/audio/loader/trigger.rs +98 -97
  104. package/rust/core/audio/mod.rs +6 -7
  105. package/rust/core/audio/special/easing.rs +189 -189
  106. package/rust/core/audio/special/env.rs +45 -45
  107. package/rust/core/audio/special/math.rs +134 -134
  108. package/rust/core/audio/special/modulator.rs +143 -143
  109. package/rust/core/builder/mod.rs +129 -86
  110. package/rust/core/debugger/{module.rs → logs.rs} +52 -55
  111. package/rust/core/debugger/mod.rs +30 -30
  112. package/rust/core/debugger/store.rs +38 -40
  113. package/rust/core/error/mod.rs +269 -269
  114. package/rust/core/lexer/driver.rs +2 -4
  115. package/rust/core/mod.rs +9 -10
  116. package/rust/core/parser/driver/block.rs +111 -0
  117. package/rust/core/parser/driver/cursor.rs +82 -0
  118. package/rust/core/parser/driver/driver_impl.rs +159 -0
  119. package/rust/core/parser/driver/mod.rs +6 -0
  120. package/rust/core/parser/driver/parse_array.rs +120 -0
  121. package/rust/core/parser/driver/parse_map.rs +247 -0
  122. package/rust/core/parser/driver/parser.rs +160 -0
  123. package/rust/core/parser/handler/arrow_call.rs +90 -15
  124. package/rust/core/parser/handler/at.rs +279 -279
  125. package/rust/core/parser/handler/bank.rs +104 -104
  126. package/rust/core/parser/handler/condition.rs +83 -83
  127. package/rust/core/parser/handler/dot.rs +148 -148
  128. package/rust/core/parser/handler/identifier/automate.rs +254 -254
  129. package/rust/core/parser/handler/identifier/call.rs +91 -91
  130. package/rust/core/parser/handler/identifier/emit.rs +70 -70
  131. package/rust/core/parser/handler/identifier/function.rs +113 -113
  132. package/rust/core/parser/handler/identifier/group.rs +89 -89
  133. package/rust/core/parser/handler/identifier/let_.rs +173 -173
  134. package/rust/core/parser/handler/identifier/mod.rs +55 -55
  135. package/rust/core/parser/handler/identifier/on.rs +107 -107
  136. package/rust/core/parser/handler/identifier/print.rs +49 -49
  137. package/rust/core/parser/handler/identifier/sleep.rs +96 -43
  138. package/rust/core/parser/handler/identifier/spawn.rs +91 -91
  139. package/rust/core/parser/handler/identifier/synth.rs +39 -3
  140. package/rust/core/parser/handler/loop_.rs +194 -194
  141. package/rust/core/parser/handler/pattern.rs +25 -2
  142. package/rust/core/parser/handler/tempo.rs +105 -57
  143. package/rust/core/parser/statement.rs +10 -11
  144. package/rust/core/plugin/loader.rs +137 -137
  145. package/rust/core/plugin/runner/mod.rs +11 -0
  146. package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +206 -72
  147. package/rust/core/plugin/runner/wasm32.rs +44 -0
  148. package/rust/core/preprocessor/loader/inject.rs +313 -0
  149. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  150. package/rust/core/preprocessor/loader/mod.rs +235 -0
  151. package/rust/core/preprocessor/module.rs +55 -60
  152. package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +107 -114
  153. package/rust/core/preprocessor/processor/mod.rs +1 -0
  154. package/rust/core/preprocessor/resolver/function.rs +69 -69
  155. package/rust/core/preprocessor/resolver/group.rs +122 -94
  156. package/rust/core/preprocessor/resolver/pattern.rs +14 -2
  157. package/rust/core/store/global.rs +57 -61
  158. package/rust/core/store/mod.rs +1 -5
  159. package/rust/lib.rs +323 -308
  160. package/rust/macros/Cargo.toml +14 -0
  161. package/rust/macros/src/lib.rs +52 -0
  162. package/rust/main.rs +336 -143
  163. package/rust/types/Cargo.toml +1 -1
  164. package/rust/types/src/addons.rs +57 -55
  165. package/rust/types/src/config.rs +82 -74
  166. package/rust/types/src/lib.rs +15 -12
  167. package/rust/types/src/plugin.rs +20 -0
  168. package/rust/types/src/store.rs +139 -0
  169. package/rust/types/src/telemetry.rs +85 -85
  170. package/rust/utils/Cargo.toml +5 -2
  171. package/rust/utils/src/file.rs +477 -94
  172. package/rust/utils/src/first_usage.rs +97 -97
  173. package/rust/utils/src/lib.rs +9 -9
  174. package/rust/utils/src/logger.rs +200 -200
  175. package/rust/utils/src/path.rs +158 -88
  176. package/rust/utils/src/signature.rs +41 -41
  177. package/rust/utils/src/spinner.rs +20 -20
  178. package/rust/utils/src/version.rs +58 -27
  179. package/rust/utils/src/watcher.rs +46 -46
  180. package/rust/web/api.rs +5 -5
  181. package/rust/web/auth.rs +5 -0
  182. package/rust/web/cdn.rs +34 -34
  183. package/rust/web/forge.rs +5 -0
  184. package/rust/web/mod.rs +2 -0
  185. package/tests/integration.rs +21 -21
  186. package/typescript/core/functions/index.ts +11 -0
  187. package/typescript/pkg/devalang_core.ts +20 -4
  188. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  189. package/rust/cli/bank/api.rs +0 -122
  190. package/rust/cli/bank/commands.rs +0 -275
  191. package/rust/cli/bank/mod.rs +0 -29
  192. package/rust/cli/install/bank.rs +0 -53
  193. package/rust/cli/install/commands.rs +0 -35
  194. package/rust/cli/install/mod.rs +0 -4
  195. package/rust/cli/install/plugin.rs +0 -61
  196. package/rust/core/audio/engine/sample.rs +0 -366
  197. package/rust/core/audio/engine/synth.rs +0 -325
  198. package/rust/core/audio/interpreter/arrow_call.rs +0 -311
  199. package/rust/core/audio/renderer.rs +0 -54
  200. package/rust/core/parser/driver.rs +0 -584
  201. package/rust/core/preprocessor/loader.rs +0 -637
  202. package/rust/core/store/export.rs +0 -28
  203. package/rust/core/store/function.rs +0 -40
  204. package/rust/core/store/import.rs +0 -28
  205. package/rust/core/store/variable.rs +0 -51
  206. package/rust/core/utils/mod.rs +0 -1
  207. package/rust/core/utils/path.rs +0 -37
@@ -1,26 +1,26 @@
1
- use crate::config::driver::ProjectConfig;
2
- use std::fs;
3
- use std::path::Path;
4
-
5
- pub fn load_config(path: Option<&Path>) -> Option<ProjectConfig> {
6
- let config_path_buf;
7
- let config_path = match path {
8
- Some(p) => p,
9
- None => {
10
- config_path_buf = match devalang_utils::path::get_devalang_config_path() {
11
- Ok(p) => p,
12
- Err(_) => {
13
- return None;
14
- }
15
- };
16
- &config_path_buf
17
- }
18
- };
19
-
20
- if config_path.exists() {
21
- let content = fs::read_to_string(config_path).ok()?;
22
- toml::from_str(&content).ok()
23
- } else {
24
- None
25
- }
26
- }
1
+ use crate::config::driver::ProjectConfig;
2
+ use std::fs;
3
+ use std::path::Path;
4
+
5
+ pub fn load_config(path: Option<&Path>) -> Option<ProjectConfig> {
6
+ let config_path_buf;
7
+ let config_path = match path {
8
+ Some(p) => p,
9
+ None => {
10
+ config_path_buf = match devalang_utils::path::get_devalang_config_path() {
11
+ Ok(p) => p,
12
+ Err(_) => {
13
+ return None;
14
+ }
15
+ };
16
+ &config_path_buf
17
+ }
18
+ };
19
+
20
+ if config_path.exists() {
21
+ let content = fs::read_to_string(config_path).ok()?;
22
+ toml::from_str(&content).ok()
23
+ } else {
24
+ None
25
+ }
26
+ }
@@ -1,101 +1,101 @@
1
- use devalang_types::{TelemetrySettings, UserSettings};
2
- use serde_json::Value as JsonValue;
3
- use std::io::Write;
4
-
5
- pub fn get_home_dir() -> Option<std::path::PathBuf> {
6
- dirs::home_dir()
7
- }
8
-
9
- pub fn get_devalang_homedir() -> std::path::PathBuf {
10
- if let Some(home_dir) = get_home_dir() {
11
- home_dir.join(".devalang")
12
- } else {
13
- std::path::PathBuf::from("~/.devalang")
14
- }
15
- }
16
-
17
- pub fn get_default_user_config() -> UserSettings {
18
- UserSettings {
19
- session: "".into(),
20
- telemetry: TelemetrySettings {
21
- uuid: uuid::Uuid::new_v4().to_string(),
22
- enabled: false,
23
- level: "basic".into(),
24
- stats: false,
25
- },
26
- }
27
- }
28
-
29
- pub fn get_user_config() -> Option<UserSettings> {
30
- if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
31
- let file = std::fs::File::open(config_path).ok()?;
32
- let settings = serde_json::from_reader(file).ok()?;
33
- Some(settings)
34
- } else {
35
- None
36
- }
37
- }
38
-
39
- pub fn write_user_config_file() {
40
- if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
41
- let settings = get_user_config().unwrap_or_else(get_default_user_config);
42
-
43
- let config_json = serde_json::to_string(&settings).unwrap();
44
-
45
- if let Err(e) = write_config_atomic(&config_path, &config_json) {
46
- println!("Could not write config file: {}", e);
47
- }
48
- } else {
49
- println!("Could not create config file");
50
- }
51
- }
52
-
53
- pub fn ensure_user_config_file_exists() {
54
- if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
55
- if !config_path.exists() {
56
- write_user_config_file();
57
- }
58
- }
59
- }
60
-
61
- pub fn set_user_config_value(key: &str, value: JsonValue) {
62
- let mut settings = get_user_config().unwrap_or_default();
63
-
64
- match (key, &value) {
65
- ("telemetry", JsonValue::Bool(b)) => {
66
- settings.telemetry.enabled = *b;
67
- }
68
- ("session", JsonValue::String(s)) => {
69
- settings.session = s.clone();
70
- }
71
- _ => {
72
- println!("Unsupported key or value type for '{}': {:?}", key, value);
73
- }
74
- }
75
-
76
- if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
77
- let config_json = serde_json::to_string(&settings).unwrap();
78
- if let Err(e) = write_config_atomic(&config_path, &config_json) {
79
- println!("Could not write config file: {}", e);
80
- }
81
- } else {
82
- println!("Could not create config file");
83
- }
84
- }
85
-
86
- pub fn write_config_atomic(
87
- config_path: &std::path::PathBuf,
88
- contents: &str,
89
- ) -> std::io::Result<()> {
90
- if let Some(parent) = config_path.parent() {
91
- std::fs::create_dir_all(parent)?;
92
- }
93
-
94
- let tmp_path = config_path.with_extension("json.tmp");
95
- let mut tmp_file = std::fs::File::create(&tmp_path)?;
96
- tmp_file.write_all(contents.as_bytes())?;
97
- tmp_file.sync_all()?;
98
- std::fs::rename(&tmp_path, config_path)?;
99
-
100
- Ok(())
101
- }
1
+ use devalang_types::{TelemetrySettings, UserSettings};
2
+ use serde_json::Value as JsonValue;
3
+ use std::io::Write;
4
+
5
+ pub fn get_home_dir() -> Option<std::path::PathBuf> {
6
+ dirs::home_dir()
7
+ }
8
+
9
+ pub fn get_devalang_homedir() -> std::path::PathBuf {
10
+ if let Some(home_dir) = get_home_dir() {
11
+ home_dir.join(".devalang")
12
+ } else {
13
+ std::path::PathBuf::from("~/.devalang")
14
+ }
15
+ }
16
+
17
+ pub fn get_default_user_config() -> UserSettings {
18
+ UserSettings {
19
+ session: "".into(),
20
+ telemetry: TelemetrySettings {
21
+ uuid: uuid::Uuid::new_v4().to_string(),
22
+ enabled: false,
23
+ level: "basic".into(),
24
+ stats: false,
25
+ },
26
+ }
27
+ }
28
+
29
+ pub fn get_user_config() -> Option<UserSettings> {
30
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
31
+ let file = std::fs::File::open(config_path).ok()?;
32
+ let settings = serde_json::from_reader(file).ok()?;
33
+ Some(settings)
34
+ } else {
35
+ None
36
+ }
37
+ }
38
+
39
+ pub fn write_user_config_file() {
40
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
41
+ let settings = get_user_config().unwrap_or_else(get_default_user_config);
42
+
43
+ let config_json = serde_json::to_string(&settings).unwrap();
44
+
45
+ if let Err(e) = write_config_atomic(&config_path, &config_json) {
46
+ println!("Could not write config file: {}", e);
47
+ }
48
+ } else {
49
+ println!("Could not create config file");
50
+ }
51
+ }
52
+
53
+ pub fn ensure_user_config_file_exists() {
54
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
55
+ if !config_path.exists() {
56
+ write_user_config_file();
57
+ }
58
+ }
59
+ }
60
+
61
+ pub fn set_user_config_value(key: &str, value: JsonValue) {
62
+ let mut settings = get_user_config().unwrap_or_default();
63
+
64
+ match (key, &value) {
65
+ ("telemetry", JsonValue::Bool(b)) => {
66
+ settings.telemetry.enabled = *b;
67
+ }
68
+ ("session", JsonValue::String(s)) => {
69
+ settings.session = s.clone();
70
+ }
71
+ _ => {
72
+ println!("Unsupported key or value type for '{}': {:?}", key, value);
73
+ }
74
+ }
75
+
76
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
77
+ let config_json = serde_json::to_string(&settings).unwrap();
78
+ if let Err(e) = write_config_atomic(&config_path, &config_json) {
79
+ println!("Could not write config file: {}", e);
80
+ }
81
+ } else {
82
+ println!("Could not create config file");
83
+ }
84
+ }
85
+
86
+ pub fn write_config_atomic(
87
+ config_path: &std::path::PathBuf,
88
+ contents: &str,
89
+ ) -> std::io::Result<()> {
90
+ if let Some(parent) = config_path.parent() {
91
+ std::fs::create_dir_all(parent)?;
92
+ }
93
+
94
+ let tmp_path = config_path.with_extension("json.tmp");
95
+ let mut tmp_file = std::fs::File::create(&tmp_path)?;
96
+ tmp_file.write_all(contents.as_bytes())?;
97
+ tmp_file.sync_all()?;
98
+ std::fs::rename(&tmp_path, config_path)?;
99
+
100
+ Ok(())
101
+ }
@@ -0,0 +1,237 @@
1
+ use devalang_types::Value;
2
+ use std::collections::HashMap;
3
+
4
+ // Minimal representation of a MIDI note event for export purposes.
5
+ #[derive(Debug, Clone, PartialEq)]
6
+ pub struct MidiNoteEvent {
7
+ /// MIDI key number 0-127
8
+ pub key: u8,
9
+ /// velocity 0-127
10
+ pub vel: u8,
11
+ /// start time in milliseconds (absolute)
12
+ pub start_ms: u32,
13
+ /// duration in milliseconds
14
+ pub duration_ms: u32,
15
+ /// MIDI channel (0-15)
16
+ pub channel: u8,
17
+ }
18
+
19
+ // Sample rate and channel constants used throughout the engine.
20
+ pub const SAMPLE_RATE: u32 = 44100;
21
+ pub const CHANNELS: u16 = 2;
22
+
23
+ /// AudioEngine holds the generated interleaved stereo buffer and
24
+ /// provides simple utilities to mix/merge buffers and export WAV files.
25
+ ///
26
+ /// Notes:
27
+ /// - Buffer is interleaved stereo (L,R,L,R...).
28
+ /// - Methods are synchronous and operate on in-memory buffers.
29
+ ///
30
+ #[derive(Debug, Clone, PartialEq)]
31
+ pub struct AudioEngine {
32
+ /// Master volume multiplier (not automatically applied by helpers).
33
+ pub volume: f32,
34
+ /// Interleaved i16 PCM buffer.
35
+ pub buffer: Vec<i16>,
36
+ /// Collected MIDI note events for export (non-audio representation).
37
+ pub midi_events: Vec<MidiNoteEvent>,
38
+ /// Map target synth -> last inserted note sample ranges (start_sample, total_samples)
39
+ pub last_notes: std::collections::HashMap<String, Vec<(usize, usize)>>,
40
+ /// Logical module name used for error traces/diagnostics.
41
+ pub module_name: String,
42
+ /// Simple diagnostic counter for inserted notes.
43
+ pub note_count: usize,
44
+ /// Sample rate (can be overridden per-engine)
45
+ pub sample_rate: u32,
46
+ /// Number of channels (interleaved). Defaults to 2.
47
+ pub channels: u16,
48
+ }
49
+
50
+ impl AudioEngine {
51
+ pub fn new(module_name: String) -> Self {
52
+ AudioEngine {
53
+ volume: 1.0,
54
+ buffer: vec![],
55
+ midi_events: Vec::new(),
56
+ module_name,
57
+ note_count: 0,
58
+ sample_rate: SAMPLE_RATE,
59
+ channels: CHANNELS,
60
+ last_notes: std::collections::HashMap::new(),
61
+ }
62
+ }
63
+
64
+ pub fn get_buffer(&self) -> &[i16] {
65
+ &self.buffer
66
+ }
67
+
68
+ pub fn get_normalized_buffer(&self) -> Vec<f32> {
69
+ self.buffer.iter().map(|&s| (s as f32) / 32768.0).collect()
70
+ }
71
+
72
+ pub fn mix(&mut self, other: &AudioEngine) {
73
+ let max_len = self.buffer.len().max(other.buffer.len());
74
+ self.buffer.resize(max_len, 0);
75
+
76
+ for (i, &sample) in other.buffer.iter().enumerate() {
77
+ self.buffer[i] = self.buffer[i].saturating_add(sample);
78
+ }
79
+ }
80
+
81
+ pub fn merge_with(&mut self, other: AudioEngine) {
82
+ // If the other buffer is empty, simply return without warning (common for spawns that produced nothing)
83
+ if other.buffer.is_empty() {
84
+ return;
85
+ }
86
+
87
+ // If the other buffer is present but contains only zeros, warn and skip merge
88
+ if other.buffer.iter().all(|&s| s == 0) {
89
+ eprintln!("⚠️ Skipping merge: other buffer is silent");
90
+ return;
91
+ }
92
+
93
+ if self.buffer.iter().all(|&s| s == 0) {
94
+ self.buffer = other.buffer;
95
+ return;
96
+ }
97
+
98
+ self.mix(&other);
99
+ }
100
+
101
+ pub fn set_duration(&mut self, duration_secs: f32) {
102
+ let total_samples =
103
+ (duration_secs * (self.sample_rate as f32) * (self.channels as f32)) as usize;
104
+
105
+ if self.buffer.len() < total_samples {
106
+ self.buffer.resize(total_samples, 0);
107
+ }
108
+ }
109
+
110
+ pub fn generate_midi_file(
111
+ &mut self,
112
+ output_path: &String,
113
+ bpm: Option<f32>,
114
+ tpqn: Option<u16>,
115
+ ) -> Result<(), String> {
116
+ crate::core::audio::engine::export::generate_midi_file_impl(
117
+ &self.midi_events,
118
+ output_path,
119
+ bpm,
120
+ tpqn,
121
+ )
122
+ }
123
+
124
+ pub fn generate_wav_file(
125
+ &mut self,
126
+ output_dir: &String,
127
+ audio_format: Option<String>,
128
+ sample_rate: Option<u32>,
129
+ ) -> Result<(), String> {
130
+ crate::core::audio::engine::export::generate_wav_file_impl(
131
+ &mut self.buffer,
132
+ output_dir,
133
+ audio_format,
134
+ sample_rate,
135
+ )
136
+ }
137
+
138
+ pub fn insert_note(
139
+ &mut self,
140
+ owner: Option<String>,
141
+ waveform: String,
142
+ freq: f32,
143
+ amp: f32,
144
+ start_time_ms: f32,
145
+ duration_ms: f32,
146
+ synth_params: HashMap<String, Value>,
147
+ note_params: HashMap<String, Value>,
148
+ automation: Option<HashMap<String, Value>>,
149
+ ) -> Vec<(usize, usize)> {
150
+ // Delegated implementation lives in notes.rs
151
+ crate::core::audio::engine::notes::insert_note_impl(
152
+ self,
153
+ owner,
154
+ waveform,
155
+ freq,
156
+ amp,
157
+ start_time_ms,
158
+ duration_ms,
159
+ synth_params,
160
+ note_params,
161
+ automation,
162
+ )
163
+ }
164
+
165
+ pub fn record_last_note_range(
166
+ &mut self,
167
+ owner: &str,
168
+ start_sample: usize,
169
+ total_samples: usize,
170
+ ) {
171
+ self.last_notes
172
+ .entry(owner.to_string())
173
+ .or_default()
174
+ .push((start_sample, total_samples));
175
+ }
176
+
177
+ // helper extraction functions left in this struct for now
178
+ pub(crate) fn extract_f32(&self, map: &HashMap<String, Value>, key: &str) -> Option<f32> {
179
+ match map.get(key) {
180
+ Some(Value::Number(n)) => Some(*n),
181
+ Some(Value::String(s)) => s.parse::<f32>().ok(),
182
+ Some(Value::Boolean(b)) => Some(if *b { 1.0 } else { 0.0 }),
183
+ _ => None,
184
+ }
185
+ }
186
+
187
+ pub(crate) fn extract_boolean(&self, map: &HashMap<String, Value>, key: &str) -> Option<bool> {
188
+ match map.get(key) {
189
+ Some(Value::Boolean(b)) => Some(*b),
190
+ Some(Value::Number(n)) => Some(*n != 0.0),
191
+ Some(Value::Identifier(s)) => {
192
+ if s == "true" {
193
+ Some(true)
194
+ } else if s == "false" {
195
+ Some(false)
196
+ } else {
197
+ None
198
+ }
199
+ }
200
+ Some(Value::String(s)) => {
201
+ if s == "true" {
202
+ Some(true)
203
+ } else if s == "false" {
204
+ Some(false)
205
+ } else {
206
+ None
207
+ }
208
+ }
209
+ _ => None,
210
+ }
211
+ }
212
+ }
213
+
214
+ // Parse simple musical fraction strings like "1/16" into seconds using bpm
215
+ pub fn parse_fraction_to_seconds(s: &str, bpm: f32) -> Option<f32> {
216
+ let trimmed = s.trim();
217
+ if let Some((num, den)) = trimmed.split_once('/') {
218
+ if let (Ok(n), Ok(d)) = (num.parse::<f32>(), den.parse::<f32>()) {
219
+ if d != 0.0 {
220
+ let beats = n / d; // e.g. 1/16 -> 0.0625 beats
221
+ let secs_per_beat = 60.0 / bpm.max(1.0);
222
+ return Some(beats * secs_per_beat);
223
+ }
224
+ }
225
+ }
226
+ None
227
+ }
228
+
229
+ // Convert a devalang_types::Duration to seconds using bpm when relevant.
230
+ pub fn duration_to_seconds(d: &devalang_types::Duration, bpm: f32) -> Option<f32> {
231
+ use devalang_types::Duration as D;
232
+ match d {
233
+ D::Number(s) => Some(*s),
234
+ D::Beat(frac) | D::Identifier(frac) => parse_fraction_to_seconds(frac, bpm),
235
+ _ => None,
236
+ }
237
+ }
@@ -0,0 +1,169 @@
1
+ use crate::core::audio::engine::driver::MidiNoteEvent;
2
+ use std::fs::File;
3
+
4
+ pub fn generate_midi_file_impl(
5
+ midi_events: &Vec<MidiNoteEvent>,
6
+ output_path: &String,
7
+ bpm: Option<f32>,
8
+ tpqn: Option<u16>,
9
+ ) -> Result<(), String> {
10
+ use midly::num::{u7, u15, u24};
11
+ use midly::{
12
+ Format, Header, MetaMessage, MidiMessage, Smf, Timing, TrackEvent, TrackEventKind,
13
+ };
14
+
15
+ if midi_events.is_empty() {
16
+ return Ok(());
17
+ }
18
+
19
+ let bpm = bpm.unwrap_or(120.0_f32);
20
+ let tpqn: u16 = tpqn.unwrap_or(480u16);
21
+ let header = Header {
22
+ format: Format::SingleTrack,
23
+ timing: Timing::Metrical(u15::from(tpqn)),
24
+ };
25
+
26
+ #[derive(Clone)]
27
+ struct AbsEvent {
28
+ tick: u64,
29
+ kind: TrackEventKind<'static>,
30
+ }
31
+
32
+ let mut abs_events: Vec<AbsEvent> = Vec::new();
33
+ let microsecs_per_quarter = (60_000_000.0 / bpm) as u32;
34
+ abs_events.push(AbsEvent {
35
+ tick: 0,
36
+ kind: TrackEventKind::Meta(MetaMessage::Tempo(u24::from(microsecs_per_quarter))),
37
+ });
38
+
39
+ for ev in midi_events {
40
+ let start_secs = (ev.start_ms as f32) / 1000.0;
41
+ let dur_secs = (ev.duration_ms as f32) / 1000.0;
42
+ let start_ticks_f = start_secs * (bpm / 60.0) * (tpqn as f32);
43
+ let dur_ticks_f = dur_secs * (bpm / 60.0) * (tpqn as f32);
44
+ let start_tick = start_ticks_f.max(0.0).round() as u64;
45
+ let off_tick = (start_ticks_f + dur_ticks_f).max(start_tick as f32).round() as u64;
46
+
47
+ let key = u7::from(ev.key.min(127));
48
+ let vel = u7::from(ev.vel.min(127));
49
+
50
+ abs_events.push(AbsEvent {
51
+ tick: start_tick,
52
+ kind: TrackEventKind::Midi {
53
+ channel: (ev.channel as u8).into(),
54
+ message: MidiMessage::NoteOn { key, vel },
55
+ },
56
+ });
57
+
58
+ abs_events.push(AbsEvent {
59
+ tick: off_tick,
60
+ kind: TrackEventKind::Midi {
61
+ channel: (ev.channel as u8).into(),
62
+ message: MidiMessage::NoteOff {
63
+ key,
64
+ vel: u7::from(0),
65
+ },
66
+ },
67
+ });
68
+ }
69
+
70
+ let max_tick = abs_events.iter().map(|e| e.tick).max().unwrap_or(0);
71
+ abs_events.push(AbsEvent {
72
+ tick: max_tick + 0,
73
+ kind: TrackEventKind::Meta(MetaMessage::EndOfTrack),
74
+ });
75
+ abs_events.sort_by_key(|e| e.tick);
76
+
77
+ let mut track: Vec<TrackEvent> = Vec::new();
78
+ let mut last_tick: u64 = 0;
79
+ for e in abs_events {
80
+ let delta = (e.tick - last_tick) as u32;
81
+ track.push(TrackEvent {
82
+ delta: delta.into(),
83
+ kind: e.kind,
84
+ });
85
+ last_tick = e.tick;
86
+ }
87
+
88
+ let smf = Smf {
89
+ header,
90
+ tracks: vec![track],
91
+ };
92
+
93
+ if let Ok(mut file) = File::create(output_path) {
94
+ if let Err(e) = smf.write_std(&mut file) {
95
+ return Err(format!("Error writing MIDI file: {}", e));
96
+ }
97
+ } else {
98
+ return Err(format!("Cannot create MIDI file at {}", output_path));
99
+ }
100
+
101
+ Ok(())
102
+ }
103
+
104
+ pub fn generate_wav_file_impl(
105
+ buffer: &mut Vec<i16>,
106
+ output_dir: &String,
107
+ audio_format: Option<String>,
108
+ sample_rate: Option<u32>,
109
+ ) -> Result<(), String> {
110
+ if buffer.len() % (crate::core::audio::engine::CHANNELS as usize) != 0 {
111
+ buffer.push(0);
112
+ }
113
+
114
+ let sr = sample_rate.unwrap_or(crate::core::audio::engine::SAMPLE_RATE);
115
+ let format_str = audio_format.unwrap_or_else(|| "Wav16".to_string());
116
+ let fmt_low = format_str.to_lowercase();
117
+
118
+ match fmt_low.as_str() {
119
+ "wav16" => {
120
+ let spec = hound::WavSpec {
121
+ channels: crate::core::audio::engine::CHANNELS,
122
+ sample_rate: sr,
123
+ bits_per_sample: 16,
124
+ sample_format: hound::SampleFormat::Int,
125
+ };
126
+
127
+ let mut writer = hound::WavWriter::create(output_dir, spec)
128
+ .map_err(|e| format!("Error creating WAV file: {}", e))?;
129
+
130
+ for sample in buffer.iter() {
131
+ writer
132
+ .write_sample(*sample)
133
+ .map_err(|e| format!("Error writing sample: {:?}", e))?;
134
+ }
135
+
136
+ writer
137
+ .finalize()
138
+ .map_err(|e| format!("Error finalizing WAV: {:?}", e))?;
139
+ }
140
+ "wav24" | "wav32" => {
141
+ let bits = if fmt_low.contains("24") { 24 } else { 32 };
142
+ let spec = hound::WavSpec {
143
+ channels: crate::core::audio::engine::CHANNELS,
144
+ sample_rate: sr,
145
+ bits_per_sample: bits,
146
+ sample_format: hound::SampleFormat::Int,
147
+ };
148
+
149
+ let mut writer = hound::WavWriter::create(output_dir, spec)
150
+ .map_err(|e| format!("Error creating WAV file: {}", e))?;
151
+
152
+ for &s in buffer.iter() {
153
+ let v32 = (s as i32) << (bits - 16);
154
+ writer
155
+ .write_sample(v32)
156
+ .map_err(|e| format!("Error writing sample: {:?}", e))?;
157
+ }
158
+
159
+ writer
160
+ .finalize()
161
+ .map_err(|e| format!("Error finalizing WAV: {:?}", e))?;
162
+ }
163
+ _ => {
164
+ return Err(format!("Unsupported audio format: {}", format_str));
165
+ }
166
+ }
167
+
168
+ Ok(())
169
+ }