@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,71 +1,72 @@
1
- use devalang_types::Value;
2
-
3
- use crate::core::{
4
- audio::{
5
- engine::AudioEngine, evaluator::evaluate_condition_string,
6
- interpreter::driver::execute_audio_block,
7
- },
8
- parser::statement::Statement,
9
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
10
- };
11
-
12
- pub fn interprete_condition_statement(
13
- stmt: &Statement,
14
- audio_engine: &mut AudioEngine,
15
- global_store: &GlobalStore,
16
- variable_table: &VariableTable,
17
- functions_table: &FunctionTable,
18
- base_bpm: f32,
19
- base_duration: f32,
20
- max_end_time: f32,
21
- cursor_time: f32,
22
- ) -> (f32, f32) {
23
- let cur_time = cursor_time;
24
- let max_time = max_end_time;
25
-
26
- let mut current = stmt.value.clone();
27
-
28
- loop {
29
- let Value::Map(map) = current else {
30
- break;
31
- };
32
-
33
- let should_execute = match map.get("condition") {
34
- Some(Value::Boolean(b)) => *b,
35
- Some(Value::String(expr)) => evaluate_condition_string(expr, &variable_table.clone()),
36
- Some(_) => false,
37
- None => true,
38
- };
39
-
40
- if should_execute {
41
- if let Some(Value::Block(block)) = map.get("body") {
42
- let (new_max, cursor_time) = execute_audio_block(
43
- audio_engine,
44
- global_store,
45
- variable_table.clone(),
46
- functions_table.clone(),
47
- block,
48
- base_bpm,
49
- base_duration,
50
- max_time,
51
- cur_time,
52
- );
53
- return (new_max, cursor_time);
54
- } else {
55
- break;
56
- }
57
- }
58
-
59
- // Advance to the next condition
60
- match map.get("next") {
61
- Some(Value::Map(next_map)) => {
62
- current = Value::Map(next_map.clone());
63
- }
64
- _ => {
65
- break;
66
- }
67
- }
68
- }
69
-
70
- (max_end_time, cursor_time)
71
- }
1
+ use devalang_types::Value;
2
+
3
+ use crate::core::{
4
+ audio::{
5
+ engine::AudioEngine, evaluator::evaluate_condition_string,
6
+ interpreter::driver::execute_audio_block,
7
+ },
8
+ parser::statement::Statement,
9
+ store::global::GlobalStore,
10
+ };
11
+ use devalang_types::store::{FunctionTable, VariableTable};
12
+
13
+ pub fn interprete_condition_statement(
14
+ stmt: &Statement,
15
+ audio_engine: &mut AudioEngine,
16
+ global_store: &GlobalStore,
17
+ variable_table: &VariableTable,
18
+ functions_table: &FunctionTable,
19
+ base_bpm: f32,
20
+ base_duration: f32,
21
+ max_end_time: f32,
22
+ cursor_time: f32,
23
+ ) -> (f32, f32) {
24
+ let cur_time = cursor_time;
25
+ let max_time = max_end_time;
26
+
27
+ let mut current = stmt.value.clone();
28
+
29
+ loop {
30
+ let Value::Map(map) = current else {
31
+ break;
32
+ };
33
+
34
+ let should_execute = match map.get("condition") {
35
+ Some(Value::Boolean(b)) => *b,
36
+ Some(Value::String(expr)) => evaluate_condition_string(expr, &variable_table.clone()),
37
+ Some(_) => false,
38
+ None => true,
39
+ };
40
+
41
+ if should_execute {
42
+ if let Some(Value::Block(block)) = map.get("body") {
43
+ let (new_max, cursor_time) = execute_audio_block(
44
+ audio_engine,
45
+ global_store,
46
+ variable_table.clone(),
47
+ functions_table.clone(),
48
+ block,
49
+ base_bpm,
50
+ base_duration,
51
+ max_time,
52
+ cur_time,
53
+ );
54
+ return (new_max, cursor_time);
55
+ } else {
56
+ break;
57
+ }
58
+ }
59
+
60
+ // Advance to the next condition
61
+ match map.get("next") {
62
+ Some(Value::Map(next_map)) => {
63
+ current = Value::Map(next_map.clone());
64
+ }
65
+ _ => {
66
+ break;
67
+ }
68
+ }
69
+ }
70
+
71
+ (max_end_time, cursor_time)
72
+ }
@@ -1,26 +1,24 @@
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
+ use crate::core::parser::statement::{Statement, StatementKind};
2
+ use devalang_types::store::{FunctionDef, FunctionTable};
3
+
4
+ pub fn interprete_function_statement(
5
+ stmt: &Statement,
6
+ functions_table: &mut FunctionTable,
7
+ ) -> Option<FunctionTable> {
8
+ if let StatementKind::Function {
9
+ name,
10
+ parameters,
11
+ body,
12
+ } = &stmt.kind
13
+ {
14
+ functions_table.add_function(FunctionDef {
15
+ name: name.clone(),
16
+ parameters: parameters.clone(),
17
+ body: body.clone(),
18
+ });
19
+
20
+ return Some(functions_table.clone());
21
+ }
22
+
23
+ None
24
+ }
@@ -1,38 +1,36 @@
1
- use crate::core::{
2
- parser::statement::{Statement, StatementKind},
3
- store::variable::VariableTable,
4
- };
5
- use devalang_types::Value;
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
+ use crate::core::parser::statement::{Statement, StatementKind};
2
+ use devalang_types::Value;
3
+ use devalang_types::store::VariableTable;
4
+
5
+ pub fn interprete_let_statement(
6
+ stmt: &Statement,
7
+ variable_table: &mut VariableTable,
8
+ ) -> Option<VariableTable> {
9
+ if let StatementKind::Let { name } = &stmt.kind {
10
+ // If RHS is a string and looks like an expression, evaluate it
11
+ let evaluated = match &stmt.value {
12
+ Value::String(s) if s.contains("$env") || s.contains("$math") => {
13
+ // We don't have direct env here; use defaults or infer from table
14
+ let bpm = if let Some(Value::Number(n)) = variable_table.get("bpm") {
15
+ *n
16
+ } else {
17
+ 120.0
18
+ };
19
+ // Try to infer beat from time-based variables if any, else 0.0
20
+ let beat = if let Some(Value::Number(n)) = variable_table.get("beat") {
21
+ *n
22
+ } else {
23
+ 0.0
24
+ };
25
+ crate::core::audio::evaluator::evaluate_rhs_into_value(s, variable_table, bpm, beat)
26
+ }
27
+ other => other.clone(),
28
+ };
29
+
30
+ variable_table.set(name.to_string(), evaluated);
31
+
32
+ return Some(variable_table.clone());
33
+ }
34
+
35
+ None
36
+ }
@@ -1,19 +1,17 @@
1
- use devalang_types::Value;
2
-
3
- use crate::core::{
4
- parser::statement::{Statement, StatementKind},
5
- store::variable::VariableTable,
6
- };
7
-
8
- pub fn interprete_load_statement(
9
- stmt: &Statement,
10
- variable_table: &mut VariableTable,
11
- ) -> Option<VariableTable> {
12
- if let StatementKind::Load { source, alias } = &stmt.kind {
13
- variable_table.set(alias.to_string(), Value::Sample(source.clone()));
14
-
15
- return Some(variable_table.clone());
16
- }
17
-
18
- None
19
- }
1
+ use devalang_types::Value;
2
+
3
+ use crate::core::parser::statement::{Statement, StatementKind};
4
+ use devalang_types::store::VariableTable;
5
+
6
+ pub fn interprete_load_statement(
7
+ stmt: &Statement,
8
+ variable_table: &mut VariableTable,
9
+ ) -> Option<VariableTable> {
10
+ if let StatementKind::Load { source, alias } = &stmt.kind {
11
+ variable_table.set(alias.to_string(), Value::Sample(source.clone()));
12
+
13
+ return Some(variable_table.clone());
14
+ }
15
+
16
+ None
17
+ }
@@ -1,114 +1,115 @@
1
- use devalang_types::Value;
2
-
3
- use crate::core::{
4
- audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
5
- parser::statement::Statement,
6
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
7
- };
8
-
9
- pub fn interprete_loop_statement(
10
- stmt: &Statement,
11
- audio_engine: &mut AudioEngine,
12
- global_store: &GlobalStore,
13
- variable_table: &VariableTable,
14
- functions_table: &FunctionTable,
15
- base_bpm: f32,
16
- base_duration: f32,
17
- max_end_time: f32,
18
- cursor_time: f32,
19
- ) -> (f32, f32) {
20
- if let Value::Map(loop_value) = &stmt.value {
21
- // Foreach form: { foreach: Identifier(name), array: Array([...]), body: Block }
22
- if let (
23
- Some(Value::Identifier(var_name)),
24
- Some(Value::Array(items)),
25
- Some(Value::Block(loop_body)),
26
- ) = (
27
- loop_value.get("foreach"),
28
- loop_value.get("array"),
29
- loop_value.get("body"),
30
- ) {
31
- let engine = audio_engine;
32
- let mut cur_time = cursor_time;
33
- let mut max_time = max_end_time;
34
-
35
- for item in items {
36
- let mut scoped_vars = variable_table.clone();
37
- scoped_vars.set(var_name.clone(), item.clone());
38
-
39
- let (block_end_time, new_cursor) = execute_audio_block(
40
- engine,
41
- global_store,
42
- scoped_vars,
43
- functions_table.clone(),
44
- loop_body,
45
- base_bpm,
46
- base_duration,
47
- max_time,
48
- cur_time,
49
- );
50
-
51
- cur_time = new_cursor.max(block_end_time);
52
- max_time = max_time.max(cur_time);
53
- }
54
-
55
- return (max_time, cur_time);
56
- }
57
-
58
- let loop_count = match loop_value.get("iterator") {
59
- Some(Value::Number(n)) => *n as usize,
60
- Some(Value::Identifier(ident)) => {
61
- if let Some(Value::Number(n)) = variable_table.get(ident) {
62
- *n as usize
63
- } else {
64
- eprintln!("❌ Loop iterator must be a number, found: {:?}", ident);
65
- return (max_end_time, cursor_time);
66
- }
67
- }
68
- _ => {
69
- eprintln!(
70
- "❌ Loop iterator must be a number, found: {:?}",
71
- loop_value.get("iterator")
72
- );
73
- return (max_end_time, cursor_time);
74
- }
75
- };
76
-
77
- let loop_body = match loop_value.get("body") {
78
- Some(Value::Block(body)) => body.clone(),
79
- _ => {
80
- eprintln!(
81
- "❌ Loop body must be a block, found: {:?}",
82
- loop_value.get("body")
83
- );
84
- return (max_end_time, cursor_time);
85
- }
86
- };
87
-
88
- let engine = audio_engine;
89
- let mut cur_time = cursor_time;
90
- let mut max_time = max_end_time;
91
-
92
- for _ in 0..loop_count {
93
- let (block_end_time, new_cursor) = execute_audio_block(
94
- engine,
95
- global_store,
96
- variable_table.clone(),
97
- functions_table.clone(),
98
- &loop_body,
99
- base_bpm,
100
- base_duration,
101
- max_time,
102
- cur_time,
103
- );
104
-
105
- cur_time = new_cursor.max(block_end_time);
106
- max_time = max_time.max(cur_time);
107
- }
108
-
109
- return (max_time, cur_time);
110
- }
111
-
112
- eprintln!("❌ Loop statement value is not a map");
113
- (max_end_time, cursor_time)
114
- }
1
+ use devalang_types::Value;
2
+
3
+ use crate::core::{
4
+ audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
5
+ parser::statement::Statement,
6
+ store::global::GlobalStore,
7
+ };
8
+ use devalang_types::store::{FunctionTable, VariableTable};
9
+
10
+ pub fn interprete_loop_statement(
11
+ stmt: &Statement,
12
+ audio_engine: &mut AudioEngine,
13
+ global_store: &GlobalStore,
14
+ variable_table: &VariableTable,
15
+ functions_table: &FunctionTable,
16
+ base_bpm: f32,
17
+ base_duration: f32,
18
+ max_end_time: f32,
19
+ cursor_time: f32,
20
+ ) -> (f32, f32) {
21
+ if let Value::Map(loop_value) = &stmt.value {
22
+ // Foreach form: { foreach: Identifier(name), array: Array([...]), body: Block }
23
+ if let (
24
+ Some(Value::Identifier(var_name)),
25
+ Some(Value::Array(items)),
26
+ Some(Value::Block(loop_body)),
27
+ ) = (
28
+ loop_value.get("foreach"),
29
+ loop_value.get("array"),
30
+ loop_value.get("body"),
31
+ ) {
32
+ let engine = audio_engine;
33
+ let mut cur_time = cursor_time;
34
+ let mut max_time = max_end_time;
35
+
36
+ for item in items {
37
+ let mut scoped_vars = variable_table.clone();
38
+ scoped_vars.set(var_name.clone(), item.clone());
39
+
40
+ let (block_end_time, new_cursor) = execute_audio_block(
41
+ engine,
42
+ global_store,
43
+ scoped_vars,
44
+ functions_table.clone(),
45
+ loop_body,
46
+ base_bpm,
47
+ base_duration,
48
+ max_time,
49
+ cur_time,
50
+ );
51
+
52
+ cur_time = new_cursor.max(block_end_time);
53
+ max_time = max_time.max(cur_time);
54
+ }
55
+
56
+ return (max_time, cur_time);
57
+ }
58
+
59
+ let loop_count = match loop_value.get("iterator") {
60
+ Some(Value::Number(n)) => *n as usize,
61
+ Some(Value::Identifier(ident)) => {
62
+ if let Some(Value::Number(n)) = variable_table.get(ident) {
63
+ *n as usize
64
+ } else {
65
+ eprintln!("❌ Loop iterator must be a number, found: {:?}", ident);
66
+ return (max_end_time, cursor_time);
67
+ }
68
+ }
69
+ _ => {
70
+ eprintln!(
71
+ "❌ Loop iterator must be a number, found: {:?}",
72
+ loop_value.get("iterator")
73
+ );
74
+ return (max_end_time, cursor_time);
75
+ }
76
+ };
77
+
78
+ let loop_body = match loop_value.get("body") {
79
+ Some(Value::Block(body)) => body.clone(),
80
+ _ => {
81
+ eprintln!(
82
+ "❌ Loop body must be a block, found: {:?}",
83
+ loop_value.get("body")
84
+ );
85
+ return (max_end_time, cursor_time);
86
+ }
87
+ };
88
+
89
+ let engine = audio_engine;
90
+ let mut cur_time = cursor_time;
91
+ let mut max_time = max_end_time;
92
+
93
+ for _ in 0..loop_count {
94
+ let (block_end_time, new_cursor) = execute_audio_block(
95
+ engine,
96
+ global_store,
97
+ variable_table.clone(),
98
+ functions_table.clone(),
99
+ &loop_body,
100
+ base_bpm,
101
+ base_duration,
102
+ max_time,
103
+ cur_time,
104
+ );
105
+
106
+ cur_time = new_cursor.max(block_end_time);
107
+ max_time = max_time.max(cur_time);
108
+ }
109
+
110
+ return (max_time, cur_time);
111
+ }
112
+
113
+ eprintln!("❌ Loop statement value is not a map");
114
+ (max_end_time, cursor_time)
115
+ }
@@ -0,0 +1,12 @@
1
+ pub mod arrow_call;
2
+ pub mod automate;
3
+ pub mod call;
4
+ pub mod condition;
5
+ pub mod function;
6
+ pub mod let_;
7
+ pub mod load;
8
+ pub mod loop_;
9
+ pub mod sleep;
10
+ pub mod spawn;
11
+ pub mod tempo;
12
+ pub mod trigger;
@@ -1,28 +1,28 @@
1
- use crate::core::parser::statement::Statement;
2
- use devalang_types::Value;
3
-
4
- pub fn interprete_sleep_statement(
5
- stmt: &Statement,
6
- cursor_time: f32,
7
- max_end_time: f32,
8
- ) -> (f32, f32) {
9
- let duration_secs = match &stmt.value {
10
- Value::Number(ms) => *ms / 1000.0,
11
- Value::String(s) if s.ends_with("ms") => s
12
- .trim_end_matches("ms")
13
- .parse::<f32>()
14
- .map(|ms| ms / 1000.0)
15
- .unwrap_or_else(|_| {
16
- eprintln!("❌ Invalid sleep value (ms): {}", s);
17
- 0.0
18
- }),
19
- other => {
20
- eprintln!("❌ Invalid sleep value: {:?}", other);
21
- 0.0
22
- }
23
- };
24
-
25
- let new_cursor = cursor_time + duration_secs;
26
- let new_max = max_end_time.max(new_cursor);
27
- (new_cursor, new_max)
28
- }
1
+ use crate::core::parser::statement::Statement;
2
+ use devalang_types::Value;
3
+
4
+ pub fn interprete_sleep_statement(
5
+ stmt: &Statement,
6
+ cursor_time: f32,
7
+ max_end_time: f32,
8
+ ) -> (f32, f32) {
9
+ let duration_secs = match &stmt.value {
10
+ Value::Number(ms) => *ms / 1000.0,
11
+ Value::String(s) if s.ends_with("ms") => s
12
+ .trim_end_matches("ms")
13
+ .parse::<f32>()
14
+ .map(|ms| ms / 1000.0)
15
+ .unwrap_or_else(|_| {
16
+ eprintln!("❌ Invalid sleep value (ms): {}", s);
17
+ 0.0
18
+ }),
19
+ other => {
20
+ eprintln!("❌ Invalid sleep value: {:?}", other);
21
+ 0.0
22
+ }
23
+ };
24
+
25
+ let new_cursor = cursor_time + duration_secs;
26
+ let new_max = max_end_time.max(new_cursor);
27
+ (new_cursor, new_max)
28
+ }