@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
|
@@ -1,14 +1,2 @@
|
|
|
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
|
+
pub mod driver;
|
|
2
|
+
pub mod statements;
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
use crate::core::{
|
|
2
|
+
audio::{
|
|
3
|
+
engine::AudioEngine,
|
|
4
|
+
interpreter::statements::arrow_call::methods::{
|
|
5
|
+
chord::interprete_chord_method, effects::apply_effect_chain,
|
|
6
|
+
note::interprete_note_method,
|
|
7
|
+
},
|
|
8
|
+
},
|
|
9
|
+
parser::statement::{Statement, StatementKind},
|
|
10
|
+
store::global::GlobalStore,
|
|
11
|
+
};
|
|
12
|
+
use devalang_types::{Value, VariableTable};
|
|
13
|
+
use devalang_utils::logger::{LogLevel, Logger};
|
|
14
|
+
|
|
15
|
+
use std::collections::HashMap;
|
|
16
|
+
|
|
17
|
+
pub fn interprete_arrow_call_statement(
|
|
18
|
+
stmt: &Statement,
|
|
19
|
+
audio_engine: &mut AudioEngine,
|
|
20
|
+
variable_table: &VariableTable,
|
|
21
|
+
global_store: &GlobalStore,
|
|
22
|
+
base_bpm: f32,
|
|
23
|
+
base_duration: f32,
|
|
24
|
+
max_end_time: &mut f32,
|
|
25
|
+
cursor_time: Option<&mut f32>,
|
|
26
|
+
update_cursor: bool,
|
|
27
|
+
) -> (f32, f32) {
|
|
28
|
+
let cursor_copy = cursor_time.as_ref().map(|c| **c).unwrap_or(0.0);
|
|
29
|
+
|
|
30
|
+
if let StatementKind::ArrowCall {
|
|
31
|
+
target,
|
|
32
|
+
method,
|
|
33
|
+
args,
|
|
34
|
+
} = &stmt.kind
|
|
35
|
+
{
|
|
36
|
+
let Some(Value::Statement(synth_stmt)) = variable_table.get(target) else {
|
|
37
|
+
let logger = Logger::new();
|
|
38
|
+
logger.log_message(
|
|
39
|
+
LogLevel::Error,
|
|
40
|
+
&format!("Synth '{}' not found in variable table", target),
|
|
41
|
+
);
|
|
42
|
+
return (*max_end_time, cursor_copy);
|
|
43
|
+
};
|
|
44
|
+
|
|
45
|
+
let Value::Map(synth_map) = &synth_stmt.value else {
|
|
46
|
+
let logger = Logger::new();
|
|
47
|
+
logger.log_message(
|
|
48
|
+
LogLevel::Error,
|
|
49
|
+
&format!("Invalid synth statement for '{}', expected a map.", target),
|
|
50
|
+
);
|
|
51
|
+
return (*max_end_time, cursor_copy);
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
let Some(Value::String(entity)) = synth_map.get("entity") else {
|
|
55
|
+
let logger = Logger::new();
|
|
56
|
+
logger.log_message(
|
|
57
|
+
LogLevel::Error,
|
|
58
|
+
&format!("Missing 'entity' key in synth '{}'.", target),
|
|
59
|
+
);
|
|
60
|
+
return (*max_end_time, cursor_copy);
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
if entity != "synth" {
|
|
64
|
+
let logger = Logger::new();
|
|
65
|
+
logger.log_message(
|
|
66
|
+
LogLevel::Error,
|
|
67
|
+
&format!("'{}' is not a synth, entity is '{}'.", target, entity),
|
|
68
|
+
);
|
|
69
|
+
return (*max_end_time, cursor_copy);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
let Some(Value::Map(value_map)) = synth_map.get("value") else {
|
|
73
|
+
let logger = Logger::new();
|
|
74
|
+
logger.log_message(
|
|
75
|
+
LogLevel::Error,
|
|
76
|
+
&format!("Missing 'value' map in synth '{}'.", target),
|
|
77
|
+
);
|
|
78
|
+
return (*max_end_time, cursor_copy);
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
let waveform_str = match value_map.get("waveform") {
|
|
82
|
+
Some(Value::String(s)) => s.clone(),
|
|
83
|
+
Some(Value::Identifier(s)) => s.clone(),
|
|
84
|
+
_ => {
|
|
85
|
+
let logger = Logger::new();
|
|
86
|
+
logger.log_message(
|
|
87
|
+
LogLevel::Error,
|
|
88
|
+
&format!("Missing or invalid 'waveform' in synth '{}'.", target),
|
|
89
|
+
);
|
|
90
|
+
return (*max_end_time, cursor_copy);
|
|
91
|
+
}
|
|
92
|
+
};
|
|
93
|
+
let Some(Value::Map(params)) = value_map.get("parameters") else {
|
|
94
|
+
println!("❌ Missing or invalid 'parameters' in synth '{}'.", target);
|
|
95
|
+
return (*max_end_time, cursor_copy);
|
|
96
|
+
};
|
|
97
|
+
|
|
98
|
+
// Synth parameters (mutable so we can apply type presets)
|
|
99
|
+
let mut synth_params = params.clone();
|
|
100
|
+
|
|
101
|
+
// Apply type defaults using the modular types module
|
|
102
|
+
crate::core::audio::interpreter::statements::arrow_call::types::apply_type(
|
|
103
|
+
&mut synth_params,
|
|
104
|
+
);
|
|
105
|
+
|
|
106
|
+
let amp = extract_f32(&synth_params, "amp", base_bpm).unwrap_or(1.0);
|
|
107
|
+
|
|
108
|
+
match method.as_str() {
|
|
109
|
+
"note" => {
|
|
110
|
+
return interprete_note_method(
|
|
111
|
+
args,
|
|
112
|
+
target,
|
|
113
|
+
audio_engine,
|
|
114
|
+
variable_table,
|
|
115
|
+
global_store,
|
|
116
|
+
&waveform_str,
|
|
117
|
+
&synth_params,
|
|
118
|
+
amp,
|
|
119
|
+
base_bpm,
|
|
120
|
+
base_duration,
|
|
121
|
+
max_end_time,
|
|
122
|
+
cursor_time,
|
|
123
|
+
cursor_copy,
|
|
124
|
+
update_cursor,
|
|
125
|
+
);
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
"chord" => {
|
|
129
|
+
return interprete_chord_method(
|
|
130
|
+
args,
|
|
131
|
+
target,
|
|
132
|
+
audio_engine,
|
|
133
|
+
variable_table,
|
|
134
|
+
global_store,
|
|
135
|
+
&waveform_str,
|
|
136
|
+
&synth_params,
|
|
137
|
+
amp,
|
|
138
|
+
base_bpm,
|
|
139
|
+
base_duration,
|
|
140
|
+
max_end_time,
|
|
141
|
+
cursor_time,
|
|
142
|
+
cursor_copy,
|
|
143
|
+
update_cursor,
|
|
144
|
+
);
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
_ => {
|
|
148
|
+
// Treat unknown methods as potential chainable effects (echo, reverb, slide, ...)
|
|
149
|
+
apply_effect_chain(
|
|
150
|
+
method.as_str(),
|
|
151
|
+
args,
|
|
152
|
+
target.as_str(),
|
|
153
|
+
audio_engine,
|
|
154
|
+
variable_table,
|
|
155
|
+
);
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
(*max_end_time, cursor_copy)
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
fn extract_f32(map: &HashMap<String, Value>, key: &str, base_bpm: f32) -> Option<f32> {
|
|
164
|
+
map.get(key).and_then(|v| match v {
|
|
165
|
+
Value::Number(n) => Some(*n),
|
|
166
|
+
Value::Beat(beat_str) => {
|
|
167
|
+
let parts: Vec<&str> = beat_str.split('/').collect();
|
|
168
|
+
if parts.len() == 2 {
|
|
169
|
+
let numerator = parts[0].parse::<f32>().ok()?;
|
|
170
|
+
let denominator = parts[1].parse::<f32>().ok()?;
|
|
171
|
+
|
|
172
|
+
Some((numerator / denominator) * ((60.0 / base_bpm) * 1000.0))
|
|
173
|
+
} else {
|
|
174
|
+
None
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
_ => None,
|
|
178
|
+
})
|
|
179
|
+
}
|
|
@@ -0,0 +1,398 @@
|
|
|
1
|
+
use crate::core::audio::engine::AudioEngine;
|
|
2
|
+
use devalang_types::Value;
|
|
3
|
+
use devalang_utils::logger::{LogLevel, Logger};
|
|
4
|
+
use std::collections::HashMap;
|
|
5
|
+
|
|
6
|
+
pub fn interprete_chord_method(
|
|
7
|
+
args: &Vec<devalang_types::Value>,
|
|
8
|
+
target: &str,
|
|
9
|
+
audio_engine: &mut AudioEngine,
|
|
10
|
+
variable_table: &devalang_types::VariableTable,
|
|
11
|
+
_global_store: &crate::core::store::global::GlobalStore,
|
|
12
|
+
waveform_str: &str,
|
|
13
|
+
synth_params: &HashMap<String, devalang_types::Value>,
|
|
14
|
+
amp: f32,
|
|
15
|
+
base_bpm: f32,
|
|
16
|
+
base_duration: f32,
|
|
17
|
+
max_end_time: &mut f32,
|
|
18
|
+
mut cursor_time: Option<&mut f32>,
|
|
19
|
+
cursor_copy: f32,
|
|
20
|
+
update_cursor: bool,
|
|
21
|
+
) -> (f32, f32) {
|
|
22
|
+
// Filter unknown args
|
|
23
|
+
let filtered_args: Vec<_> = args
|
|
24
|
+
.iter()
|
|
25
|
+
.filter(|arg| !matches!(arg, Value::Unknown))
|
|
26
|
+
.collect();
|
|
27
|
+
|
|
28
|
+
// Expect at least one note identifier, up to 4 notes, optionally last arg is a map of params
|
|
29
|
+
if filtered_args.is_empty() {
|
|
30
|
+
let logger = Logger::new();
|
|
31
|
+
logger.log_message(
|
|
32
|
+
LogLevel::Error,
|
|
33
|
+
&format!("Invalid or missing arguments for 'chord' on '{}'.", target),
|
|
34
|
+
);
|
|
35
|
+
return (*max_end_time, cursor_copy);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// Collect note names (first N args that are Identifier or String)
|
|
39
|
+
let mut note_names: Vec<String> = Vec::new();
|
|
40
|
+
let mut note_params: HashMap<String, Value> = HashMap::new();
|
|
41
|
+
|
|
42
|
+
for (_i, arg) in filtered_args.iter().enumerate().take(5) {
|
|
43
|
+
match (*arg).clone() {
|
|
44
|
+
Value::Identifier(s) | Value::String(s) => {
|
|
45
|
+
if note_names.len() < 4 {
|
|
46
|
+
note_names.push(s);
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
Value::Map(m) => {
|
|
50
|
+
// treat as chord-level params (duration, glide, etc.)
|
|
51
|
+
for (k, v) in m {
|
|
52
|
+
note_params.insert(k, v);
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
_ => {}
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if note_names.is_empty() {
|
|
60
|
+
let logger = Logger::new();
|
|
61
|
+
logger.log_message(
|
|
62
|
+
LogLevel::Error,
|
|
63
|
+
&format!("No valid notes found for 'chord' on '{}'.", target),
|
|
64
|
+
);
|
|
65
|
+
return (*max_end_time, cursor_copy);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// duration & amp for chord
|
|
69
|
+
let amp_note = extract_f32(¬e_params, "amp", base_bpm).unwrap_or(amp);
|
|
70
|
+
let duration_ms =
|
|
71
|
+
extract_f32(¬e_params, "duration", base_bpm).unwrap_or(base_duration * 1000.0);
|
|
72
|
+
let duration_secs = duration_ms / 1000.0;
|
|
73
|
+
|
|
74
|
+
let start_time = cursor_copy;
|
|
75
|
+
let end_time = start_time + duration_secs;
|
|
76
|
+
|
|
77
|
+
// Expand shorthand chord notation like C#min, Dmaj, Amin7 into individual note names
|
|
78
|
+
note_names = expand_chord_shorthands(note_names);
|
|
79
|
+
|
|
80
|
+
// Prepare automation merge similar to note method
|
|
81
|
+
let auto_key = format!("{}__automation", target);
|
|
82
|
+
let synth_automation = match variable_table.get(&auto_key) {
|
|
83
|
+
Some(Value::Map(map)) => match map.get("params") {
|
|
84
|
+
Some(Value::Map(p)) => Some(p.clone()),
|
|
85
|
+
_ => None,
|
|
86
|
+
},
|
|
87
|
+
_ => None,
|
|
88
|
+
};
|
|
89
|
+
|
|
90
|
+
let chord_automation = match note_params.get("automate") {
|
|
91
|
+
Some(Value::Map(m)) => Some(m.clone()),
|
|
92
|
+
_ => None,
|
|
93
|
+
};
|
|
94
|
+
|
|
95
|
+
let automation = match (synth_automation, chord_automation) {
|
|
96
|
+
(Some(mut a), Some(n)) => {
|
|
97
|
+
for (k, v) in n {
|
|
98
|
+
a.insert(k, v);
|
|
99
|
+
}
|
|
100
|
+
Some(a)
|
|
101
|
+
}
|
|
102
|
+
(None, Some(n)) => Some(n),
|
|
103
|
+
(Some(a), None) => Some(a),
|
|
104
|
+
_ => None,
|
|
105
|
+
};
|
|
106
|
+
|
|
107
|
+
// For each note, let the selected type compute per-note scheduling and parameters
|
|
108
|
+
for (i, note_name) in note_names.iter().enumerate() {
|
|
109
|
+
// default: start at chord start
|
|
110
|
+
let start_ms_default = start_time * 1000.0;
|
|
111
|
+
let mut note_amp = amp_note;
|
|
112
|
+
let mut note_params_clone = note_params.clone();
|
|
113
|
+
let mut note_start_ms = start_ms_default;
|
|
114
|
+
let mut note_freq = note_to_freq(note_name);
|
|
115
|
+
|
|
116
|
+
if let Some(tval) = synth_params.get("type") {
|
|
117
|
+
let tname = match tval {
|
|
118
|
+
Value::String(s) => s.as_str(),
|
|
119
|
+
Value::Identifier(s) => s.as_str(),
|
|
120
|
+
_ => "",
|
|
121
|
+
};
|
|
122
|
+
match tname {
|
|
123
|
+
"arp" => {
|
|
124
|
+
let (s_ms, _f, amp_out, params_out) =
|
|
125
|
+
crate::core::audio::interpreter::statements::arrow_call::types::arp::prepare_note(
|
|
126
|
+
note_name,
|
|
127
|
+
i,
|
|
128
|
+
note_names.len(),
|
|
129
|
+
start_ms_default,
|
|
130
|
+
duration_ms,
|
|
131
|
+
amp_note,
|
|
132
|
+
&synth_params,
|
|
133
|
+
¬e_params_clone,
|
|
134
|
+
&automation,
|
|
135
|
+
);
|
|
136
|
+
note_start_ms = s_ms;
|
|
137
|
+
note_freq = _f;
|
|
138
|
+
note_amp = amp_out;
|
|
139
|
+
note_params_clone = params_out;
|
|
140
|
+
}
|
|
141
|
+
"pluck" => {
|
|
142
|
+
let (s_ms, _f, amp_out, params_out) =
|
|
143
|
+
crate::core::audio::interpreter::statements::arrow_call::types::pluck::prepare_note(
|
|
144
|
+
note_name,
|
|
145
|
+
i,
|
|
146
|
+
note_names.len(),
|
|
147
|
+
start_ms_default,
|
|
148
|
+
duration_ms,
|
|
149
|
+
amp_note,
|
|
150
|
+
&synth_params,
|
|
151
|
+
¬e_params_clone,
|
|
152
|
+
&automation,
|
|
153
|
+
);
|
|
154
|
+
// pluck.prepare_note returns start offset (s_ms) relative to default; if zero use default
|
|
155
|
+
if s_ms > 0.0 {
|
|
156
|
+
note_start_ms = s_ms
|
|
157
|
+
}
|
|
158
|
+
note_amp = amp_out;
|
|
159
|
+
note_params_clone = params_out;
|
|
160
|
+
}
|
|
161
|
+
"pad" => {
|
|
162
|
+
let (s_ms, _f, amp_out, params_out) =
|
|
163
|
+
crate::core::audio::interpreter::statements::arrow_call::types::pad::prepare_note(
|
|
164
|
+
note_name,
|
|
165
|
+
i,
|
|
166
|
+
note_names.len(),
|
|
167
|
+
start_ms_default,
|
|
168
|
+
duration_ms,
|
|
169
|
+
amp_note,
|
|
170
|
+
&synth_params,
|
|
171
|
+
¬e_params_clone,
|
|
172
|
+
&automation,
|
|
173
|
+
);
|
|
174
|
+
if s_ms > 0.0 {
|
|
175
|
+
note_start_ms = s_ms
|
|
176
|
+
}
|
|
177
|
+
note_amp = amp_out;
|
|
178
|
+
note_params_clone = params_out;
|
|
179
|
+
}
|
|
180
|
+
_ => {}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
let ranges = audio_engine.insert_note(
|
|
185
|
+
Some(target.to_string()),
|
|
186
|
+
waveform_str.to_string(),
|
|
187
|
+
note_freq,
|
|
188
|
+
note_amp,
|
|
189
|
+
note_start_ms,
|
|
190
|
+
duration_ms,
|
|
191
|
+
synth_params.clone(),
|
|
192
|
+
note_params_clone.clone(),
|
|
193
|
+
automation.clone(),
|
|
194
|
+
);
|
|
195
|
+
|
|
196
|
+
// Apply simple per-note effects if provided under note_params_clone.effects (map)
|
|
197
|
+
if let Some(Value::Map(eff_map)) = note_params_clone.get("effects") {
|
|
198
|
+
for (_start, _len) in ranges.iter() {
|
|
199
|
+
crate::core::audio::interpreter::statements::arrow_call::methods::effects::apply_effect_chain(
|
|
200
|
+
"echo",
|
|
201
|
+
&vec![Value::Map(eff_map.clone())],
|
|
202
|
+
target,
|
|
203
|
+
audio_engine,
|
|
204
|
+
variable_table,
|
|
205
|
+
);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
let note_end = (note_start_ms / 1000.0) + (duration_ms / 1000.0);
|
|
210
|
+
*max_end_time = (*max_end_time).max(note_end);
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
*max_end_time = (*max_end_time).max(end_time);
|
|
214
|
+
|
|
215
|
+
if update_cursor {
|
|
216
|
+
if let Some(c) = cursor_time.as_mut() {
|
|
217
|
+
**c = end_time;
|
|
218
|
+
}
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
(*max_end_time, end_time)
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
fn note_to_freq(note: &str) -> f32 {
|
|
225
|
+
let notes = [
|
|
226
|
+
"C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
|
|
227
|
+
];
|
|
228
|
+
|
|
229
|
+
if note.len() < 2 || note.len() > 3 {
|
|
230
|
+
return 440.0;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
let (name, octave_str) = note.split_at(note.len() - 1);
|
|
234
|
+
let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
|
|
235
|
+
let octave = octave_str.parse::<i32>().unwrap_or(4);
|
|
236
|
+
let midi_note = (octave + 1) * 12 + semitone;
|
|
237
|
+
|
|
238
|
+
440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
fn extract_f32(map: &HashMap<String, Value>, key: &str, base_bpm: f32) -> Option<f32> {
|
|
242
|
+
map.get(key).and_then(|v| match v {
|
|
243
|
+
Value::Number(n) => Some(*n),
|
|
244
|
+
Value::Beat(beat_str) => {
|
|
245
|
+
let parts: Vec<&str> = beat_str.split('/').collect();
|
|
246
|
+
if parts.len() == 2 {
|
|
247
|
+
let numerator = parts[0].parse::<f32>().ok()?;
|
|
248
|
+
let denominator = parts[1].parse::<f32>().ok()?;
|
|
249
|
+
|
|
250
|
+
Some((numerator / denominator) * ((60.0 / base_bpm) * 1000.0))
|
|
251
|
+
} else {
|
|
252
|
+
None
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
_ => None,
|
|
256
|
+
})
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
// Expand shorthand chords into constituent note names (strings with octave if applicable)
|
|
260
|
+
fn expand_chord_shorthands(names: Vec<String>) -> Vec<String> {
|
|
261
|
+
let mut out: Vec<String> = Vec::new();
|
|
262
|
+
for name in names.into_iter() {
|
|
263
|
+
if let Some(parts) = parse_chord_shorthand(&name) {
|
|
264
|
+
// parts contains (root_semitone, octave, chord_type)
|
|
265
|
+
let (root_semitone, octave, chord_type) = parts;
|
|
266
|
+
let root_midi = (octave + 1) * 12 + root_semitone as i32;
|
|
267
|
+
match chord_type.as_str() {
|
|
268
|
+
"min" | "m" => {
|
|
269
|
+
let third = root_midi + 3;
|
|
270
|
+
let fifth = root_midi + 7;
|
|
271
|
+
out.push(midi_to_note(root_midi));
|
|
272
|
+
out.push(midi_to_note(third));
|
|
273
|
+
out.push(midi_to_note(fifth));
|
|
274
|
+
}
|
|
275
|
+
"7" => {
|
|
276
|
+
let third = root_midi + 4;
|
|
277
|
+
let fifth = root_midi + 7;
|
|
278
|
+
let seventh = root_midi + 10;
|
|
279
|
+
out.push(midi_to_note(root_midi));
|
|
280
|
+
out.push(midi_to_note(third));
|
|
281
|
+
out.push(midi_to_note(fifth));
|
|
282
|
+
out.push(midi_to_note(seventh));
|
|
283
|
+
}
|
|
284
|
+
_ => {
|
|
285
|
+
// default to major triad
|
|
286
|
+
let third = root_midi + 4;
|
|
287
|
+
let fifth = root_midi + 7;
|
|
288
|
+
out.push(midi_to_note(root_midi));
|
|
289
|
+
out.push(midi_to_note(third));
|
|
290
|
+
out.push(midi_to_note(fifth));
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
} else {
|
|
294
|
+
out.push(name);
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
out
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
fn parse_chord_shorthand(s: &str) -> Option<(u8, i32, String)> {
|
|
301
|
+
// examples: C#min, Amin, Dmaj7, Ebm
|
|
302
|
+
let s = s.trim();
|
|
303
|
+
if s.is_empty() {
|
|
304
|
+
return None;
|
|
305
|
+
}
|
|
306
|
+
let mut chars = s.chars().peekable();
|
|
307
|
+
let first = chars.next()?;
|
|
308
|
+
let letter = first.to_ascii_uppercase();
|
|
309
|
+
if !matches!(letter, 'A'..='G') {
|
|
310
|
+
return None;
|
|
311
|
+
}
|
|
312
|
+
let mut name = String::new();
|
|
313
|
+
name.push(letter);
|
|
314
|
+
// accidental
|
|
315
|
+
if let Some(&c) = chars.peek() {
|
|
316
|
+
if c == '#' || c == 'b' {
|
|
317
|
+
name.push(c);
|
|
318
|
+
chars.next();
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
|
|
322
|
+
let rest: String = chars.collect();
|
|
323
|
+
// extract trailing digits as octave
|
|
324
|
+
let mut octave: i32 = 4; // default
|
|
325
|
+
let mut type_part = rest.as_str();
|
|
326
|
+
// if ends with digit(s)
|
|
327
|
+
if !rest.is_empty() {
|
|
328
|
+
let mut digits = String::new();
|
|
329
|
+
for ch in rest.chars().rev() {
|
|
330
|
+
if ch.is_ascii_digit() {
|
|
331
|
+
digits.insert(0, ch);
|
|
332
|
+
} else {
|
|
333
|
+
break;
|
|
334
|
+
}
|
|
335
|
+
}
|
|
336
|
+
if !digits.is_empty() {
|
|
337
|
+
if let Ok(o) = digits.parse::<i32>() {
|
|
338
|
+
octave = o;
|
|
339
|
+
// remove digits from end
|
|
340
|
+
let split_at = rest.len() - digits.len();
|
|
341
|
+
type_part = &rest[..split_at];
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
let chord_type = type_part.to_ascii_lowercase();
|
|
347
|
+
// compute semitone index
|
|
348
|
+
let semitone = match name.as_str() {
|
|
349
|
+
"C" => 0,
|
|
350
|
+
"C#" => 1,
|
|
351
|
+
"DB" => 1,
|
|
352
|
+
"D" => 2,
|
|
353
|
+
"D#" => 3,
|
|
354
|
+
"EB" => 3,
|
|
355
|
+
"E" => 4,
|
|
356
|
+
"F" => 5,
|
|
357
|
+
"F#" => 6,
|
|
358
|
+
"GB" => 6,
|
|
359
|
+
"G" => 7,
|
|
360
|
+
"G#" => 8,
|
|
361
|
+
"AB" => 8,
|
|
362
|
+
"A" => 9,
|
|
363
|
+
"A#" => 10,
|
|
364
|
+
"BB" => 10,
|
|
365
|
+
"B" => 11,
|
|
366
|
+
_ => return None,
|
|
367
|
+
};
|
|
368
|
+
|
|
369
|
+
// normalize chord_type: if empty => major
|
|
370
|
+
let chord_type = if chord_type.is_empty() {
|
|
371
|
+
"maj".to_string()
|
|
372
|
+
} else {
|
|
373
|
+
chord_type
|
|
374
|
+
};
|
|
375
|
+
|
|
376
|
+
Some((semitone as u8, octave, chord_type))
|
|
377
|
+
}
|
|
378
|
+
|
|
379
|
+
fn midi_to_note(m: i32) -> String {
|
|
380
|
+
let semitone = (m % 12 + 12) % 12;
|
|
381
|
+
let octave = (m / 12) - 1;
|
|
382
|
+
let name = match semitone {
|
|
383
|
+
0 => "C",
|
|
384
|
+
1 => "C#",
|
|
385
|
+
2 => "D",
|
|
386
|
+
3 => "D#",
|
|
387
|
+
4 => "E",
|
|
388
|
+
5 => "F",
|
|
389
|
+
6 => "F#",
|
|
390
|
+
7 => "G",
|
|
391
|
+
8 => "G#",
|
|
392
|
+
9 => "A",
|
|
393
|
+
10 => "A#",
|
|
394
|
+
11 => "B",
|
|
395
|
+
_ => "C",
|
|
396
|
+
};
|
|
397
|
+
format!("{}{}", name, octave)
|
|
398
|
+
}
|