@devaloop/devalang 0.0.1-beta.2 → 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.
- package/Cargo.toml +84 -81
- package/README.md +3 -2
- package/docs/CHANGELOG.md +41 -0
- package/docs/ROADMAP.md +3 -3
- package/examples/chain.deva +19 -0
- package/examples/plugin.deva +10 -10
- package/examples/routing.deva +23 -0
- package/out-tsc/bin/project-version.json +6 -0
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
- package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
- package/out-tsc/scripts/version/copy-to-binary.js +79 -0
- package/package.json +23 -10
- package/project-version.json +3 -3
- package/rust/bindings/Cargo.toml +9 -0
- package/rust/bindings/src/lib.rs +86 -0
- package/rust/cli/addon/commands.rs +35 -0
- package/rust/cli/addon/download.rs +234 -0
- package/rust/cli/addon/install.rs +33 -0
- package/rust/cli/addon/list.rs +224 -0
- package/rust/cli/addon/metadata.rs +124 -0
- package/rust/cli/addon/mod.rs +8 -0
- package/rust/cli/addon/remove.rs +271 -0
- package/rust/cli/addon/update.rs +305 -0
- package/rust/cli/{install/addon.rs → addon/utils.rs} +109 -118
- package/rust/cli/build/commands.rs +153 -153
- package/rust/cli/build/process.rs +165 -165
- package/rust/cli/check/mod.rs +208 -208
- package/rust/cli/discover/commands.rs +275 -253
- package/rust/cli/discover/config.rs +109 -111
- package/rust/cli/discover/fs.rs +19 -19
- package/rust/cli/discover/install.rs +214 -103
- package/rust/cli/discover/metadata.rs +48 -48
- package/rust/cli/discover/mod.rs +5 -5
- package/rust/cli/me/commands.rs +52 -0
- package/rust/cli/me/mod.rs +1 -0
- package/rust/cli/mod.rs +12 -12
- package/rust/cli/parser.rs +30 -69
- package/rust/cli/play/commands.rs +375 -375
- package/rust/cli/play/process.rs +159 -159
- package/rust/core/audio/engine/driver.rs +19 -2
- package/rust/core/audio/engine/export.rs +169 -169
- package/rust/core/audio/engine/mod.rs +56 -56
- package/rust/core/audio/engine/notes/dsp.rs +88 -85
- package/rust/core/audio/engine/notes/mod.rs +53 -44
- package/rust/core/audio/engine/notes/params.rs +294 -294
- package/rust/core/audio/engine/sample/insert.rs +148 -47
- package/rust/core/audio/engine/sample/mod.rs +40 -40
- package/rust/core/audio/engine/sample/padding.rs +170 -170
- package/rust/core/audio/evaluator/condition.rs +61 -61
- package/rust/core/audio/evaluator/numeric.rs +152 -152
- package/rust/core/audio/evaluator/rhs.rs +16 -16
- package/rust/core/audio/evaluator/string_expr.rs +94 -94
- package/rust/core/audio/interpreter/driver.rs +574 -574
- package/rust/core/audio/interpreter/mod.rs +2 -2
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
- package/rust/core/audio/interpreter/statements/automate.rs +16 -16
- package/rust/core/audio/interpreter/statements/call.rs +31 -1
- package/rust/core/audio/interpreter/statements/condition.rs +72 -72
- package/rust/core/audio/interpreter/statements/function.rs +24 -24
- package/rust/core/audio/interpreter/statements/let_.rs +36 -36
- package/rust/core/audio/interpreter/statements/load.rs +17 -17
- package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
- package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
- package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
- package/rust/core/audio/loader/trigger.rs +98 -98
- package/rust/core/audio/player.rs +70 -70
- package/rust/core/audio/special/mod.rs +9 -9
- package/rust/core/builder/mod.rs +129 -129
- package/rust/core/debugger/lexer.rs +27 -27
- package/rust/core/debugger/logs.rs +52 -52
- package/rust/core/debugger/preprocessor.rs +27 -27
- package/rust/core/debugger/store.rs +38 -38
- package/rust/core/lexer/driver.rs +59 -59
- package/rust/core/lexer/handler/arrow.rs +82 -82
- package/rust/core/lexer/handler/at.rs +21 -21
- package/rust/core/lexer/handler/brace.rs +41 -41
- package/rust/core/lexer/handler/colon.rs +21 -21
- package/rust/core/lexer/handler/comment.rs +30 -30
- package/rust/core/lexer/handler/dot.rs +21 -21
- package/rust/core/lexer/handler/driver.rs +337 -337
- package/rust/core/lexer/handler/identifier.rs +47 -47
- package/rust/core/lexer/handler/indent.rs +66 -66
- package/rust/core/lexer/handler/mod.rs +15 -15
- package/rust/core/lexer/handler/newline.rs +23 -23
- package/rust/core/lexer/handler/number.rs +31 -31
- package/rust/core/lexer/handler/operator.rs +46 -46
- package/rust/core/lexer/handler/parenthesis.rs +41 -41
- package/rust/core/lexer/handler/slash.rs +21 -21
- package/rust/core/lexer/handler/string.rs +63 -63
- package/rust/core/lexer/mod.rs +3 -3
- package/rust/core/mod.rs +9 -9
- package/rust/core/parser/driver/block.rs +111 -111
- package/rust/core/parser/driver/cursor.rs +82 -82
- package/rust/core/parser/driver/driver_impl.rs +21 -1
- package/rust/core/parser/driver/mod.rs +6 -6
- package/rust/core/parser/driver/parse_array.rs +120 -120
- package/rust/core/parser/driver/parse_map.rs +247 -223
- package/rust/core/parser/driver/parser.rs +160 -160
- package/rust/core/parser/handler/arrow_call.rs +65 -14
- package/rust/core/parser/handler/identifier/synth.rs +171 -135
- package/rust/core/parser/handler/mod.rs +9 -9
- package/rust/core/parser/handler/pattern.rs +24 -1
- package/rust/core/plugin/loader.rs +137 -137
- package/rust/core/plugin/mod.rs +2 -2
- package/rust/core/plugin/runner/non_wasm.rs +481 -297
- package/rust/core/plugin/runner/wasm32.rs +1 -0
- package/rust/core/preprocessor/loader/inject.rs +313 -278
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
- package/rust/core/preprocessor/loader/mod.rs +235 -235
- package/rust/core/preprocessor/module.rs +55 -55
- package/rust/core/preprocessor/processor/handlers.rs +107 -107
- package/rust/core/preprocessor/resolver/bank.rs +49 -49
- package/rust/core/preprocessor/resolver/call.rs +124 -124
- package/rust/core/preprocessor/resolver/condition.rs +95 -95
- package/rust/core/preprocessor/resolver/driver.rs +324 -324
- package/rust/core/preprocessor/resolver/function.rs +69 -69
- package/rust/core/preprocessor/resolver/group.rs +122 -122
- package/rust/core/preprocessor/resolver/let_.rs +32 -32
- package/rust/core/preprocessor/resolver/loop_.rs +318 -318
- package/rust/core/preprocessor/resolver/mod.rs +16 -16
- package/rust/core/preprocessor/resolver/pattern.rs +95 -83
- package/rust/core/preprocessor/resolver/spawn.rs +99 -99
- package/rust/core/preprocessor/resolver/synth.rs +54 -54
- package/rust/core/preprocessor/resolver/tempo.rs +48 -48
- package/rust/core/preprocessor/resolver/trigger.rs +116 -116
- package/rust/core/preprocessor/resolver/value.rs +176 -176
- package/rust/core/store/global.rs +57 -57
- package/rust/lib.rs +323 -323
- package/rust/macros/Cargo.toml +14 -0
- package/rust/macros/src/lib.rs +52 -0
- package/rust/main.rs +311 -142
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +3 -1
- package/rust/types/src/config.rs +1 -3
- package/rust/utils/Cargo.toml +5 -2
- package/rust/utils/src/file.rs +397 -14
- package/rust/utils/src/path.rs +31 -2
- package/rust/utils/src/version.rs +38 -7
- package/rust/web/auth.rs +5 -0
- package/rust/web/forge.rs +5 -0
- package/rust/web/mod.rs +5 -3
- package/typescript/scripts/version/copy-to-binary.ts +82 -0
- package/rust/cli/bank/api.rs +0 -122
- package/rust/cli/bank/commands.rs +0 -306
- package/rust/cli/bank/mod.rs +0 -29
- package/rust/cli/install/bank.rs +0 -72
- package/rust/cli/install/commands.rs +0 -35
- package/rust/cli/install/mod.rs +0 -4
- package/rust/cli/install/plugin.rs +0 -80
|
@@ -1,61 +1,61 @@
|
|
|
1
|
-
use crate::core::audio::special::resolve_env_atom;
|
|
2
|
-
use devalang_types::Value;
|
|
3
|
-
use devalang_types::VariableTable;
|
|
4
|
-
|
|
5
|
-
pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
|
|
6
|
-
let tokens: Vec<&str> = expr.split_whitespace().collect();
|
|
7
|
-
if tokens.len() != 3 {
|
|
8
|
-
return false;
|
|
9
|
-
}
|
|
10
|
-
|
|
11
|
-
let left = tokens[0];
|
|
12
|
-
let op = tokens[1];
|
|
13
|
-
let right = tokens[2];
|
|
14
|
-
|
|
15
|
-
// Resolve left and right to numeric values where possible. Accept numbers, variables or env atoms.
|
|
16
|
-
fn resolve_for_cond(s: &str, vars: &VariableTable) -> Option<f32> {
|
|
17
|
-
if let Ok(n) = s.parse::<f32>() {
|
|
18
|
-
return Some(n);
|
|
19
|
-
}
|
|
20
|
-
if let Some(Value::Number(n)) = vars.get(s) {
|
|
21
|
-
return Some(*n);
|
|
22
|
-
}
|
|
23
|
-
if let Some(v) = resolve_env_atom(s, 120.0, 1.0) {
|
|
24
|
-
return Some(v);
|
|
25
|
-
}
|
|
26
|
-
None
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
let left_val = match resolve_for_cond(left, vars) {
|
|
30
|
-
Some(v) => v,
|
|
31
|
-
None => {
|
|
32
|
-
return false;
|
|
33
|
-
}
|
|
34
|
-
};
|
|
35
|
-
|
|
36
|
-
let right_val = match resolve_for_cond(right, vars) {
|
|
37
|
-
Some(v) => v,
|
|
38
|
-
None => {
|
|
39
|
-
return false;
|
|
40
|
-
}
|
|
41
|
-
};
|
|
42
|
-
|
|
43
|
-
match op {
|
|
44
|
-
">" => left_val > right_val,
|
|
45
|
-
"<" => left_val < right_val,
|
|
46
|
-
">=" => left_val >= right_val,
|
|
47
|
-
"<=" => left_val <= right_val,
|
|
48
|
-
"==" => {
|
|
49
|
-
// relative epsilon for floating comparisons
|
|
50
|
-
let diff = (left_val - right_val).abs();
|
|
51
|
-
let largest = left_val.abs().max(right_val.abs()).max(1.0);
|
|
52
|
-
diff <= f32::EPSILON * largest
|
|
53
|
-
}
|
|
54
|
-
"!=" => {
|
|
55
|
-
let diff = (left_val - right_val).abs();
|
|
56
|
-
let largest = left_val.abs().max(right_val.abs()).max(1.0);
|
|
57
|
-
diff > f32::EPSILON * largest
|
|
58
|
-
}
|
|
59
|
-
_ => false,
|
|
60
|
-
}
|
|
61
|
-
}
|
|
1
|
+
use crate::core::audio::special::resolve_env_atom;
|
|
2
|
+
use devalang_types::Value;
|
|
3
|
+
use devalang_types::VariableTable;
|
|
4
|
+
|
|
5
|
+
pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
|
|
6
|
+
let tokens: Vec<&str> = expr.split_whitespace().collect();
|
|
7
|
+
if tokens.len() != 3 {
|
|
8
|
+
return false;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
let left = tokens[0];
|
|
12
|
+
let op = tokens[1];
|
|
13
|
+
let right = tokens[2];
|
|
14
|
+
|
|
15
|
+
// Resolve left and right to numeric values where possible. Accept numbers, variables or env atoms.
|
|
16
|
+
fn resolve_for_cond(s: &str, vars: &VariableTable) -> Option<f32> {
|
|
17
|
+
if let Ok(n) = s.parse::<f32>() {
|
|
18
|
+
return Some(n);
|
|
19
|
+
}
|
|
20
|
+
if let Some(Value::Number(n)) = vars.get(s) {
|
|
21
|
+
return Some(*n);
|
|
22
|
+
}
|
|
23
|
+
if let Some(v) = resolve_env_atom(s, 120.0, 1.0) {
|
|
24
|
+
return Some(v);
|
|
25
|
+
}
|
|
26
|
+
None
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
let left_val = match resolve_for_cond(left, vars) {
|
|
30
|
+
Some(v) => v,
|
|
31
|
+
None => {
|
|
32
|
+
return false;
|
|
33
|
+
}
|
|
34
|
+
};
|
|
35
|
+
|
|
36
|
+
let right_val = match resolve_for_cond(right, vars) {
|
|
37
|
+
Some(v) => v,
|
|
38
|
+
None => {
|
|
39
|
+
return false;
|
|
40
|
+
}
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
match op {
|
|
44
|
+
">" => left_val > right_val,
|
|
45
|
+
"<" => left_val < right_val,
|
|
46
|
+
">=" => left_val >= right_val,
|
|
47
|
+
"<=" => left_val <= right_val,
|
|
48
|
+
"==" => {
|
|
49
|
+
// relative epsilon for floating comparisons
|
|
50
|
+
let diff = (left_val - right_val).abs();
|
|
51
|
+
let largest = left_val.abs().max(right_val.abs()).max(1.0);
|
|
52
|
+
diff <= f32::EPSILON * largest
|
|
53
|
+
}
|
|
54
|
+
"!=" => {
|
|
55
|
+
let diff = (left_val - right_val).abs();
|
|
56
|
+
let largest = left_val.abs().max(right_val.abs()).max(1.0);
|
|
57
|
+
diff > f32::EPSILON * largest
|
|
58
|
+
}
|
|
59
|
+
_ => false,
|
|
60
|
+
}
|
|
61
|
+
}
|
|
@@ -1,152 +1,152 @@
|
|
|
1
|
-
use crate::core::audio::special::{
|
|
2
|
-
find_and_eval_first_easing_call, find_and_eval_first_math_call, find_and_eval_first_mod_call,
|
|
3
|
-
resolve_env_atom,
|
|
4
|
-
};
|
|
5
|
-
use devalang_types::Value;
|
|
6
|
-
use devalang_types::VariableTable;
|
|
7
|
-
|
|
8
|
-
// Very small expression evaluator for `$env.*`, `$math.*` and variables.
|
|
9
|
-
// Supports: +, -, *, / and simple parentheses, left-to-right (no precedence), and $math.(sin|cos)(expr)
|
|
10
|
-
pub fn evaluate_numeric_expression(
|
|
11
|
-
expr: &str,
|
|
12
|
-
vars: &VariableTable,
|
|
13
|
-
env_bpm: f32,
|
|
14
|
-
env_beat: f32,
|
|
15
|
-
) -> Option<f32> {
|
|
16
|
-
let expr = expr.replace(" ", "");
|
|
17
|
-
|
|
18
|
-
// Helper to resolve an atom to a number
|
|
19
|
-
fn resolve_atom(atom: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
|
|
20
|
-
if let Some(v) = resolve_env_atom(atom, bpm, beat) {
|
|
21
|
-
return Some(v);
|
|
22
|
-
}
|
|
23
|
-
if let Ok(n) = atom.parse::<f32>() {
|
|
24
|
-
return Some(n);
|
|
25
|
-
}
|
|
26
|
-
if let Some(Value::Number(n)) = vars.get(atom) {
|
|
27
|
-
return Some(*n);
|
|
28
|
-
}
|
|
29
|
-
None
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
// Shunting-like, simplified: first evaluate any $math.func(...) calls anywhere in the expression,
|
|
33
|
-
// then fold remaining parentheses and evaluate left-to-right.
|
|
34
|
-
fn eval(expr: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
|
|
35
|
-
// 1) Replace $math/$easing/$mod calls progressively with a max iteration guard
|
|
36
|
-
let mut s = expr.to_string();
|
|
37
|
-
let mut iterations = 0u32;
|
|
38
|
-
const MAX_ITER: u32 = 64;
|
|
39
|
-
|
|
40
|
-
// Evaluate modulators first (they may feed easing/math)
|
|
41
|
-
while iterations < MAX_ITER {
|
|
42
|
-
if let Some(next) =
|
|
43
|
-
find_and_eval_first_mod_call(&s, evaluate_numeric_expression, vars, bpm, beat)
|
|
44
|
-
{
|
|
45
|
-
s = next;
|
|
46
|
-
iterations += 1;
|
|
47
|
-
continue;
|
|
48
|
-
}
|
|
49
|
-
break;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
iterations = 0;
|
|
53
|
-
while iterations < MAX_ITER {
|
|
54
|
-
if let Some(next) =
|
|
55
|
-
find_and_eval_first_easing_call(&s, evaluate_numeric_expression, vars, bpm, beat)
|
|
56
|
-
{
|
|
57
|
-
s = next;
|
|
58
|
-
iterations += 1;
|
|
59
|
-
continue;
|
|
60
|
-
}
|
|
61
|
-
break;
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
iterations = 0;
|
|
65
|
-
while iterations < MAX_ITER {
|
|
66
|
-
if let Some(next) =
|
|
67
|
-
find_and_eval_first_math_call(&s, evaluate_numeric_expression, vars, bpm, beat)
|
|
68
|
-
{
|
|
69
|
-
s = next;
|
|
70
|
-
iterations += 1;
|
|
71
|
-
continue;
|
|
72
|
-
}
|
|
73
|
-
break;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
// 2) Evaluate remaining (pure) parentheses starting from innermost
|
|
77
|
-
if let Some(open) = s.rfind('(') {
|
|
78
|
-
if let Some(close_rel) = s[open..].find(')') {
|
|
79
|
-
// index relatif
|
|
80
|
-
let close = open + close_rel;
|
|
81
|
-
let inner = &s[open + 1..close];
|
|
82
|
-
let val = eval(inner, vars, bpm, beat)?;
|
|
83
|
-
let mut replaced = String::new();
|
|
84
|
-
replaced.push_str(&s[..open]);
|
|
85
|
-
replaced.push_str(&val.to_string());
|
|
86
|
-
replaced.push_str(&s[close + 1..]);
|
|
87
|
-
return eval(&replaced, vars, bpm, beat);
|
|
88
|
-
}
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
// Tokenize by operators left-to-right
|
|
92
|
-
let mut parts: Vec<String> = Vec::new();
|
|
93
|
-
let mut cur = String::new();
|
|
94
|
-
for ch in s.chars() {
|
|
95
|
-
if "+-*/".contains(ch) {
|
|
96
|
-
if !cur.is_empty() {
|
|
97
|
-
parts.push(cur.clone());
|
|
98
|
-
cur.clear();
|
|
99
|
-
}
|
|
100
|
-
parts.push(ch.to_string());
|
|
101
|
-
} else {
|
|
102
|
-
cur.push(ch);
|
|
103
|
-
}
|
|
104
|
-
}
|
|
105
|
-
if !cur.is_empty() {
|
|
106
|
-
parts.push(cur);
|
|
107
|
-
}
|
|
108
|
-
if parts.is_empty() {
|
|
109
|
-
return None;
|
|
110
|
-
}
|
|
111
|
-
|
|
112
|
-
// Resolve atoms and compute
|
|
113
|
-
let mut acc: Option<f32> = None;
|
|
114
|
-
let mut op: Option<char> = None;
|
|
115
|
-
for part in parts {
|
|
116
|
-
if part.len() == 1 && "+-*/".contains(part.chars().next().unwrap()) {
|
|
117
|
-
op = part.chars().next();
|
|
118
|
-
continue;
|
|
119
|
-
}
|
|
120
|
-
let val = if let Some(v) = resolve_atom(&part, vars, bpm, beat) {
|
|
121
|
-
v
|
|
122
|
-
} else if part.starts_with("$env.") {
|
|
123
|
-
// $env atom not handled by resolve_atom (when composed), try recursive eval
|
|
124
|
-
eval(&part, vars, bpm, beat)?
|
|
125
|
-
} else {
|
|
126
|
-
return None;
|
|
127
|
-
};
|
|
128
|
-
|
|
129
|
-
acc = Some(match (acc, op) {
|
|
130
|
-
(None, _) => val,
|
|
131
|
-
(Some(a), Some('+')) => a + val,
|
|
132
|
-
(Some(a), Some('-')) => a - val,
|
|
133
|
-
(Some(a), Some('*')) => a * val,
|
|
134
|
-
(Some(a), Some('/')) => {
|
|
135
|
-
if val != 0.0 {
|
|
136
|
-
a / val
|
|
137
|
-
} else {
|
|
138
|
-
return Some(f32::INFINITY);
|
|
139
|
-
}
|
|
140
|
-
}
|
|
141
|
-
(Some(_), None) => val,
|
|
142
|
-
_ => {
|
|
143
|
-
return None;
|
|
144
|
-
}
|
|
145
|
-
});
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
acc
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
eval(&expr, vars, env_bpm, env_beat)
|
|
152
|
-
}
|
|
1
|
+
use crate::core::audio::special::{
|
|
2
|
+
find_and_eval_first_easing_call, find_and_eval_first_math_call, find_and_eval_first_mod_call,
|
|
3
|
+
resolve_env_atom,
|
|
4
|
+
};
|
|
5
|
+
use devalang_types::Value;
|
|
6
|
+
use devalang_types::VariableTable;
|
|
7
|
+
|
|
8
|
+
// Very small expression evaluator for `$env.*`, `$math.*` and variables.
|
|
9
|
+
// Supports: +, -, *, / and simple parentheses, left-to-right (no precedence), and $math.(sin|cos)(expr)
|
|
10
|
+
pub fn evaluate_numeric_expression(
|
|
11
|
+
expr: &str,
|
|
12
|
+
vars: &VariableTable,
|
|
13
|
+
env_bpm: f32,
|
|
14
|
+
env_beat: f32,
|
|
15
|
+
) -> Option<f32> {
|
|
16
|
+
let expr = expr.replace(" ", "");
|
|
17
|
+
|
|
18
|
+
// Helper to resolve an atom to a number
|
|
19
|
+
fn resolve_atom(atom: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
|
|
20
|
+
if let Some(v) = resolve_env_atom(atom, bpm, beat) {
|
|
21
|
+
return Some(v);
|
|
22
|
+
}
|
|
23
|
+
if let Ok(n) = atom.parse::<f32>() {
|
|
24
|
+
return Some(n);
|
|
25
|
+
}
|
|
26
|
+
if let Some(Value::Number(n)) = vars.get(atom) {
|
|
27
|
+
return Some(*n);
|
|
28
|
+
}
|
|
29
|
+
None
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// Shunting-like, simplified: first evaluate any $math.func(...) calls anywhere in the expression,
|
|
33
|
+
// then fold remaining parentheses and evaluate left-to-right.
|
|
34
|
+
fn eval(expr: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
|
|
35
|
+
// 1) Replace $math/$easing/$mod calls progressively with a max iteration guard
|
|
36
|
+
let mut s = expr.to_string();
|
|
37
|
+
let mut iterations = 0u32;
|
|
38
|
+
const MAX_ITER: u32 = 64;
|
|
39
|
+
|
|
40
|
+
// Evaluate modulators first (they may feed easing/math)
|
|
41
|
+
while iterations < MAX_ITER {
|
|
42
|
+
if let Some(next) =
|
|
43
|
+
find_and_eval_first_mod_call(&s, evaluate_numeric_expression, vars, bpm, beat)
|
|
44
|
+
{
|
|
45
|
+
s = next;
|
|
46
|
+
iterations += 1;
|
|
47
|
+
continue;
|
|
48
|
+
}
|
|
49
|
+
break;
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
iterations = 0;
|
|
53
|
+
while iterations < MAX_ITER {
|
|
54
|
+
if let Some(next) =
|
|
55
|
+
find_and_eval_first_easing_call(&s, evaluate_numeric_expression, vars, bpm, beat)
|
|
56
|
+
{
|
|
57
|
+
s = next;
|
|
58
|
+
iterations += 1;
|
|
59
|
+
continue;
|
|
60
|
+
}
|
|
61
|
+
break;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
iterations = 0;
|
|
65
|
+
while iterations < MAX_ITER {
|
|
66
|
+
if let Some(next) =
|
|
67
|
+
find_and_eval_first_math_call(&s, evaluate_numeric_expression, vars, bpm, beat)
|
|
68
|
+
{
|
|
69
|
+
s = next;
|
|
70
|
+
iterations += 1;
|
|
71
|
+
continue;
|
|
72
|
+
}
|
|
73
|
+
break;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// 2) Evaluate remaining (pure) parentheses starting from innermost
|
|
77
|
+
if let Some(open) = s.rfind('(') {
|
|
78
|
+
if let Some(close_rel) = s[open..].find(')') {
|
|
79
|
+
// index relatif
|
|
80
|
+
let close = open + close_rel;
|
|
81
|
+
let inner = &s[open + 1..close];
|
|
82
|
+
let val = eval(inner, vars, bpm, beat)?;
|
|
83
|
+
let mut replaced = String::new();
|
|
84
|
+
replaced.push_str(&s[..open]);
|
|
85
|
+
replaced.push_str(&val.to_string());
|
|
86
|
+
replaced.push_str(&s[close + 1..]);
|
|
87
|
+
return eval(&replaced, vars, bpm, beat);
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
|
|
91
|
+
// Tokenize by operators left-to-right
|
|
92
|
+
let mut parts: Vec<String> = Vec::new();
|
|
93
|
+
let mut cur = String::new();
|
|
94
|
+
for ch in s.chars() {
|
|
95
|
+
if "+-*/".contains(ch) {
|
|
96
|
+
if !cur.is_empty() {
|
|
97
|
+
parts.push(cur.clone());
|
|
98
|
+
cur.clear();
|
|
99
|
+
}
|
|
100
|
+
parts.push(ch.to_string());
|
|
101
|
+
} else {
|
|
102
|
+
cur.push(ch);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
if !cur.is_empty() {
|
|
106
|
+
parts.push(cur);
|
|
107
|
+
}
|
|
108
|
+
if parts.is_empty() {
|
|
109
|
+
return None;
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// Resolve atoms and compute
|
|
113
|
+
let mut acc: Option<f32> = None;
|
|
114
|
+
let mut op: Option<char> = None;
|
|
115
|
+
for part in parts {
|
|
116
|
+
if part.len() == 1 && "+-*/".contains(part.chars().next().unwrap()) {
|
|
117
|
+
op = part.chars().next();
|
|
118
|
+
continue;
|
|
119
|
+
}
|
|
120
|
+
let val = if let Some(v) = resolve_atom(&part, vars, bpm, beat) {
|
|
121
|
+
v
|
|
122
|
+
} else if part.starts_with("$env.") {
|
|
123
|
+
// $env atom not handled by resolve_atom (when composed), try recursive eval
|
|
124
|
+
eval(&part, vars, bpm, beat)?
|
|
125
|
+
} else {
|
|
126
|
+
return None;
|
|
127
|
+
};
|
|
128
|
+
|
|
129
|
+
acc = Some(match (acc, op) {
|
|
130
|
+
(None, _) => val,
|
|
131
|
+
(Some(a), Some('+')) => a + val,
|
|
132
|
+
(Some(a), Some('-')) => a - val,
|
|
133
|
+
(Some(a), Some('*')) => a * val,
|
|
134
|
+
(Some(a), Some('/')) => {
|
|
135
|
+
if val != 0.0 {
|
|
136
|
+
a / val
|
|
137
|
+
} else {
|
|
138
|
+
return Some(f32::INFINITY);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
(Some(_), None) => val,
|
|
142
|
+
_ => {
|
|
143
|
+
return None;
|
|
144
|
+
}
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
acc
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
eval(&expr, vars, env_bpm, env_beat)
|
|
152
|
+
}
|
|
@@ -1,16 +1,16 @@
|
|
|
1
|
-
use crate::core::audio::evaluator::numeric::evaluate_numeric_expression;
|
|
2
|
-
use devalang_types::Value;
|
|
3
|
-
use devalang_types::VariableTable;
|
|
4
|
-
|
|
5
|
-
pub fn evaluate_rhs_into_value(
|
|
6
|
-
raw: &str,
|
|
7
|
-
vars: &VariableTable,
|
|
8
|
-
env_bpm: f32,
|
|
9
|
-
env_beat: f32,
|
|
10
|
-
) -> Value {
|
|
11
|
-
if let Some(num) = evaluate_numeric_expression(raw, vars, env_bpm, env_beat) {
|
|
12
|
-
Value::Number(num)
|
|
13
|
-
} else {
|
|
14
|
-
Value::String(raw.to_string())
|
|
15
|
-
}
|
|
16
|
-
}
|
|
1
|
+
use crate::core::audio::evaluator::numeric::evaluate_numeric_expression;
|
|
2
|
+
use devalang_types::Value;
|
|
3
|
+
use devalang_types::VariableTable;
|
|
4
|
+
|
|
5
|
+
pub fn evaluate_rhs_into_value(
|
|
6
|
+
raw: &str,
|
|
7
|
+
vars: &VariableTable,
|
|
8
|
+
env_bpm: f32,
|
|
9
|
+
env_beat: f32,
|
|
10
|
+
) -> Value {
|
|
11
|
+
if let Some(num) = evaluate_numeric_expression(raw, vars, env_bpm, env_beat) {
|
|
12
|
+
Value::Number(num)
|
|
13
|
+
} else {
|
|
14
|
+
Value::String(raw.to_string())
|
|
15
|
+
}
|
|
16
|
+
}
|
|
@@ -1,94 +1,94 @@
|
|
|
1
|
-
use crate::core::audio::evaluator::numeric::evaluate_numeric_expression;
|
|
2
|
-
use devalang_types::Value;
|
|
3
|
-
use devalang_types::VariableTable;
|
|
4
|
-
|
|
5
|
-
// Evaluate a simple string concatenation expression like: "hello " + name + "!" + $env.beat
|
|
6
|
-
// - Splits on + outside quotes
|
|
7
|
-
// - Terms can be string literals (double quotes), variables (Number/String/Boolean), or numeric env/math expressions
|
|
8
|
-
// Returns None if parsing fails (fallback to raw print)
|
|
9
|
-
pub fn evaluate_string_expression(
|
|
10
|
-
expr: &str,
|
|
11
|
-
vars: &VariableTable,
|
|
12
|
-
env_bpm: f32,
|
|
13
|
-
env_beat: f32,
|
|
14
|
-
) -> Option<String> {
|
|
15
|
-
// Quick reject if no '+' present
|
|
16
|
-
if !expr.contains('+') {
|
|
17
|
-
return None;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
// Split by '+' outside of quotes
|
|
21
|
-
let mut parts: Vec<String> = Vec::new();
|
|
22
|
-
let mut cur = String::new();
|
|
23
|
-
let mut in_quotes = false;
|
|
24
|
-
let mut escape = false;
|
|
25
|
-
for ch in expr.chars() {
|
|
26
|
-
if escape {
|
|
27
|
-
cur.push(ch);
|
|
28
|
-
escape = false;
|
|
29
|
-
continue;
|
|
30
|
-
}
|
|
31
|
-
if ch == '\\' {
|
|
32
|
-
// escape next char
|
|
33
|
-
escape = true;
|
|
34
|
-
continue;
|
|
35
|
-
}
|
|
36
|
-
if ch == '"' {
|
|
37
|
-
in_quotes = !in_quotes;
|
|
38
|
-
cur.push(ch);
|
|
39
|
-
continue;
|
|
40
|
-
}
|
|
41
|
-
if ch == '+' && !in_quotes {
|
|
42
|
-
parts.push(cur.to_string());
|
|
43
|
-
cur.clear();
|
|
44
|
-
continue;
|
|
45
|
-
}
|
|
46
|
-
cur.push(ch);
|
|
47
|
-
}
|
|
48
|
-
if !cur.is_empty() {
|
|
49
|
-
parts.push(cur.to_string());
|
|
50
|
-
}
|
|
51
|
-
if parts.is_empty() {
|
|
52
|
-
return None;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
// Resolve each part into a string
|
|
56
|
-
fn strip_quotes(s: &str) -> Option<String> {
|
|
57
|
-
let st = s.trim();
|
|
58
|
-
if st.len() >= 2 && st.starts_with('"') && st.ends_with('"') {
|
|
59
|
-
Some(st[1..st.len() - 1].to_string())
|
|
60
|
-
} else {
|
|
61
|
-
None
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
let mut out = String::new();
|
|
66
|
-
for p in parts {
|
|
67
|
-
if p.is_empty() {
|
|
68
|
-
continue;
|
|
69
|
-
}
|
|
70
|
-
if let Some(lit) = strip_quotes(&p) {
|
|
71
|
-
out.push_str(&lit);
|
|
72
|
-
continue;
|
|
73
|
-
}
|
|
74
|
-
// Try variables first
|
|
75
|
-
if let Some(val) = vars.get(&p) {
|
|
76
|
-
match val {
|
|
77
|
-
Value::String(s) => out.push_str(s),
|
|
78
|
-
Value::Number(n) => out.push_str(&format!("{}", n)),
|
|
79
|
-
Value::Boolean(b) => out.push_str(&format!("{}", b)),
|
|
80
|
-
other => out.push_str(&format!("{:?}", other)),
|
|
81
|
-
}
|
|
82
|
-
continue;
|
|
83
|
-
}
|
|
84
|
-
// Try env/math/numeric expression for this term
|
|
85
|
-
if let Some(n) = evaluate_numeric_expression(&p, vars, env_bpm, env_beat) {
|
|
86
|
-
out.push_str(&format!("{}", n));
|
|
87
|
-
continue;
|
|
88
|
-
}
|
|
89
|
-
// Bareword not resolved: include as-is (safe fallback)
|
|
90
|
-
out.push_str(&p);
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
Some(out)
|
|
94
|
-
}
|
|
1
|
+
use crate::core::audio::evaluator::numeric::evaluate_numeric_expression;
|
|
2
|
+
use devalang_types::Value;
|
|
3
|
+
use devalang_types::VariableTable;
|
|
4
|
+
|
|
5
|
+
// Evaluate a simple string concatenation expression like: "hello " + name + "!" + $env.beat
|
|
6
|
+
// - Splits on + outside quotes
|
|
7
|
+
// - Terms can be string literals (double quotes), variables (Number/String/Boolean), or numeric env/math expressions
|
|
8
|
+
// Returns None if parsing fails (fallback to raw print)
|
|
9
|
+
pub fn evaluate_string_expression(
|
|
10
|
+
expr: &str,
|
|
11
|
+
vars: &VariableTable,
|
|
12
|
+
env_bpm: f32,
|
|
13
|
+
env_beat: f32,
|
|
14
|
+
) -> Option<String> {
|
|
15
|
+
// Quick reject if no '+' present
|
|
16
|
+
if !expr.contains('+') {
|
|
17
|
+
return None;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
// Split by '+' outside of quotes
|
|
21
|
+
let mut parts: Vec<String> = Vec::new();
|
|
22
|
+
let mut cur = String::new();
|
|
23
|
+
let mut in_quotes = false;
|
|
24
|
+
let mut escape = false;
|
|
25
|
+
for ch in expr.chars() {
|
|
26
|
+
if escape {
|
|
27
|
+
cur.push(ch);
|
|
28
|
+
escape = false;
|
|
29
|
+
continue;
|
|
30
|
+
}
|
|
31
|
+
if ch == '\\' {
|
|
32
|
+
// escape next char
|
|
33
|
+
escape = true;
|
|
34
|
+
continue;
|
|
35
|
+
}
|
|
36
|
+
if ch == '"' {
|
|
37
|
+
in_quotes = !in_quotes;
|
|
38
|
+
cur.push(ch);
|
|
39
|
+
continue;
|
|
40
|
+
}
|
|
41
|
+
if ch == '+' && !in_quotes {
|
|
42
|
+
parts.push(cur.to_string());
|
|
43
|
+
cur.clear();
|
|
44
|
+
continue;
|
|
45
|
+
}
|
|
46
|
+
cur.push(ch);
|
|
47
|
+
}
|
|
48
|
+
if !cur.is_empty() {
|
|
49
|
+
parts.push(cur.to_string());
|
|
50
|
+
}
|
|
51
|
+
if parts.is_empty() {
|
|
52
|
+
return None;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Resolve each part into a string
|
|
56
|
+
fn strip_quotes(s: &str) -> Option<String> {
|
|
57
|
+
let st = s.trim();
|
|
58
|
+
if st.len() >= 2 && st.starts_with('"') && st.ends_with('"') {
|
|
59
|
+
Some(st[1..st.len() - 1].to_string())
|
|
60
|
+
} else {
|
|
61
|
+
None
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
let mut out = String::new();
|
|
66
|
+
for p in parts {
|
|
67
|
+
if p.is_empty() {
|
|
68
|
+
continue;
|
|
69
|
+
}
|
|
70
|
+
if let Some(lit) = strip_quotes(&p) {
|
|
71
|
+
out.push_str(&lit);
|
|
72
|
+
continue;
|
|
73
|
+
}
|
|
74
|
+
// Try variables first
|
|
75
|
+
if let Some(val) = vars.get(&p) {
|
|
76
|
+
match val {
|
|
77
|
+
Value::String(s) => out.push_str(s),
|
|
78
|
+
Value::Number(n) => out.push_str(&format!("{}", n)),
|
|
79
|
+
Value::Boolean(b) => out.push_str(&format!("{}", b)),
|
|
80
|
+
other => out.push_str(&format!("{:?}", other)),
|
|
81
|
+
}
|
|
82
|
+
continue;
|
|
83
|
+
}
|
|
84
|
+
// Try env/math/numeric expression for this term
|
|
85
|
+
if let Some(n) = evaluate_numeric_expression(&p, vars, env_bpm, env_beat) {
|
|
86
|
+
out.push_str(&format!("{}", n));
|
|
87
|
+
continue;
|
|
88
|
+
}
|
|
89
|
+
// Bareword not resolved: include as-is (safe fallback)
|
|
90
|
+
out.push_str(&p);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
Some(out)
|
|
94
|
+
}
|