@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.
- package/.devalang +9 -10
- package/Cargo.toml +84 -80
- package/README.md +10 -7
- package/docs/CHANGELOG.md +83 -0
- package/docs/ROADMAP.md +6 -2
- package/docs/TODO.md +3 -14
- package/examples/bus.deva +10 -0
- package/examples/chain.deva +19 -0
- package/examples/effect.deva +2 -0
- package/examples/filter.deva +11 -0
- package/examples/lfo.deva +9 -0
- package/examples/plugin.deva +10 -10
- package/examples/routing.deva +23 -0
- package/examples/synth.deva +11 -1
- package/examples/synth_types.deva +17 -0
- package/out-tsc/bin/project-version.json +6 -0
- package/out-tsc/core/functions/index.d.ts +5 -0
- package/out-tsc/core/functions/index.js +11 -0
- package/out-tsc/pkg/devalang_core.d.ts +2 -0
- package/out-tsc/pkg/devalang_core.js +17 -2
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +1 -0
- 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} +34 -43
- package/rust/cli/build/commands.rs +153 -103
- package/rust/cli/build/mod.rs +2 -2
- package/rust/cli/build/process.rs +165 -146
- package/rust/cli/check/mod.rs +208 -208
- package/rust/cli/discover/commands.rs +53 -31
- package/rust/cli/discover/config.rs +2 -4
- package/rust/cli/discover/install.rs +139 -28
- package/rust/cli/discover/metadata.rs +3 -3
- package/rust/cli/login/commands.rs +124 -124
- package/rust/cli/me/commands.rs +52 -0
- package/rust/cli/me/mod.rs +1 -0
- package/rust/cli/mod.rs +2 -2
- package/rust/cli/parser.rs +76 -70
- package/rust/cli/play/commands.rs +375 -324
- package/rust/cli/play/mod.rs +5 -5
- package/rust/cli/play/process.rs +159 -150
- package/rust/cli/play/realtime.rs +91 -91
- package/rust/cli/telemetry/commands.rs +22 -22
- package/rust/cli/telemetry/event_creator.rs +80 -80
- package/rust/cli/telemetry/mod.rs +3 -3
- package/rust/cli/telemetry/send.rs +51 -51
- package/rust/cli/template/commands.rs +69 -69
- package/rust/config/driver.rs +112 -103
- package/rust/config/mod.rs +3 -3
- package/rust/config/ops.rs +26 -26
- package/rust/config/settings.rs +101 -101
- package/rust/core/audio/engine/driver.rs +237 -0
- package/rust/core/audio/engine/export.rs +169 -0
- package/rust/core/audio/engine/helpers.rs +178 -170
- package/rust/core/audio/engine/mod.rs +56 -7
- package/rust/core/audio/engine/notes/dsp.rs +88 -0
- package/rust/core/audio/engine/notes/mod.rs +53 -0
- package/rust/core/audio/engine/notes/params.rs +294 -0
- package/rust/core/audio/engine/sample/insert.rs +300 -0
- package/rust/core/audio/engine/sample/mod.rs +40 -0
- package/rust/core/audio/engine/sample/padding.rs +170 -0
- package/rust/core/audio/evaluator/condition.rs +61 -0
- package/rust/core/audio/evaluator/mod.rs +9 -0
- package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +152 -310
- package/rust/core/audio/evaluator/rhs.rs +16 -0
- package/rust/core/audio/evaluator/string_expr.rs +94 -0
- package/rust/core/audio/interpreter/driver.rs +574 -542
- package/rust/core/audio/interpreter/mod.rs +2 -14
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +179 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +3 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +371 -0
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
- package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +2 -4
- package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +36 -5
- package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -71
- package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +24 -26
- package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +36 -38
- package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +17 -19
- package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +115 -114
- package/rust/core/audio/interpreter/statements/mod.rs +12 -0
- package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
- package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +54 -4
- package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
- package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +242 -239
- package/rust/core/audio/loader/trigger.rs +98 -97
- package/rust/core/audio/mod.rs +6 -7
- package/rust/core/audio/special/easing.rs +189 -189
- package/rust/core/audio/special/env.rs +45 -45
- package/rust/core/audio/special/math.rs +134 -134
- package/rust/core/audio/special/modulator.rs +143 -143
- package/rust/core/builder/mod.rs +129 -86
- package/rust/core/debugger/{module.rs → logs.rs} +52 -55
- package/rust/core/debugger/mod.rs +30 -30
- package/rust/core/debugger/store.rs +38 -40
- package/rust/core/error/mod.rs +269 -269
- package/rust/core/lexer/driver.rs +2 -4
- package/rust/core/mod.rs +9 -10
- package/rust/core/parser/driver/block.rs +111 -0
- package/rust/core/parser/driver/cursor.rs +82 -0
- package/rust/core/parser/driver/driver_impl.rs +159 -0
- package/rust/core/parser/driver/mod.rs +6 -0
- package/rust/core/parser/driver/parse_array.rs +120 -0
- package/rust/core/parser/driver/parse_map.rs +247 -0
- package/rust/core/parser/driver/parser.rs +160 -0
- package/rust/core/parser/handler/arrow_call.rs +90 -15
- package/rust/core/parser/handler/at.rs +279 -279
- package/rust/core/parser/handler/bank.rs +104 -104
- package/rust/core/parser/handler/condition.rs +83 -83
- package/rust/core/parser/handler/dot.rs +148 -148
- package/rust/core/parser/handler/identifier/automate.rs +254 -254
- package/rust/core/parser/handler/identifier/call.rs +91 -91
- package/rust/core/parser/handler/identifier/emit.rs +70 -70
- package/rust/core/parser/handler/identifier/function.rs +113 -113
- package/rust/core/parser/handler/identifier/group.rs +89 -89
- package/rust/core/parser/handler/identifier/let_.rs +173 -173
- package/rust/core/parser/handler/identifier/mod.rs +55 -55
- package/rust/core/parser/handler/identifier/on.rs +107 -107
- package/rust/core/parser/handler/identifier/print.rs +49 -49
- package/rust/core/parser/handler/identifier/sleep.rs +96 -43
- package/rust/core/parser/handler/identifier/spawn.rs +91 -91
- package/rust/core/parser/handler/identifier/synth.rs +39 -3
- package/rust/core/parser/handler/loop_.rs +194 -194
- package/rust/core/parser/handler/pattern.rs +25 -2
- package/rust/core/parser/handler/tempo.rs +105 -57
- package/rust/core/parser/statement.rs +10 -11
- package/rust/core/plugin/loader.rs +137 -137
- package/rust/core/plugin/runner/mod.rs +11 -0
- package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +206 -72
- package/rust/core/plugin/runner/wasm32.rs +44 -0
- package/rust/core/preprocessor/loader/inject.rs +313 -0
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
- package/rust/core/preprocessor/loader/mod.rs +235 -0
- package/rust/core/preprocessor/module.rs +55 -60
- package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +107 -114
- package/rust/core/preprocessor/processor/mod.rs +1 -0
- package/rust/core/preprocessor/resolver/function.rs +69 -69
- package/rust/core/preprocessor/resolver/group.rs +122 -94
- package/rust/core/preprocessor/resolver/pattern.rs +14 -2
- package/rust/core/store/global.rs +57 -61
- package/rust/core/store/mod.rs +1 -5
- package/rust/lib.rs +323 -308
- package/rust/macros/Cargo.toml +14 -0
- package/rust/macros/src/lib.rs +52 -0
- package/rust/main.rs +336 -143
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +57 -55
- package/rust/types/src/config.rs +82 -74
- package/rust/types/src/lib.rs +15 -12
- package/rust/types/src/plugin.rs +20 -0
- package/rust/types/src/store.rs +139 -0
- package/rust/types/src/telemetry.rs +85 -85
- package/rust/utils/Cargo.toml +5 -2
- package/rust/utils/src/file.rs +477 -94
- package/rust/utils/src/first_usage.rs +97 -97
- package/rust/utils/src/lib.rs +9 -9
- package/rust/utils/src/logger.rs +200 -200
- package/rust/utils/src/path.rs +158 -88
- package/rust/utils/src/signature.rs +41 -41
- package/rust/utils/src/spinner.rs +20 -20
- package/rust/utils/src/version.rs +58 -27
- package/rust/utils/src/watcher.rs +46 -46
- package/rust/web/api.rs +5 -5
- package/rust/web/auth.rs +5 -0
- package/rust/web/cdn.rs +34 -34
- package/rust/web/forge.rs +5 -0
- package/rust/web/mod.rs +2 -0
- package/tests/integration.rs +21 -21
- package/typescript/core/functions/index.ts +11 -0
- package/typescript/pkg/devalang_core.ts +20 -4
- 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 -275
- package/rust/cli/bank/mod.rs +0 -29
- package/rust/cli/install/bank.rs +0 -53
- package/rust/cli/install/commands.rs +0 -35
- package/rust/cli/install/mod.rs +0 -4
- package/rust/cli/install/plugin.rs +0 -61
- package/rust/core/audio/engine/sample.rs +0 -366
- package/rust/core/audio/engine/synth.rs +0 -325
- package/rust/core/audio/interpreter/arrow_call.rs +0 -311
- package/rust/core/audio/renderer.rs +0 -54
- package/rust/core/parser/driver.rs +0 -584
- package/rust/core/preprocessor/loader.rs +0 -637
- package/rust/core/store/export.rs +0 -28
- package/rust/core/store/function.rs +0 -40
- package/rust/core/store/import.rs +0 -28
- package/rust/core/store/variable.rs +0 -51
- package/rust/core/utils/mod.rs +0 -1
- package/rust/core/utils/path.rs +0 -37
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
use crate::core::{
|
|
2
|
+
lexer::token::{Token, TokenKind},
|
|
3
|
+
parser::statement::Statement,
|
|
4
|
+
store::global::GlobalStore,
|
|
5
|
+
};
|
|
6
|
+
use devalang_types::Value;
|
|
7
|
+
|
|
8
|
+
#[derive(Debug, Clone, PartialEq)]
|
|
9
|
+
pub struct Parser {
|
|
10
|
+
pub resolve_modules: bool,
|
|
11
|
+
pub tokens: Vec<Token>,
|
|
12
|
+
pub token_index: usize,
|
|
13
|
+
pub current_module: String,
|
|
14
|
+
pub previous: Option<Token>,
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
impl Default for Parser {
|
|
18
|
+
fn default() -> Self {
|
|
19
|
+
Self::new()
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
impl Parser {
|
|
24
|
+
pub fn new() -> Self {
|
|
25
|
+
Parser {
|
|
26
|
+
resolve_modules: false,
|
|
27
|
+
tokens: Vec::new(),
|
|
28
|
+
token_index: 0,
|
|
29
|
+
current_module: String::new(),
|
|
30
|
+
previous: None,
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
pub fn set_current_module(&mut self, module_path: String) {
|
|
35
|
+
self.current_module = module_path;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
pub fn advance(&mut self) -> Option<&Token> {
|
|
39
|
+
crate::core::parser::driver::cursor::advance_impl(self)
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
pub fn peek_is(&self, expected: &str) -> bool {
|
|
43
|
+
crate::core::parser::driver::cursor::peek_is_impl(self, expected)
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
pub fn peek_nth(&self, n: usize) -> Option<&Token> {
|
|
47
|
+
crate::core::parser::driver::cursor::peek_nth_impl(self, n)
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
pub fn peek_nth_kind(&self, n: usize) -> Option<TokenKind> {
|
|
51
|
+
crate::core::parser::driver::cursor::peek_nth_kind_impl(self, n)
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
pub fn advance_if(&mut self, kind: TokenKind) -> bool {
|
|
55
|
+
crate::core::parser::driver::cursor::advance_if_impl(self, kind)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
pub fn match_token(&mut self, kind: TokenKind) -> bool {
|
|
59
|
+
crate::core::parser::driver::cursor::match_token_impl(self, kind)
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
pub fn previous_clone(&self) -> Option<Token> {
|
|
63
|
+
crate::core::parser::driver::cursor::previous_clone_impl(self)
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
pub fn parse_block(
|
|
67
|
+
&self,
|
|
68
|
+
tokens: Vec<Token>,
|
|
69
|
+
global_store: &mut GlobalStore,
|
|
70
|
+
) -> Vec<Statement> {
|
|
71
|
+
crate::core::parser::driver::block::parse_block(self, tokens, global_store)
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
pub fn parse_tokens(
|
|
75
|
+
&mut self,
|
|
76
|
+
tokens: Vec<Token>,
|
|
77
|
+
global_store: &mut GlobalStore,
|
|
78
|
+
) -> Vec<Statement> {
|
|
79
|
+
crate::core::parser::driver::driver_impl::parse_tokens_impl(self, tokens, global_store)
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
pub fn check_token(&self, kind: TokenKind) -> bool {
|
|
83
|
+
crate::core::parser::driver::cursor::peek_impl(self).is_some_and(|t| t.kind == kind)
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
pub fn peek_kind(&self) -> Option<TokenKind> {
|
|
87
|
+
crate::core::parser::driver::cursor::peek_impl(self).map(|t| t.kind.clone())
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
pub fn parse_map_value(&mut self) -> Option<Value> {
|
|
91
|
+
// Delegated to parse_map.rs
|
|
92
|
+
crate::core::parser::driver::parse_map::parse_map_value(self)
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Parse an array value like [1, 2, 3] or ["a", b]
|
|
96
|
+
pub fn parse_array_value(&mut self) -> Option<Value> {
|
|
97
|
+
// delegated to parse_array.rs
|
|
98
|
+
crate::core::parser::driver::parse_array::parse_array_value(self)
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
pub fn peek(&self) -> Option<&Token> {
|
|
102
|
+
self.tokens.get(self.token_index)
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
pub fn peek_clone(&self) -> Option<Token> {
|
|
106
|
+
self.tokens.get(self.token_index).cloned()
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
pub fn expect(&mut self, kind: TokenKind) -> Result<&Token, String> {
|
|
110
|
+
let tok = self.advance().ok_or("Unexpected end of input")?;
|
|
111
|
+
if tok.kind == kind {
|
|
112
|
+
Ok(tok)
|
|
113
|
+
} else {
|
|
114
|
+
Err(format!("Expected {:?}, got {:?}", kind, tok.kind))
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
pub fn collect_block_tokens(&mut self, base_indent: usize) -> Vec<Token> {
|
|
119
|
+
crate::core::parser::driver::block::collect_block_tokens(self, base_indent)
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
pub fn collect_until<F>(&mut self, condition: F) -> Vec<Token>
|
|
123
|
+
where
|
|
124
|
+
F: Fn(&Token) -> bool,
|
|
125
|
+
{
|
|
126
|
+
crate::core::parser::driver::block::collect_until(self, condition)
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
pub fn is_eof(&self) -> bool {
|
|
130
|
+
self.token_index >= self.tokens.len()
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
pub fn parse_block_until_next_else(
|
|
134
|
+
&mut self,
|
|
135
|
+
base_indent: usize,
|
|
136
|
+
global_store: &mut GlobalStore,
|
|
137
|
+
) -> Vec<Statement> {
|
|
138
|
+
crate::core::parser::driver::block::parse_block_until_next_else(
|
|
139
|
+
self,
|
|
140
|
+
base_indent,
|
|
141
|
+
global_store,
|
|
142
|
+
)
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
pub fn parse_condition_until_colon(&mut self) -> Option<Value> {
|
|
146
|
+
crate::core::parser::driver::driver_impl::parse_condition_until_colon_impl(self)
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
pub fn parse_block_until_else_or_dedent(
|
|
150
|
+
&mut self,
|
|
151
|
+
base_indent: usize,
|
|
152
|
+
global_store: &mut GlobalStore,
|
|
153
|
+
) -> Vec<Statement> {
|
|
154
|
+
crate::core::parser::driver::block::parse_block_until_else_or_dedent(
|
|
155
|
+
self,
|
|
156
|
+
base_indent,
|
|
157
|
+
global_store,
|
|
158
|
+
)
|
|
159
|
+
}
|
|
160
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
use crate::core::{
|
|
2
2
|
lexer::token::TokenKind,
|
|
3
3
|
parser::{
|
|
4
|
-
driver::Parser,
|
|
4
|
+
driver::parser::Parser,
|
|
5
5
|
statement::{Statement, StatementKind},
|
|
6
6
|
},
|
|
7
7
|
store::global::GlobalStore,
|
|
@@ -66,10 +66,15 @@ fn parse_map_literal(parser: &mut Parser) -> Value {
|
|
|
66
66
|
if let Some(TokenKind::Slash) = parser.peek_kind() {
|
|
67
67
|
parser.advance(); // '/'
|
|
68
68
|
if let Some(den) = parser.peek_clone() {
|
|
69
|
-
if den.kind == TokenKind::Number
|
|
69
|
+
if den.kind == TokenKind::Number
|
|
70
|
+
|| den.kind == TokenKind::Identifier
|
|
71
|
+
{
|
|
70
72
|
parser.advance();
|
|
71
73
|
let beat = format!("{}/{}", value_token.lexeme, den.lexeme);
|
|
72
|
-
map.insert(
|
|
74
|
+
map.insert(
|
|
75
|
+
key,
|
|
76
|
+
Value::Duration(devalang_types::Duration::Beat(beat)),
|
|
77
|
+
);
|
|
73
78
|
continue;
|
|
74
79
|
}
|
|
75
80
|
}
|
|
@@ -175,6 +180,22 @@ pub fn parse_arrow_call(parser: &mut Parser, _global_store: &mut GlobalStore) ->
|
|
|
175
180
|
parser.advance(); // ->
|
|
176
181
|
parser.advance(); // method
|
|
177
182
|
|
|
183
|
+
let args = parse_arrow_args(parser);
|
|
184
|
+
Statement {
|
|
185
|
+
kind: StatementKind::ArrowCall {
|
|
186
|
+
target: target_token.lexeme.clone(),
|
|
187
|
+
method: method_token.lexeme.clone(),
|
|
188
|
+
args,
|
|
189
|
+
},
|
|
190
|
+
value: Value::Null,
|
|
191
|
+
indent: target_token.indent,
|
|
192
|
+
line: target_token.line,
|
|
193
|
+
column: target_token.column,
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
// Parse args after the method token. Reused by continuation parser.
|
|
198
|
+
fn parse_arrow_args(parser: &mut Parser) -> Vec<Value> {
|
|
178
199
|
let mut args = Vec::new();
|
|
179
200
|
let mut paren_depth = 0;
|
|
180
201
|
let mut map_depth = 0;
|
|
@@ -219,35 +240,89 @@ pub fn parse_arrow_call(parser: &mut Parser, _global_store: &mut GlobalStore) ->
|
|
|
219
240
|
let value = match token.kind {
|
|
220
241
|
TokenKind::Identifier => Value::Identifier(token.lexeme.clone()),
|
|
221
242
|
TokenKind::String => Value::String(token.lexeme.clone()),
|
|
222
|
-
TokenKind::Number =>
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
243
|
+
TokenKind::Number => {
|
|
244
|
+
if let Some(TokenKind::Slash) = parser.peek_kind() {
|
|
245
|
+
parser.advance(); // consume '/'
|
|
246
|
+
if let Some(den) = parser.peek_clone() {
|
|
247
|
+
if den.kind == TokenKind::Number || den.kind == TokenKind::Identifier {
|
|
248
|
+
parser.advance();
|
|
249
|
+
let beat = format!("{}/{}", token.lexeme, den.lexeme);
|
|
250
|
+
Value::Duration(devalang_types::Duration::Beat(beat))
|
|
251
|
+
} else {
|
|
252
|
+
Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0))
|
|
253
|
+
}
|
|
254
|
+
} else {
|
|
255
|
+
Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0))
|
|
256
|
+
}
|
|
257
|
+
} else {
|
|
258
|
+
Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0))
|
|
259
|
+
}
|
|
229
260
|
}
|
|
261
|
+
TokenKind::LBrace => parse_map_literal(parser),
|
|
230
262
|
_ => Value::Unknown,
|
|
231
263
|
};
|
|
232
264
|
|
|
233
265
|
args.push(value);
|
|
234
266
|
|
|
235
|
-
// Stop if we reach the end of the statement
|
|
236
267
|
if paren_depth == 0 && (token.kind == TokenKind::RParen || token.kind == TokenKind::RBrace)
|
|
237
268
|
{
|
|
238
269
|
break;
|
|
239
270
|
}
|
|
240
271
|
}
|
|
241
272
|
|
|
273
|
+
args
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
// Parse an arrow continuation that begins with an Arrow token. If prev_target is Some,
|
|
277
|
+
// use it as the call target; otherwise produce an Unknown statement.
|
|
278
|
+
pub fn parse_arrow_continuation(
|
|
279
|
+
parser: &mut Parser,
|
|
280
|
+
_global_store: &mut GlobalStore,
|
|
281
|
+
prev_target: Option<String>,
|
|
282
|
+
) -> Statement {
|
|
283
|
+
// We expect current token to be Arrow
|
|
284
|
+
let arrow_tok = parser.peek_clone();
|
|
285
|
+
if arrow_tok.is_none() {
|
|
286
|
+
return Statement::unknown();
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
// If there is no previous target, consume arrow and return unknown
|
|
290
|
+
let Some(target) = prev_target else {
|
|
291
|
+
parser.advance(); // consume Arrow
|
|
292
|
+
return Statement::unknown();
|
|
293
|
+
};
|
|
294
|
+
|
|
295
|
+
// consume '->'
|
|
296
|
+
parser.advance();
|
|
297
|
+
|
|
298
|
+
// next token should be method identifier
|
|
299
|
+
let Some(method_token) = parser.peek_nth(0).cloned() else {
|
|
300
|
+
return Statement::unknown();
|
|
301
|
+
};
|
|
302
|
+
|
|
303
|
+
if method_token.kind != TokenKind::Identifier {
|
|
304
|
+
parser.advance();
|
|
305
|
+
return Statement::unknown_with_pos(
|
|
306
|
+
method_token.indent,
|
|
307
|
+
method_token.line,
|
|
308
|
+
method_token.column,
|
|
309
|
+
);
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
// consume method
|
|
313
|
+
parser.advance();
|
|
314
|
+
|
|
315
|
+
let args = parse_arrow_args(parser);
|
|
316
|
+
|
|
242
317
|
Statement {
|
|
243
318
|
kind: StatementKind::ArrowCall {
|
|
244
|
-
target
|
|
319
|
+
target,
|
|
245
320
|
method: method_token.lexeme.clone(),
|
|
246
321
|
args,
|
|
247
322
|
},
|
|
248
323
|
value: Value::Null,
|
|
249
|
-
indent:
|
|
250
|
-
line:
|
|
251
|
-
column:
|
|
324
|
+
indent: method_token.indent,
|
|
325
|
+
line: method_token.line,
|
|
326
|
+
column: method_token.column,
|
|
252
327
|
}
|
|
253
328
|
}
|