@devaloop/devalang 0.0.1-alpha.14 → 0.0.1-alpha.16

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 (177) hide show
  1. package/.devalang +10 -8
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +32 -15
  5. package/docs/CHANGELOG.md +93 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +2 -2
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +42 -0
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +39 -25
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +16 -0
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -456
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -8
  38. package/rust/common/cdn.rs +3 -6
  39. package/rust/common/mod.rs +3 -3
  40. package/rust/common/sso.rs +3 -6
  41. package/rust/config/driver.rs +118 -94
  42. package/rust/config/loader.rs +165 -156
  43. package/rust/config/mod.rs +4 -2
  44. package/rust/config/settings.rs +91 -0
  45. package/rust/config/stats.rs +257 -0
  46. package/rust/core/audio/engine.rs +696 -518
  47. package/rust/core/audio/evaluator.rs +263 -31
  48. package/rust/core/audio/interpreter/arrow_call.rs +198 -161
  49. package/rust/core/audio/interpreter/automate.rs +18 -0
  50. package/rust/core/audio/interpreter/call.rs +98 -95
  51. package/rust/core/audio/interpreter/condition.rs +70 -71
  52. package/rust/core/audio/interpreter/driver.rs +487 -198
  53. package/rust/core/audio/interpreter/function.rs +26 -21
  54. package/rust/core/audio/interpreter/let_.rs +38 -19
  55. package/rust/core/audio/interpreter/load.rs +18 -18
  56. package/rust/core/audio/interpreter/loop_.rs +113 -73
  57. package/rust/core/audio/interpreter/mod.rs +14 -13
  58. package/rust/core/audio/interpreter/sleep.rs +27 -30
  59. package/rust/core/audio/interpreter/spawn.rs +105 -102
  60. package/rust/core/audio/interpreter/tempo.rs +19 -16
  61. package/rust/core/audio/interpreter/trigger.rs +239 -210
  62. package/rust/core/audio/loader/mod.rs +1 -1
  63. package/rust/core/audio/loader/trigger.rs +100 -97
  64. package/rust/core/audio/mod.rs +7 -6
  65. package/rust/core/audio/player.rs +64 -64
  66. package/rust/core/audio/renderer.rs +56 -53
  67. package/rust/core/audio/special/easing.rs +189 -0
  68. package/rust/core/audio/special/env.rs +43 -0
  69. package/rust/core/audio/special/math.rs +102 -0
  70. package/rust/core/audio/special/mod.rs +9 -0
  71. package/rust/core/audio/special/modulator.rs +143 -0
  72. package/rust/core/builder/mod.rs +80 -85
  73. package/rust/core/debugger/lexer.rs +27 -27
  74. package/rust/core/debugger/mod.rs +24 -23
  75. package/rust/core/debugger/module.rs +55 -47
  76. package/rust/core/debugger/preprocessor.rs +27 -27
  77. package/rust/core/debugger/store.rs +40 -39
  78. package/rust/core/error/mod.rs +80 -66
  79. package/rust/core/lexer/handler/arrow.rs +82 -31
  80. package/rust/core/lexer/handler/at.rs +21 -21
  81. package/rust/core/lexer/handler/brace.rs +41 -41
  82. package/rust/core/lexer/handler/colon.rs +21 -21
  83. package/rust/core/lexer/handler/comment.rs +30 -30
  84. package/rust/core/lexer/handler/dot.rs +21 -21
  85. package/rust/core/lexer/handler/driver.rs +337 -263
  86. package/rust/core/lexer/handler/identifier.rs +46 -42
  87. package/rust/core/lexer/handler/indent.rs +66 -66
  88. package/rust/core/lexer/handler/mod.rs +16 -16
  89. package/rust/core/lexer/handler/newline.rs +23 -23
  90. package/rust/core/lexer/handler/number.rs +31 -31
  91. package/rust/core/lexer/handler/operator.rs +46 -44
  92. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  93. package/rust/core/lexer/handler/slash.rs +21 -21
  94. package/rust/core/lexer/handler/string.rs +63 -63
  95. package/rust/core/lexer/mod.rs +54 -51
  96. package/rust/core/lexer/token.rs +97 -91
  97. package/rust/core/mod.rs +11 -11
  98. package/rust/core/parser/driver.rs +513 -408
  99. package/rust/core/parser/handler/arrow_call.rs +233 -211
  100. package/rust/core/parser/handler/at.rs +245 -162
  101. package/rust/core/parser/handler/bank.rs +94 -69
  102. package/rust/core/parser/handler/condition.rs +80 -74
  103. package/rust/core/parser/handler/dot.rs +143 -135
  104. package/rust/core/parser/handler/identifier/automate.rs +257 -0
  105. package/rust/core/parser/handler/identifier/call.rs +91 -88
  106. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  107. package/rust/core/parser/handler/identifier/function.rs +100 -92
  108. package/rust/core/parser/handler/identifier/group.rs +85 -75
  109. package/rust/core/parser/handler/identifier/let_.rs +158 -127
  110. package/rust/core/parser/handler/identifier/mod.rs +54 -52
  111. package/rust/core/parser/handler/identifier/on.rs +98 -0
  112. package/rust/core/parser/handler/identifier/print.rs +52 -0
  113. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  114. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  115. package/rust/core/parser/handler/identifier/synth.rs +65 -65
  116. package/rust/core/parser/handler/loop_.rs +170 -72
  117. package/rust/core/parser/handler/mod.rs +8 -8
  118. package/rust/core/parser/handler/tempo.rs +53 -47
  119. package/rust/core/parser/mod.rs +4 -4
  120. package/rust/core/parser/statement.rs +142 -108
  121. package/rust/core/plugin/loader.rs +123 -48
  122. package/rust/core/plugin/mod.rs +2 -1
  123. package/rust/core/plugin/runner.rs +296 -0
  124. package/rust/core/preprocessor/loader.rs +515 -326
  125. package/rust/core/preprocessor/mod.rs +4 -4
  126. package/rust/core/preprocessor/module.rs +60 -58
  127. package/rust/core/preprocessor/processor.rs +99 -101
  128. package/rust/core/preprocessor/resolver/bank.rs +51 -49
  129. package/rust/core/preprocessor/resolver/call.rs +100 -100
  130. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  131. package/rust/core/preprocessor/resolver/driver.rs +310 -278
  132. package/rust/core/preprocessor/resolver/function.rs +69 -78
  133. package/rust/core/preprocessor/resolver/group.rs +96 -91
  134. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  135. package/rust/core/preprocessor/resolver/loop_.rs +320 -91
  136. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  137. package/rust/core/preprocessor/resolver/spawn.rs +76 -92
  138. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  139. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  140. package/rust/core/preprocessor/resolver/trigger.rs +113 -116
  141. package/rust/core/preprocessor/resolver/value.rs +81 -87
  142. package/rust/core/shared/bank.rs +1 -1
  143. package/rust/core/shared/duration.rs +9 -9
  144. package/rust/core/shared/mod.rs +3 -3
  145. package/rust/core/shared/value.rs +35 -32
  146. package/rust/core/store/function.rs +34 -34
  147. package/rust/core/store/global.rs +55 -38
  148. package/rust/core/store/mod.rs +5 -5
  149. package/rust/core/store/variable.rs +37 -34
  150. package/rust/core/utils/mod.rs +2 -2
  151. package/rust/core/utils/path.rs +37 -31
  152. package/rust/core/utils/validation.rs +35 -37
  153. package/rust/installer/addon.rs +84 -80
  154. package/rust/installer/bank.rs +62 -65
  155. package/rust/installer/mod.rs +5 -5
  156. package/rust/installer/plugin.rs +54 -55
  157. package/rust/installer/utils.rs +56 -56
  158. package/rust/lib.rs +156 -164
  159. package/rust/main.rs +250 -145
  160. package/rust/utils/error.rs +200 -0
  161. package/rust/utils/file.rs +38 -35
  162. package/rust/utils/first_usage.rs +76 -0
  163. package/rust/utils/logger.rs +195 -139
  164. package/rust/utils/mod.rs +9 -50
  165. package/rust/utils/signature.rs +19 -17
  166. package/rust/utils/spinner.rs +22 -19
  167. package/rust/utils/telemetry.rs +292 -0
  168. package/rust/utils/watcher.rs +34 -33
  169. package/templates/minimal/README.md +97 -121
  170. package/templates/welcome/README.md +97 -121
  171. package/typescript/bin/index.ts +19 -5
  172. package/typescript/index.ts +3 -1
  173. package/typescript/scripts/postbuild.ts +10 -6
  174. package/typescript/scripts/postinstall.ts +56 -0
  175. package/typescript/scripts/version/bump.ts +0 -1
  176. package/typescript/scripts/version/index.ts +0 -1
  177. package/out-tsc/bin/devalang.exe +0 -0
@@ -1,21 +1,26 @@
1
- use crate::core::{
2
- parser::statement::{ Statement, StatementKind },
3
- store::function::{ FunctionDef, FunctionTable },
4
- };
5
-
6
- pub fn interprete_function_statement(
7
- stmt: &Statement,
8
- functions_table: &mut FunctionTable
9
- ) -> Option<FunctionTable> {
10
- if let StatementKind::Function { name, parameters, body } = &stmt.kind {
11
- functions_table.add_function(FunctionDef {
12
- name: name.clone(),
13
- parameters: parameters.clone(),
14
- body: body.clone(),
15
- });
16
-
17
- return Some(functions_table.clone());
18
- }
19
-
20
- None
21
- }
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ store::function::{FunctionDef, FunctionTable},
4
+ };
5
+
6
+ pub fn interprete_function_statement(
7
+ stmt: &Statement,
8
+ functions_table: &mut FunctionTable,
9
+ ) -> Option<FunctionTable> {
10
+ if let StatementKind::Function {
11
+ name,
12
+ parameters,
13
+ body,
14
+ } = &stmt.kind
15
+ {
16
+ functions_table.add_function(FunctionDef {
17
+ name: name.clone(),
18
+ parameters: parameters.clone(),
19
+ body: body.clone(),
20
+ });
21
+
22
+ return Some(functions_table.clone());
23
+ }
24
+
25
+ None
26
+ }
@@ -1,19 +1,38 @@
1
- use crate::core::{
2
- audio::engine::AudioEngine,
3
- parser::statement::{ Statement, StatementKind },
4
- shared::value::Value,
5
- store::variable::VariableTable,
6
- };
7
-
8
- pub fn interprete_let_statement(
9
- stmt: &Statement,
10
- variable_table: &mut VariableTable
11
- ) -> Option<VariableTable> {
12
- if let StatementKind::Let { name } = &stmt.kind {
13
- variable_table.set(name.to_string(), stmt.value.clone());
14
-
15
- return Some(variable_table.clone())
16
- }
17
-
18
- None
19
- }
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ shared::value::Value,
4
+ store::variable::VariableTable,
5
+ };
6
+
7
+ pub fn interprete_let_statement(
8
+ stmt: &Statement,
9
+ variable_table: &mut VariableTable,
10
+ ) -> Option<VariableTable> {
11
+ if let StatementKind::Let { name } = &stmt.kind {
12
+ // If RHS is a string and looks like an expression, evaluate it
13
+ let evaluated = match &stmt.value {
14
+ Value::String(s) if s.contains("$env") || s.contains("$math") => {
15
+ // We don't have direct env here; use defaults or infer from table
16
+ let bpm = if let Some(Value::Number(n)) = variable_table.get("bpm") {
17
+ *n
18
+ } else {
19
+ 120.0
20
+ };
21
+ // Try to infer beat from time-based variables if any, else 0.0
22
+ let beat = if let Some(Value::Number(n)) = variable_table.get("beat") {
23
+ *n
24
+ } else {
25
+ 0.0
26
+ };
27
+ crate::core::audio::evaluator::evaluate_rhs_into_value(s, variable_table, bpm, beat)
28
+ }
29
+ other => other.clone(),
30
+ };
31
+
32
+ variable_table.set(name.to_string(), evaluated);
33
+
34
+ return Some(variable_table.clone());
35
+ }
36
+
37
+ None
38
+ }
@@ -1,18 +1,18 @@
1
- use crate::core::{
2
- parser::statement::{ Statement, StatementKind },
3
- shared::value::Value,
4
- store::variable::VariableTable,
5
- };
6
-
7
- pub fn interprete_load_statement(
8
- stmt: &Statement,
9
- variable_table: &mut VariableTable
10
- ) -> Option<VariableTable> {
11
- if let StatementKind::Load { source, alias } = &stmt.kind {
12
- variable_table.set(alias.to_string(), Value::Sample(source.clone()));
13
-
14
- return Some(variable_table.clone());
15
- }
16
-
17
- None
18
- }
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ shared::value::Value,
4
+ store::variable::VariableTable,
5
+ };
6
+
7
+ pub fn interprete_load_statement(
8
+ stmt: &Statement,
9
+ variable_table: &mut VariableTable,
10
+ ) -> Option<VariableTable> {
11
+ if let StatementKind::Load { source, alias } = &stmt.kind {
12
+ variable_table.set(alias.to_string(), Value::Sample(source.clone()));
13
+
14
+ return Some(variable_table.clone());
15
+ }
16
+
17
+ None
18
+ }
@@ -1,73 +1,113 @@
1
- use crate::core::{
2
- audio::{ engine::AudioEngine, interpreter::driver::execute_audio_block },
3
- parser::statement::{ Statement, StatementKind },
4
- shared::{ duration::Duration, value::Value },
5
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
6
- };
7
-
8
- pub fn interprete_loop_statement(
9
- stmt: &Statement,
10
- audio_engine: &mut AudioEngine,
11
- global_store: &GlobalStore,
12
- variable_table: &VariableTable,
13
- functions_table: &FunctionTable,
14
- base_bpm: f32,
15
- base_duration: f32,
16
- max_end_time: f32,
17
- cursor_time: f32
18
- ) -> (f32, f32) {
19
- if let Value::Map(loop_value) = &stmt.value {
20
- let loop_count = match loop_value.get("iterator") {
21
- Some(Value::Number(n)) => *n as usize,
22
- Some(Value::Identifier(ident)) => {
23
- if let Some(Value::Number(n)) = variable_table.get(ident) {
24
- *n as usize
25
- } else {
26
- eprintln!("❌ Loop iterator must be a number, found: {:?}", ident);
27
- return (max_end_time, cursor_time);
28
- }
29
- }
30
- _ => {
31
- eprintln!(
32
- "❌ Loop iterator must be a number, found: {:?}",
33
- loop_value.get("iterator")
34
- );
35
- return (max_end_time, cursor_time);
36
- }
37
- };
38
-
39
- let loop_body = match loop_value.get("body") {
40
- Some(Value::Block(body)) => body.clone(),
41
- _ => {
42
- eprintln!("❌ Loop body must be a block, found: {:?}", loop_value.get("body"));
43
- return (max_end_time, cursor_time);
44
- }
45
- };
46
-
47
- let mut engine = audio_engine;
48
- let mut cur_time = cursor_time;
49
- let mut max_time = max_end_time;
50
-
51
- for i in 0..loop_count {
52
- let (block_end_time, cursor_time) = execute_audio_block(
53
- &mut engine,
54
- global_store,
55
- variable_table.clone(),
56
- functions_table.clone(),
57
- loop_body.clone(),
58
- base_bpm,
59
- base_duration,
60
- max_time,
61
- cur_time
62
- );
63
-
64
- cur_time = block_end_time;
65
- max_time = max_time.max(cur_time);
66
- }
67
-
68
- return (max_time, cur_time);
69
- }
70
-
71
- eprintln!("❌ Loop statement value is not a map");
72
- (max_end_time, cursor_time)
73
- }
1
+ use crate::core::{
2
+ audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
3
+ parser::statement::Statement,
4
+ shared::value::Value,
5
+ store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
6
+ };
7
+
8
+ pub fn interprete_loop_statement(
9
+ stmt: &Statement,
10
+ audio_engine: &mut AudioEngine,
11
+ global_store: &GlobalStore,
12
+ variable_table: &VariableTable,
13
+ functions_table: &FunctionTable,
14
+ base_bpm: f32,
15
+ base_duration: f32,
16
+ max_end_time: f32,
17
+ cursor_time: f32,
18
+ ) -> (f32, f32) {
19
+ if let Value::Map(loop_value) = &stmt.value {
20
+ // Foreach form: { foreach: Identifier(name), array: Array([...]), body: Block }
21
+ if let (
22
+ Some(Value::Identifier(var_name)),
23
+ Some(Value::Array(items)),
24
+ Some(Value::Block(loop_body)),
25
+ ) = (
26
+ loop_value.get("foreach"),
27
+ loop_value.get("array"),
28
+ loop_value.get("body"),
29
+ ) {
30
+ let mut engine = audio_engine;
31
+ let mut cur_time = cursor_time;
32
+ let mut max_time = max_end_time;
33
+
34
+ for item in items {
35
+ let mut scoped_vars = variable_table.clone();
36
+ scoped_vars.set(var_name.clone(), item.clone());
37
+
38
+ let (block_end_time, new_cursor) = execute_audio_block(
39
+ &mut engine,
40
+ global_store,
41
+ scoped_vars,
42
+ functions_table.clone(),
43
+ &loop_body,
44
+ base_bpm,
45
+ base_duration,
46
+ max_time,
47
+ cur_time,
48
+ );
49
+
50
+ cur_time = new_cursor.max(block_end_time);
51
+ max_time = max_time.max(cur_time);
52
+ }
53
+
54
+ return (max_time, cur_time);
55
+ }
56
+
57
+ let loop_count = match loop_value.get("iterator") {
58
+ Some(Value::Number(n)) => *n as usize,
59
+ Some(Value::Identifier(ident)) => {
60
+ if let Some(Value::Number(n)) = variable_table.get(ident) {
61
+ *n as usize
62
+ } else {
63
+ eprintln!("❌ Loop iterator must be a number, found: {:?}", ident);
64
+ return (max_end_time, cursor_time);
65
+ }
66
+ }
67
+ _ => {
68
+ eprintln!(
69
+ "❌ Loop iterator must be a number, found: {:?}",
70
+ loop_value.get("iterator")
71
+ );
72
+ return (max_end_time, cursor_time);
73
+ }
74
+ };
75
+
76
+ let loop_body = match loop_value.get("body") {
77
+ Some(Value::Block(body)) => body.clone(),
78
+ _ => {
79
+ eprintln!(
80
+ "❌ Loop body must be a block, found: {:?}",
81
+ loop_value.get("body")
82
+ );
83
+ return (max_end_time, cursor_time);
84
+ }
85
+ };
86
+
87
+ let mut engine = audio_engine;
88
+ let mut cur_time = cursor_time;
89
+ let mut max_time = max_end_time;
90
+
91
+ for _ in 0..loop_count {
92
+ let (block_end_time, new_cursor) = execute_audio_block(
93
+ &mut engine,
94
+ global_store,
95
+ variable_table.clone(),
96
+ functions_table.clone(),
97
+ &loop_body,
98
+ base_bpm,
99
+ base_duration,
100
+ max_time,
101
+ cur_time,
102
+ );
103
+
104
+ cur_time = new_cursor.max(block_end_time);
105
+ max_time = max_time.max(cur_time);
106
+ }
107
+
108
+ return (max_time, cur_time);
109
+ }
110
+
111
+ eprintln!("❌ Loop statement value is not a map");
112
+ (max_end_time, cursor_time)
113
+ }
@@ -1,13 +1,14 @@
1
- pub mod driver;
2
-
3
- pub mod load;
4
- pub mod let_;
5
- pub mod tempo;
6
- pub mod trigger;
7
- pub mod spawn;
8
- pub mod sleep;
9
- pub mod loop_;
10
- pub mod call;
11
- pub mod condition;
12
- pub mod arrow_call;
13
- pub mod function;
1
+ pub mod driver;
2
+
3
+ pub mod arrow_call;
4
+ pub mod automate;
5
+ pub mod call;
6
+ pub mod condition;
7
+ pub mod function;
8
+ pub mod let_;
9
+ pub mod load;
10
+ pub mod loop_;
11
+ pub mod sleep;
12
+ pub mod spawn;
13
+ pub mod tempo;
14
+ pub mod trigger;
@@ -1,30 +1,27 @@
1
- use crate::core::{
2
- audio::{ engine::AudioEngine, loader::trigger::load_trigger },
3
- parser::statement::{ Statement, StatementKind },
4
- shared::{ duration::Duration, value::Value },
5
- store::variable::VariableTable,
6
- };
7
-
8
- pub fn interprete_sleep_statement(
9
- stmt: &Statement,
10
- cursor_time: f32,
11
- max_end_time: f32,
12
- ) -> (f32, f32) {
13
- let duration_secs = match &stmt.value {
14
- Value::Number(ms) => *ms / 1000.0,
15
- Value::String(s) if s.ends_with("ms") => {
16
- s.trim_end_matches("ms").parse::<f32>().map(|ms| ms / 1000.0).unwrap_or_else(|_| {
17
- eprintln!("❌ Invalid sleep value (ms): {}", s);
18
- 0.0
19
- })
20
- }
21
- other => {
22
- eprintln!("❌ Invalid sleep value: {:?}", other);
23
- 0.0
24
- }
25
- };
26
-
27
- let new_cursor = cursor_time + duration_secs;
28
- let new_max = max_end_time.max(new_cursor);
29
- (new_cursor, new_max)
30
- }
1
+ use crate::core::{parser::statement::Statement, shared::value::Value};
2
+
3
+ pub fn interprete_sleep_statement(
4
+ stmt: &Statement,
5
+ cursor_time: f32,
6
+ max_end_time: f32,
7
+ ) -> (f32, f32) {
8
+ let duration_secs = match &stmt.value {
9
+ Value::Number(ms) => *ms / 1000.0,
10
+ Value::String(s) if s.ends_with("ms") => s
11
+ .trim_end_matches("ms")
12
+ .parse::<f32>()
13
+ .map(|ms| ms / 1000.0)
14
+ .unwrap_or_else(|_| {
15
+ eprintln!("❌ Invalid sleep value (ms): {}", s);
16
+ 0.0
17
+ }),
18
+ other => {
19
+ eprintln!("❌ Invalid sleep value: {:?}", other);
20
+ 0.0
21
+ }
22
+ };
23
+
24
+ let new_cursor = cursor_time + duration_secs;
25
+ let new_max = max_end_time.max(new_cursor);
26
+ (new_cursor, new_max)
27
+ }
@@ -1,102 +1,105 @@
1
- use crate::core::{
2
- audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
3
- parser::statement::{Statement, StatementKind},
4
- shared::value::Value,
5
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
6
- };
7
-
8
- pub fn interprete_spawn_statement(
9
- stmt: &Statement,
10
- audio_engine: &mut AudioEngine,
11
- variable_table: &VariableTable,
12
- functions: &FunctionTable,
13
- global_store: &GlobalStore,
14
- base_bpm: f32,
15
- base_duration: f32,
16
- max_end_time: f32,
17
- cursor_time: f32,
18
- ) -> (f32, f32) {
19
- match &stmt.kind {
20
- StatementKind::Spawn { name, args } => {
21
- let mut local_engine = AudioEngine::new(audio_engine.module_name.clone());
22
-
23
- // Function case
24
- if let Some(func) = functions.functions.get(name) {
25
- if func.parameters.len() != args.len() {
26
- eprintln!(
27
- "❌ Function '{}' expects {} args, got {}",
28
- name,
29
- func.parameters.len(),
30
- args.len()
31
- );
32
- return (max_end_time, cursor_time);
33
- }
34
-
35
- let mut local_vars = VariableTable::with_parent(variable_table.clone());
36
- for (param, arg) in func.parameters.iter().zip(args) {
37
- local_vars.set(param.clone(), arg.clone());
38
- }
39
-
40
- let (spawn_max, _) = execute_audio_block(
41
- &mut local_engine,
42
- global_store,
43
- local_vars,
44
- functions.clone(),
45
- func.body.clone(),
46
- base_bpm,
47
- base_duration,
48
- 0.0,
49
- 0.0,
50
- );
51
-
52
- audio_engine.merge_with(local_engine);
53
- return (spawn_max.max(max_end_time), cursor_time);
54
- }
55
-
56
- // Group case
57
- if let Some(group_stmt) = find_group(name, variable_table, global_store) {
58
- if let Value::Map(map) = &group_stmt.value {
59
- if let Some(Value::Block(body)) = map.get("body") {
60
- let (spawn_max, _) = execute_audio_block(
61
- &mut local_engine,
62
- global_store,
63
- variable_table.clone(),
64
- functions.clone(),
65
- body.clone(),
66
- base_bpm,
67
- base_duration,
68
- 0.0,
69
- 0.0,
70
- );
71
- audio_engine.merge_with(local_engine);
72
- return (spawn_max.max(max_end_time), cursor_time);
73
- }
74
- }
75
- }
76
-
77
- eprintln!("❌ Function or group '{}' not found", name);
78
- }
79
-
80
- _ => eprintln!("❌ interprete_spawn_statement expected Spawn, got {:?}", stmt.kind),
81
- }
82
-
83
- (max_end_time, cursor_time)
84
- }
85
-
86
- fn find_group<'a>(
87
- name: &str,
88
- variable_table: &'a VariableTable,
89
- global_store: &'a GlobalStore,
90
- ) -> Option<&'a Statement> {
91
- if let Some(Value::Statement(stmt_box)) = variable_table.get(name) {
92
- if let StatementKind::Group = stmt_box.kind {
93
- return Some(stmt_box);
94
- }
95
- }
96
- if let Some(Value::Statement(stmt_box)) = global_store.variables.variables.get(name) {
97
- if let StatementKind::Group = stmt_box.kind {
98
- return Some(stmt_box);
99
- }
100
- }
101
- None
102
- }
1
+ use crate::core::{
2
+ audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
3
+ parser::statement::{Statement, StatementKind},
4
+ shared::value::Value,
5
+ store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
6
+ };
7
+
8
+ pub fn interprete_spawn_statement(
9
+ stmt: &Statement,
10
+ audio_engine: &mut AudioEngine,
11
+ variable_table: &VariableTable,
12
+ functions: &FunctionTable,
13
+ global_store: &GlobalStore,
14
+ base_bpm: f32,
15
+ base_duration: f32,
16
+ max_end_time: f32,
17
+ cursor_time: f32,
18
+ ) -> (f32, f32) {
19
+ match &stmt.kind {
20
+ StatementKind::Spawn { name, args } => {
21
+ let mut local_engine = AudioEngine::new(audio_engine.module_name.clone());
22
+
23
+ // Function case
24
+ if let Some(func) = functions.functions.get(name) {
25
+ if func.parameters.len() != args.len() {
26
+ eprintln!(
27
+ "❌ Function '{}' expects {} args, got {}",
28
+ name,
29
+ func.parameters.len(),
30
+ args.len()
31
+ );
32
+ return (max_end_time, cursor_time);
33
+ }
34
+
35
+ let mut local_vars = VariableTable::with_parent(variable_table.clone());
36
+ for (param, arg) in func.parameters.iter().zip(args) {
37
+ local_vars.set(param.clone(), arg.clone());
38
+ }
39
+
40
+ let (spawn_max, _) = execute_audio_block(
41
+ &mut local_engine,
42
+ global_store,
43
+ local_vars,
44
+ functions.clone(),
45
+ &func.body,
46
+ base_bpm,
47
+ base_duration,
48
+ 0.0,
49
+ 0.0,
50
+ );
51
+
52
+ audio_engine.merge_with(local_engine);
53
+ return (spawn_max.max(max_end_time), cursor_time);
54
+ }
55
+
56
+ // Group case
57
+ if let Some(group_stmt) = find_group(name, variable_table, global_store) {
58
+ if let Value::Map(map) = &group_stmt.value {
59
+ if let Some(Value::Block(body)) = map.get("body") {
60
+ let (spawn_max, _) = execute_audio_block(
61
+ &mut local_engine,
62
+ global_store,
63
+ variable_table.clone(),
64
+ functions.clone(),
65
+ &body,
66
+ base_bpm,
67
+ base_duration,
68
+ 0.0,
69
+ 0.0,
70
+ );
71
+ audio_engine.merge_with(local_engine);
72
+ return (spawn_max.max(max_end_time), cursor_time);
73
+ }
74
+ }
75
+ }
76
+
77
+ eprintln!("❌ Function or group '{}' not found", name);
78
+ }
79
+
80
+ _ => eprintln!(
81
+ "❌ interprete_spawn_statement expected Spawn, got {:?}",
82
+ stmt.kind
83
+ ),
84
+ }
85
+
86
+ (max_end_time, cursor_time)
87
+ }
88
+
89
+ fn find_group<'a>(
90
+ name: &str,
91
+ variable_table: &'a VariableTable,
92
+ global_store: &'a GlobalStore,
93
+ ) -> Option<&'a Statement> {
94
+ if let Some(Value::Statement(stmt_box)) = variable_table.get(name) {
95
+ if let StatementKind::Group = stmt_box.kind {
96
+ return Some(stmt_box);
97
+ }
98
+ }
99
+ if let Some(Value::Statement(stmt_box)) = global_store.variables.variables.get(name) {
100
+ if let StatementKind::Group = stmt_box.kind {
101
+ return Some(stmt_box);
102
+ }
103
+ }
104
+ None
105
+ }
@@ -1,16 +1,19 @@
1
- use crate::core::{ parser::statement::{ Statement, StatementKind }, shared::value::Value };
2
-
3
- pub fn interprete_tempo_statement(stmt: &Statement) -> Option<(f32, f32)> {
4
- if let StatementKind::Tempo = &stmt.kind {
5
- if let Value::Number(bpm) = &stmt.value {
6
- let bpm = *bpm as f32;
7
- let duration = 60.0 / bpm;
8
-
9
- return Some((bpm, duration));
10
- } else {
11
- eprintln!("❌ Invalid tempo value: {:?}", stmt.value);
12
- }
13
- }
14
-
15
- None
16
- }
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ shared::value::Value,
4
+ };
5
+
6
+ pub fn interprete_tempo_statement(stmt: &Statement) -> Option<(f32, f32)> {
7
+ if let StatementKind::Tempo = &stmt.kind {
8
+ if let Value::Number(bpm) = &stmt.value {
9
+ let bpm = *bpm as f32;
10
+ let duration = 60.0 / bpm;
11
+
12
+ return Some((bpm, duration));
13
+ } else {
14
+ eprintln!("❌ Invalid tempo value: {:?}", stmt.value);
15
+ }
16
+ }
17
+
18
+ None
19
+ }