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