Compare commits
48 Commits
2cdfcb72ee
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
251ae7f308 | ||
|
|
5b9d3a8bb7 | ||
|
|
fedd6abbff | ||
|
|
7f0b55ac13 | ||
|
|
fb44f680e6 | ||
|
|
c6a6d59f1c | ||
|
|
6c0b918612 | ||
|
|
f00923b414 | ||
|
|
de5717a61a | ||
| 36c047caf7 | |||
| f4f074b8e7 | |||
| a08246b30f | |||
| eb1a537ce2 | |||
| 31faeb3c1b | |||
| a3bb1cf11b | |||
|
|
0be0c3c0c9 | ||
|
|
b1ad4a2cea | ||
|
|
5cde54faeb | ||
|
|
faca56e9df | ||
|
|
cd9c34cb3e | ||
|
|
8cbbb9b3ec | ||
|
|
208d804f8f | ||
|
|
8f4c7596c4 | ||
|
|
05b49a3777 | ||
|
|
bd0a6007f7 | ||
|
|
beb6595f42 | ||
|
|
a2a5465fb0 | ||
| 5ba9e2ef96 | |||
| 59c7386e85 | |||
| acdac24d1a | |||
| f50a435200 | |||
| 7524ca3192 | |||
| 85b189fad2 | |||
| 63f96b2d5f | |||
| 5f149414b2 | |||
| b2095c4229 | |||
| c0930d1ccc | |||
| fc91b666ce | |||
| 683d3ef315 | |||
| a66dd03cc6 | |||
| 852648f5bc | |||
| 0e4cfa8b30 | |||
| ff3911fee9 | |||
| 37fb8940ec | |||
| 12c6f6b1ae | |||
| ef779ef75f | |||
| b8910becc6 | |||
| 0d4d92f374 |
86
Cargo.lock
generated
86
Cargo.lock
generated
@@ -33,6 +33,15 @@ dependencies = [
|
||||
"pkg-config",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "android_system_properties"
|
||||
version = "0.1.5"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311"
|
||||
dependencies = [
|
||||
"libc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "anstream"
|
||||
version = "0.6.21"
|
||||
@@ -83,6 +92,12 @@ dependencies = [
|
||||
"windows-sys",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "autocfg"
|
||||
version = "1.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c08606f8c3cbf4ce6ec8e28fb0014a2c086708fe954eaa885384a6165172e7e8"
|
||||
|
||||
[[package]]
|
||||
name = "bindgen"
|
||||
version = "0.72.1"
|
||||
@@ -125,6 +140,16 @@ version = "3.20.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5d20789868f4b01b2f2caec9f5c4e0213b41e3e5702a50157d699ae31ced2fcb"
|
||||
|
||||
[[package]]
|
||||
name = "cc"
|
||||
version = "1.2.56"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "aebf35691d1bfb0ac386a69bac2fde4dd276fb618cf8bf4f5318fe285e821bb2"
|
||||
dependencies = [
|
||||
"find-msvc-tools",
|
||||
"shlex",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "cexpr"
|
||||
version = "0.6.0"
|
||||
@@ -140,6 +165,19 @@ version = "1.0.4"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "9330f8b2ff13f34540b44e946ef35111825727b38d33286ef986142615121801"
|
||||
|
||||
[[package]]
|
||||
name = "chrono"
|
||||
version = "0.4.44"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "c673075a2e0e5f4a1dde27ce9dee1ea4558c7ffe648f576438a20ca1d2acc4b0"
|
||||
dependencies = [
|
||||
"iana-time-zone",
|
||||
"js-sys",
|
||||
"num-traits",
|
||||
"wasm-bindgen",
|
||||
"windows-link",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "clang-sys"
|
||||
version = "1.8.1"
|
||||
@@ -201,10 +239,13 @@ checksum = "b05b61dc5112cbb17e4b6cd61790d9845d13888356391624cbe7e41efeac1e75"
|
||||
name = "composer"
|
||||
version = "0.1.0"
|
||||
dependencies = [
|
||||
"chrono",
|
||||
"clap",
|
||||
"env_logger",
|
||||
"io-prompt-prototype",
|
||||
"itertools 0.14.0",
|
||||
"midir",
|
||||
"regex",
|
||||
"rustysynth",
|
||||
"tinyaudio",
|
||||
]
|
||||
@@ -290,6 +331,12 @@ version = "1.0.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f"
|
||||
|
||||
[[package]]
|
||||
name = "find-msvc-tools"
|
||||
version = "0.1.9"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5baebc0774151f905a1a2cc41989300b1e6fbb29aff0ceffa1064fdd3088d582"
|
||||
|
||||
[[package]]
|
||||
name = "futures-core"
|
||||
version = "0.3.32"
|
||||
@@ -332,6 +379,30 @@ version = "0.5.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea"
|
||||
|
||||
[[package]]
|
||||
name = "iana-time-zone"
|
||||
version = "0.1.65"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e31bc9ad994ba00e440a8aa5c9ef0ec67d5cb5e5cb0cc7f8b744a35b389cc470"
|
||||
dependencies = [
|
||||
"android_system_properties",
|
||||
"core-foundation-sys",
|
||||
"iana-time-zone-haiku",
|
||||
"js-sys",
|
||||
"log",
|
||||
"wasm-bindgen",
|
||||
"windows-core",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "iana-time-zone-haiku"
|
||||
version = "0.1.2"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f"
|
||||
dependencies = [
|
||||
"cc",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "indexmap"
|
||||
version = "2.13.0"
|
||||
@@ -342,6 +413,12 @@ dependencies = [
|
||||
"hashbrown",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "io-prompt-prototype"
|
||||
version = "1.0.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ca9d71ad6b790f2522eeb0a4f207fa7ef21826777d4714c7ff09483bf17b0ab8"
|
||||
|
||||
[[package]]
|
||||
name = "is_terminal_polyfill"
|
||||
version = "1.70.2"
|
||||
@@ -499,6 +576,15 @@ dependencies = [
|
||||
"minimal-lexical",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "num-traits"
|
||||
version = "0.2.19"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841"
|
||||
dependencies = [
|
||||
"autocfg",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "num_enum"
|
||||
version = "0.7.5"
|
||||
|
||||
@@ -4,9 +4,12 @@ version = "0.1.0"
|
||||
edition = "2024"
|
||||
|
||||
[dependencies]
|
||||
chrono = "0.4.44"
|
||||
clap = { version = "4.5.60", features = ["derive"] }
|
||||
env_logger = "0.11.9"
|
||||
io-prompt-prototype = "1.0.0"
|
||||
itertools = "0.14.0"
|
||||
midir = "0.10.3"
|
||||
regex = "1.12.3"
|
||||
rustysynth = "1.3.6"
|
||||
tinyaudio = "2.0.0"
|
||||
|
||||
@@ -5,10 +5,10 @@ pub struct Flags {
|
||||
#[arg(short, long, default_value_t = false)]
|
||||
pub debug: bool,
|
||||
|
||||
#[arg(long, default_value_t = 16)]
|
||||
pub smallest_note: usize,
|
||||
#[arg(short, long, default_value_t = false)]
|
||||
pub interactive: bool,
|
||||
|
||||
#[arg(long, default_value_t = 60)]
|
||||
#[arg(long, default_value_t = 120)]
|
||||
pub bpm: usize,
|
||||
|
||||
#[arg(long, default_value_t = 44100)]
|
||||
@@ -16,6 +16,9 @@ pub struct Flags {
|
||||
|
||||
#[arg(long, default_value = "super_small_font.sf2")]
|
||||
pub sound_font: String,
|
||||
|
||||
#[arg(short, long, default_value = "c 2*e+")]
|
||||
pub play: Vec<String>,
|
||||
}
|
||||
|
||||
impl Flags {
|
||||
|
||||
103
src/main.rs
103
src/main.rs
@@ -1,43 +1,78 @@
|
||||
use io_prompt_prototype::prompt;
|
||||
use itertools::Itertools;
|
||||
|
||||
mod flags;
|
||||
mod parse;
|
||||
mod player;
|
||||
mod seq;
|
||||
mod syn;
|
||||
mod tone;
|
||||
|
||||
fn main() {
|
||||
let flags = flags::Flags::new();
|
||||
let mut flags = flags::Flags::new();
|
||||
if flags.debug {
|
||||
eprintln!("{:?}", flags);
|
||||
}
|
||||
if flags.interactive {
|
||||
flags.play = vec![];
|
||||
}
|
||||
|
||||
let mut syn = syn::Syn::new(
|
||||
flags.debug,
|
||||
flags.sound_font,
|
||||
flags.sample_rate,
|
||||
);
|
||||
// Play some notes (middle C, E, G). // 16 channels actually // 60=c 64=e 67=g //up to 128velocity though dont go below 50 tbh // 12 notes per octave
|
||||
syn.note_on(0, 64 + 12, 127);
|
||||
|
||||
play(syn, flags.sample_rate, flags.bpm, flags.smallest_note);
|
||||
}
|
||||
|
||||
fn play(mut s: syn::Syn, sample_rate: usize, bpm: usize, smallest_note: usize) {
|
||||
let params = tinyaudio::prelude::OutputDeviceParameters {
|
||||
channels_count: 2,
|
||||
sample_rate: sample_rate,
|
||||
channel_sample_count: sample_rate / bpm * 60 / smallest_note,
|
||||
};
|
||||
|
||||
let sample_count = (params.channel_sample_count) as usize;
|
||||
let mut left: Vec<f32> = vec![0_f32; sample_count];
|
||||
let mut right: Vec<f32> = vec![0_f32; sample_count];
|
||||
|
||||
let _device = tinyaudio::prelude::run_output_device(params, {
|
||||
move |data| {
|
||||
s.render(&mut left[..], &mut right[..]); // put in a state of rendering the next loop of these notes
|
||||
for (i, value) in left.iter().interleave(right.iter()).enumerate() {
|
||||
data[i] = *value;
|
||||
}
|
||||
let once = flags.play.iter().filter(|x| x.len() > 0).count() > 0;
|
||||
let mut player = player::new(flags.sample_rate.clone(), flags.bpm.clone());
|
||||
loop {
|
||||
play_with_flags(&flags, &mut player);
|
||||
if once {
|
||||
break;
|
||||
}
|
||||
})
|
||||
.unwrap();
|
||||
|
||||
// Wait it out
|
||||
std::thread::sleep(std::time::Duration::from_secs(2));
|
||||
}
|
||||
}
|
||||
|
||||
fn play_with_flags(flags: &flags::Flags, player: &mut player::Player) {
|
||||
let mut syn_seq = seq::new_syn(syn::Syn::new(
|
||||
flags.debug.clone(),
|
||||
flags.sound_font.clone(),
|
||||
flags.sample_rate.clone(),
|
||||
));
|
||||
|
||||
let mut i = 0;
|
||||
for p in flags.play.iter() {
|
||||
for p in parse::new(p.clone()) {
|
||||
syn_seq.append(i as i32, p);
|
||||
i += 1;
|
||||
}
|
||||
}
|
||||
|
||||
let sync = i > 0;
|
||||
if !sync {
|
||||
let mut to_break = false;
|
||||
let mut payload = vec![];
|
||||
loop {
|
||||
let mut s: String = prompt!("> ").parse().expect("failed to readline");
|
||||
s = s.split_whitespace().join(" ");
|
||||
payload.push(s.clone());
|
||||
match s.as_ref() {
|
||||
"" => match to_break {
|
||||
false => {
|
||||
to_break = true;
|
||||
}
|
||||
true => {
|
||||
break;
|
||||
}
|
||||
},
|
||||
_ => {
|
||||
to_break = false;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
for p in parse::new(payload.join("\n")) {
|
||||
syn_seq.append(i as i32, p);
|
||||
i += 1;
|
||||
}
|
||||
}
|
||||
|
||||
match sync {
|
||||
true => player.play(syn_seq),
|
||||
false => player.play_async(syn_seq),
|
||||
};
|
||||
}
|
||||
|
||||
119
src/parse.rs
Normal file
119
src/parse.rs
Normal file
@@ -0,0 +1,119 @@
|
||||
use std::io::Read;
|
||||
|
||||
pub fn new<S: ToString>(s: S) -> Vec<String> {
|
||||
parse(from_string(s.to_string()))
|
||||
}
|
||||
|
||||
fn from_string(s: String) -> String {
|
||||
match std::fs::File::open(s.clone()) {
|
||||
Ok(mut reader) => {
|
||||
let mut content = String::new();
|
||||
reader
|
||||
.read_to_string(&mut content)
|
||||
.expect(format!("failed to read {}", s).as_ref());
|
||||
content
|
||||
}
|
||||
Err(_) => s,
|
||||
}
|
||||
}
|
||||
|
||||
fn parse(s: String) -> Vec<String> {
|
||||
let s = s
|
||||
.split("\n")
|
||||
.filter(|x: &&str| {
|
||||
!x // doesnt start with #
|
||||
.split_whitespace()
|
||||
.collect::<Vec<_>>()
|
||||
.join("")
|
||||
.starts_with("#")
|
||||
})
|
||||
.map(|x| {
|
||||
x.split("#")
|
||||
.take(1) // drop after #
|
||||
.collect::<Vec<_>>()
|
||||
.join("")
|
||||
.split_whitespace()
|
||||
.collect::<Vec<_>>()
|
||||
.join(" ")
|
||||
})
|
||||
.collect::<Vec<_>>()
|
||||
.join("\n");
|
||||
let mut channels = vec![];
|
||||
let lines = s.split("\n").collect::<Vec<_>>();
|
||||
let mut j = 0;
|
||||
for i in 0..lines.len() {
|
||||
match lines[i] {
|
||||
"" => {
|
||||
j = 0;
|
||||
}
|
||||
_ => {
|
||||
while channels.len() <= j {
|
||||
channels.push("".to_string());
|
||||
}
|
||||
channels[{
|
||||
let tmp = j;
|
||||
j += 1;
|
||||
tmp
|
||||
}] += &(" ".to_string() + lines[i]);
|
||||
}
|
||||
};
|
||||
}
|
||||
channels
|
||||
.iter()
|
||||
.map(|x| x.split_whitespace().collect::<Vec<_>>().join(" "))
|
||||
.collect()
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
#[test]
|
||||
fn test_string() {
|
||||
assert_eq!(super::new("a b c")[0], "a b c".to_string());
|
||||
assert_eq!(super::new("a b c".to_string())[0], "a b c".to_string());
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_single_line_file() {
|
||||
assert_eq!(
|
||||
super::new("src/testdata/single_line_file.txt")[0],
|
||||
"a b c".to_string()
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_two_channels_one_bar() {
|
||||
assert_eq!(
|
||||
super::new("src/testdata/two_channels_one_bar.txt")[0],
|
||||
"2*a 2*b 2*c".to_string()
|
||||
);
|
||||
assert_eq!(
|
||||
super::new("src/testdata/two_channels_one_bar.txt")[1],
|
||||
". a5 . b5 . c5".to_string()
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_two_channels_two_bars() {
|
||||
assert_eq!(
|
||||
super::new("src/testdata/two_channels_two_bars.txt")[0],
|
||||
"a b c d e f".to_string()
|
||||
);
|
||||
assert_eq!(
|
||||
super::new("src/testdata/two_channels_two_bars.txt")[1],
|
||||
"2*a 2*b 2*c 2*d 2*e 2*f".to_string()
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn drop_comment_lines() {
|
||||
assert_eq!(
|
||||
super::new("# hello\n # world\na b c")[0],
|
||||
"a b c".to_string()
|
||||
);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn drop_comment_trailer() {
|
||||
assert_eq!(super::new("a b c # hello world")[0], "a b c".to_string());
|
||||
}
|
||||
}
|
||||
61
src/player.rs
Normal file
61
src/player.rs
Normal file
@@ -0,0 +1,61 @@
|
||||
use itertools::Itertools;
|
||||
|
||||
use crate::seq;
|
||||
|
||||
pub struct Player {
|
||||
bpm: usize,
|
||||
params: tinyaudio::OutputDeviceParameters,
|
||||
device: Option<tinyaudio::OutputDevice>,
|
||||
}
|
||||
|
||||
pub fn new(sample_rate: usize, bpm: usize) -> Player {
|
||||
let samples_per_beat = sample_rate / bpm * 60;
|
||||
Player {
|
||||
bpm: bpm,
|
||||
params: tinyaudio::prelude::OutputDeviceParameters {
|
||||
channels_count: 2,
|
||||
sample_rate: sample_rate,
|
||||
channel_sample_count: samples_per_beat,
|
||||
},
|
||||
device: None,
|
||||
}
|
||||
}
|
||||
|
||||
impl Player {
|
||||
pub fn play(&mut self, s: seq::SynSeq) {
|
||||
self.play_with_sync(s, true);
|
||||
}
|
||||
|
||||
pub fn play_async(&mut self, s: seq::SynSeq) {
|
||||
self.play_with_sync(s, false);
|
||||
}
|
||||
|
||||
fn play_with_sync(&mut self, mut s: seq::SynSeq, sync: bool) {
|
||||
let sample_count = (self.params.channel_sample_count) as usize;
|
||||
let mut left: Vec<f32> = vec![0_f32; sample_count];
|
||||
let mut right: Vec<f32> = vec![0_f32; sample_count];
|
||||
|
||||
let beats = s.beats() + 3;
|
||||
let duration = 60 * beats / self.bpm;
|
||||
|
||||
if self.device.is_some() {
|
||||
self.device.as_mut().expect("lost device").close();
|
||||
}
|
||||
self.device = Some(
|
||||
tinyaudio::prelude::run_output_device(self.params, {
|
||||
move |data| {
|
||||
s.render(&mut left[..], &mut right[..]); // put in a state of rendering the next loop of these notes
|
||||
for (i, value) in left.iter().interleave(right.iter()).enumerate() {
|
||||
data[i] = *value;
|
||||
}
|
||||
}
|
||||
})
|
||||
.unwrap(),
|
||||
);
|
||||
|
||||
if sync {
|
||||
// Wait it out
|
||||
std::thread::sleep(std::time::Duration::from_secs(duration as u64));
|
||||
}
|
||||
}
|
||||
}
|
||||
171
src/seq.rs
Normal file
171
src/seq.rs
Normal file
@@ -0,0 +1,171 @@
|
||||
use crate::syn;
|
||||
use crate::tone;
|
||||
|
||||
pub struct SynSeq {
|
||||
seqs: std::collections::HashMap<i32, Seq>,
|
||||
syn: syn::Syn,
|
||||
}
|
||||
|
||||
pub fn new_syn(syn: syn::Syn) -> SynSeq {
|
||||
SynSeq {
|
||||
seqs: std::collections::HashMap::new(),
|
||||
syn: syn,
|
||||
}
|
||||
}
|
||||
|
||||
impl SynSeq {
|
||||
pub fn append<S: ToString>(&mut self, ch: i32, s: S) {
|
||||
match self.seqs.get_mut(&ch) {
|
||||
Some(seq) => seq.append(s),
|
||||
None => {
|
||||
let mut seq = new();
|
||||
seq.append(s);
|
||||
self.seqs.insert(ch, seq);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
pub fn render(&mut self, left: &mut [f32], right: &mut [f32]) {
|
||||
for (ch, seq) in self.seqs.iter_mut() {
|
||||
let ch = ch.clone();
|
||||
match seq.pop() {
|
||||
(Some(tone), changed) if changed => self.syn.set(ch, Some(tone.clone())),
|
||||
(None, changed) if changed => {
|
||||
self.syn.set(ch, None);
|
||||
}
|
||||
_ => {}
|
||||
};
|
||||
}
|
||||
self.syn.render(left, right);
|
||||
}
|
||||
|
||||
pub fn beats(&self) -> usize {
|
||||
let mut longest = 0 as usize;
|
||||
for (_, seq) in self.seqs.iter() {
|
||||
longest = if seq.len() > longest {
|
||||
seq.len()
|
||||
} else {
|
||||
longest
|
||||
};
|
||||
}
|
||||
longest
|
||||
}
|
||||
}
|
||||
|
||||
#[derive(PartialEq)]
|
||||
pub struct Seq {
|
||||
beats: Vec<(usize, tone::Tone)>,
|
||||
had_more: bool,
|
||||
eof: bool,
|
||||
}
|
||||
|
||||
fn new() -> Seq {
|
||||
Seq::new()
|
||||
}
|
||||
|
||||
impl Seq {
|
||||
fn new() -> Seq {
|
||||
Seq {
|
||||
beats: vec![],
|
||||
had_more: false,
|
||||
eof: false,
|
||||
}
|
||||
}
|
||||
|
||||
fn len(&self) -> usize {
|
||||
let mut sum = 0 as usize;
|
||||
for beat in self.beats.iter() {
|
||||
sum += beat.0 as usize;
|
||||
}
|
||||
sum
|
||||
}
|
||||
|
||||
fn pop(&mut self) -> (Option<tone::Tone>, bool) {
|
||||
let mut changed = !self.had_more && self.beats.len() > 0;
|
||||
self.had_more = match self.beats.len() {
|
||||
0 => false,
|
||||
_ => self.beats[0].0 > 1,
|
||||
};
|
||||
let eof_before = self.eof;
|
||||
self.eof = self.beats.len() == 0;
|
||||
if !eof_before && self.eof {
|
||||
changed = true;
|
||||
}
|
||||
let tone = self._pop();
|
||||
(tone, changed)
|
||||
}
|
||||
|
||||
fn _pop(&mut self) -> Option<tone::Tone> {
|
||||
match self.beats.len() {
|
||||
0 => None,
|
||||
_ => match self.beats[0].0 {
|
||||
1 => Some(self.beats.remove(0).1),
|
||||
_ => {
|
||||
self.beats[0].0 -= 1;
|
||||
Some(self.beats[0].1.clone())
|
||||
}
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
fn append<S: ToString>(&mut self, s: S) {
|
||||
let s: String = s.to_string();
|
||||
let s: &str = s.as_ref();
|
||||
for split in s.split_whitespace() {
|
||||
self.append_one(split.to_string());
|
||||
}
|
||||
}
|
||||
|
||||
fn append_one(&mut self, s: String) {
|
||||
let re = regex::Regex::new(r"^((?<count>[0-9]+)\*)?(?<tone>.*)$").unwrap();
|
||||
let captures = re.captures(&s).unwrap();
|
||||
|
||||
let n = match captures.name("count") {
|
||||
Some(number) if number.as_str().len() > 0 => number.as_str().parse::<usize>().unwrap(),
|
||||
_ => 1,
|
||||
} as usize;
|
||||
|
||||
let tone_s = captures.name("tone").unwrap().as_str();
|
||||
let tone = tone::new(tone_s);
|
||||
self.beats.push((n, tone));
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn test_seq() {
|
||||
let mut seq = new();
|
||||
|
||||
seq.append("c c");
|
||||
seq.append("4*d");
|
||||
seq.append("2*-");
|
||||
seq.append("g 2*e");
|
||||
|
||||
assert_eq!(seq.beats.len(), 6);
|
||||
assert_eq!(seq.len(), 11);
|
||||
|
||||
assert_eq!(seq.beats[0], (1 as usize, tone::new("c")));
|
||||
assert_eq!(seq.beats[1], (1 as usize, tone::new("c")));
|
||||
assert_eq!(seq.beats[2], (4 as usize, tone::new("d")));
|
||||
assert_eq!(seq.beats[3], (2 as usize, tone::new("!")));
|
||||
assert_eq!(seq.beats[4], (1 as usize, tone::new("g")));
|
||||
assert_eq!(seq.beats[5], (2 as usize, tone::new("e")));
|
||||
|
||||
assert_eq!(seq.pop(), (Some(tone::new("c")), true));
|
||||
assert_eq!(seq.pop(), (Some(tone::new("c")), true));
|
||||
assert_eq!(seq.pop(), (Some(tone::new("d")), true));
|
||||
for _ in 1..4 {
|
||||
assert_eq!(seq.pop(), (Some(tone::new("d")), false));
|
||||
}
|
||||
assert_eq!(seq.pop(), (Some(tone::new(".")), true));
|
||||
assert_eq!(seq.pop(), (Some(tone::new("?")), false));
|
||||
assert_eq!(seq.pop(), (Some(tone::new("g")), true));
|
||||
assert_eq!(seq.pop(), (Some(tone::new("e")), true));
|
||||
assert_eq!(seq.pop(), (Some(tone::new("e")), false));
|
||||
assert_eq!(seq.pop(), (None, true), "shouldve newly had no more");
|
||||
assert_eq!(seq.pop(), (None, false), "shouldve still had no more");
|
||||
}
|
||||
}
|
||||
113
src/syn.rs
113
src/syn.rs
@@ -1,13 +1,15 @@
|
||||
use rustysynth::Synthesizer;
|
||||
use rustysynth::SoundFont;
|
||||
use rustysynth::Synthesizer;
|
||||
use rustysynth::SynthesizerSettings;
|
||||
use std::sync::Arc;
|
||||
|
||||
use crate::tone;
|
||||
|
||||
pub enum Syn {
|
||||
Real(Synthesizer),
|
||||
Text{
|
||||
m: std::collections::HashMap<i32, std::collections::HashMap<i32, i32>>,
|
||||
i: u32,
|
||||
Text {
|
||||
m: std::collections::HashMap<i32, std::collections::HashMap<i32, i32>>,
|
||||
i: u32,
|
||||
},
|
||||
}
|
||||
|
||||
@@ -15,7 +17,10 @@ impl Syn {
|
||||
pub fn new(debug: bool, sound_font: String, sample_rate: usize) -> Syn {
|
||||
match debug {
|
||||
false => Syn::new_real(sound_font, sample_rate),
|
||||
true => Syn::Text{m: std::collections::HashMap::new(), i: 0},
|
||||
true => Syn::Text {
|
||||
m: std::collections::HashMap::new(),
|
||||
i: 0,
|
||||
},
|
||||
}
|
||||
}
|
||||
|
||||
@@ -29,44 +34,92 @@ impl Syn {
|
||||
Syn::Real(synthesizer)
|
||||
}
|
||||
|
||||
pub fn note_on(&mut self, a: i32, b: i32, c: i32) {
|
||||
pub fn set(&mut self, ch: i32, b: Option<tone::Tone>) {
|
||||
match self {
|
||||
Syn::Real(syn) => syn.note_on(a, b, c),
|
||||
Syn::Text{m, ..} => {
|
||||
eprintln!("note_on({:?}, {:?}, {:?})", a, b, c);
|
||||
match m.get_mut(&a) {
|
||||
Some(m2) => { m2.insert(b, c); },
|
||||
None => {
|
||||
let mut m2 = std::collections::HashMap::new();
|
||||
m2.insert(b, c);
|
||||
m.insert(a, m2);
|
||||
},
|
||||
};
|
||||
},
|
||||
// channel=[0..16)
|
||||
// velocity=[0..128)
|
||||
Syn::Real(syn) => {
|
||||
syn.note_off_all_channel(ch, false);
|
||||
}
|
||||
Syn::Text { m, .. } => {
|
||||
m.remove(&ch);
|
||||
}
|
||||
};
|
||||
if let Some(tone) = b {
|
||||
self.tone_on(ch, tone);
|
||||
};
|
||||
}
|
||||
|
||||
pub fn note_off(&mut self, a: i32, b: i32) {
|
||||
fn tone_on(&mut self, ch: i32, b: tone::Tone) {
|
||||
match self {
|
||||
Syn::Real(syn) => syn.note_off(a, b),
|
||||
Syn::Text{m, ..} => {
|
||||
eprintln!("note_off({:?}, {:?})", a, b);
|
||||
match m.get_mut(&a) {
|
||||
Some(m) => { m.remove(&b); },
|
||||
None => {},
|
||||
// channel=[0..16)
|
||||
// velocity=[0..128)
|
||||
Syn::Real(syn) => syn.note_on(ch, b.i32(), 127),
|
||||
Syn::Text { m, .. } => {
|
||||
match m.get_mut(&ch) {
|
||||
Some(m2) => {
|
||||
m2.insert(b.i32(), 127);
|
||||
}
|
||||
None => {
|
||||
let mut m2 = std::collections::HashMap::new();
|
||||
m2.insert(b.i32(), 127);
|
||||
m.insert(ch, m2);
|
||||
}
|
||||
};
|
||||
|
||||
},
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
pub fn render(&mut self, a: &mut [f32], b: &mut [f32]) {
|
||||
match self {
|
||||
Syn::Real(syn) => syn.render(a, b),
|
||||
Syn::Text{m, i} => {
|
||||
eprintln!("render[{}]({:?})", i, m);
|
||||
Syn::Text { m, i } => {
|
||||
eprintln!(
|
||||
"{} | render[{}]({:?})",
|
||||
chrono::prelude::Utc::now(),
|
||||
i,
|
||||
m.iter()
|
||||
.map(|tuple| (
|
||||
tuple.0,
|
||||
tuple
|
||||
.1
|
||||
.iter()
|
||||
.map(|v| tone::new(v.0.to_string()).string())
|
||||
.collect::<Vec<_>>()
|
||||
))
|
||||
.collect::<Vec<_>>(),
|
||||
);
|
||||
*i += 1;
|
||||
},
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
use super::*;
|
||||
|
||||
#[test]
|
||||
fn test_new_real() {
|
||||
let mut syn = Syn::new(false, "super_small_font.sf2".to_string(), 44100);
|
||||
|
||||
syn.tone_on(0, tone::new("c"));
|
||||
syn.tone_on(0, tone::new("d"));
|
||||
|
||||
let mut buffer1 = Vec::<f32>::new();
|
||||
let mut buffer2 = Vec::<f32>::new();
|
||||
syn.render(&mut buffer1, &mut buffer2);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_text() {
|
||||
let mut syn = Syn::new(true, ".sf2".to_string(), 1);
|
||||
|
||||
syn.tone_on(0, tone::new("c"));
|
||||
syn.tone_on(0, tone::new("d"));
|
||||
|
||||
let mut buffer1 = Vec::<f32>::new();
|
||||
let mut buffer2 = Vec::<f32>::new();
|
||||
syn.render(&mut buffer1, &mut buffer2);
|
||||
}
|
||||
}
|
||||
|
||||
9
src/testdata/battle_against_a_true_hero.txt
vendored
Normal file
9
src/testdata/battle_against_a_true_hero.txt
vendored
Normal file
@@ -0,0 +1,9 @@
|
||||
# https://www.musicnotes.com/sheetmusic/chrissy-ricker/battle-against-a-true-hero-easy/MN0235808
|
||||
# c major
|
||||
|
||||
CM4 CM5 CM5 CM1 CM1 CM2 . .
|
||||
|
||||
CM3 CM0 CM0 CM4 CM4 . . .
|
||||
|
||||
CM4 CM5 CM5 CM1 CM1 CM2 . .
|
||||
|
||||
12
src/testdata/chord_progressions.txt
vendored
Normal file
12
src/testdata/chord_progressions.txt
vendored
Normal file
@@ -0,0 +1,12 @@
|
||||
# x+0 x+5 x+7 x+0
|
||||
CM0 CM3 CM4 CM0
|
||||
|
||||
.
|
||||
|
||||
# x+2 x+7 x+0
|
||||
CM1 CM4 CM0
|
||||
|
||||
.
|
||||
|
||||
# x+0 x+7 x+9 x+5
|
||||
CM0 CM4 CM5 CM3
|
||||
5
src/testdata/fifth_wheel.txt
vendored
Normal file
5
src/testdata/fifth_wheel.txt
vendored
Normal file
@@ -0,0 +1,5 @@
|
||||
# fifth wheel = +7 semitones
|
||||
# f -> c -> g -> d -> a -> e -> b
|
||||
|
||||
f c g d a e b
|
||||
ff cf gf df af ef bf
|
||||
19
src/testdata/major_chord_progressions.txt
vendored
Normal file
19
src/testdata/major_chord_progressions.txt
vendored
Normal file
@@ -0,0 +1,19 @@
|
||||
CM0 CM3 CM4 CM0
|
||||
CM0T CM3T CM4T CM0T
|
||||
CM0f CM3f CM4f CM0f
|
||||
|
||||
.
|
||||
.
|
||||
.
|
||||
|
||||
CM1 CM4 CM0
|
||||
CM1T CM4T CM0T
|
||||
CM1f CM4f CM0f
|
||||
|
||||
.
|
||||
.
|
||||
.
|
||||
|
||||
CM0 CM4 CM5 CM3
|
||||
CM0T CM4T CM5T CM3T
|
||||
CM0f CM4f CM5f CM3f
|
||||
27
src/testdata/major_chords.txt
vendored
Normal file
27
src/testdata/major_chords.txt
vendored
Normal file
@@ -0,0 +1,27 @@
|
||||
# major CHORD = root + major_third + perfect_fifth
|
||||
# c = c,e,g
|
||||
|
||||
f
|
||||
a
|
||||
c
|
||||
|
||||
c
|
||||
e
|
||||
g
|
||||
|
||||
g
|
||||
b
|
||||
d
|
||||
|
||||
d
|
||||
f+
|
||||
a
|
||||
|
||||
a
|
||||
c+
|
||||
e
|
||||
|
||||
e
|
||||
g+
|
||||
b
|
||||
|
||||
18
src/testdata/major_third.txt
vendored
Normal file
18
src/testdata/major_third.txt
vendored
Normal file
@@ -0,0 +1,18 @@
|
||||
# major third
|
||||
# +2 letters (the third letter) with exactly 2 sharps between
|
||||
|
||||
# c,e c+,f d,f+
|
||||
c c+ d
|
||||
cT c+T dT
|
||||
|
||||
# d+,g e,g+ f,a
|
||||
d+ e f
|
||||
d+T eT fT
|
||||
|
||||
# f+,a+ g,b g+,c
|
||||
f+ g g+
|
||||
f+T gT g+T
|
||||
|
||||
# a,c+ a+,d b,d+
|
||||
a a+ b
|
||||
aT a+T bT
|
||||
27
src/testdata/minor_chords.txt
vendored
Normal file
27
src/testdata/minor_chords.txt
vendored
Normal file
@@ -0,0 +1,27 @@
|
||||
# minor CHORD = root + minor_third + perfect_fifth
|
||||
# c = c,e-,g
|
||||
|
||||
f
|
||||
g+
|
||||
c
|
||||
|
||||
c
|
||||
e-
|
||||
g
|
||||
|
||||
g
|
||||
a+
|
||||
d
|
||||
|
||||
d
|
||||
f
|
||||
a
|
||||
|
||||
a
|
||||
c
|
||||
e
|
||||
|
||||
e
|
||||
g
|
||||
b
|
||||
|
||||
18
src/testdata/minor_third.txt
vendored
Normal file
18
src/testdata/minor_third.txt
vendored
Normal file
@@ -0,0 +1,18 @@
|
||||
# minor third
|
||||
# +2 letters (the third letter) with exactly 1 half step between
|
||||
|
||||
# c,e- c+,e d,f
|
||||
c c+ d
|
||||
ct c+t dt
|
||||
|
||||
# d+,f+ e,g f,g+
|
||||
d+ e f
|
||||
d+t et ft
|
||||
|
||||
# f+,a g,a+ g+,b
|
||||
f+ g g+
|
||||
f+t gt g+t
|
||||
|
||||
# a,c a+,c+ b,d
|
||||
a a+ b
|
||||
at a+t bt
|
||||
12
src/testdata/plus_three_sounds_nice.txt
vendored
Normal file
12
src/testdata/plus_three_sounds_nice.txt
vendored
Normal file
@@ -0,0 +1,12 @@
|
||||
#CM0 CM3 CM4 CM0
|
||||
#CM1 CM4 CM0
|
||||
#CM0 CM4 CM5 CM3
|
||||
|
||||
# 1 4 6 4 # pop101
|
||||
#GM0 GM3 GM5 GM3
|
||||
|
||||
60
|
||||
63
|
||||
66
|
||||
69
|
||||
72
|
||||
12
src/testdata/sandbox.txt
vendored
Normal file
12
src/testdata/sandbox.txt
vendored
Normal file
@@ -0,0 +1,12 @@
|
||||
#CM0 CM3 CM4 CM0
|
||||
#CM1 CM4 CM0
|
||||
#CM0 CM4 CM5 CM3
|
||||
|
||||
# 1 4 6 4 # pop101
|
||||
#GM0 GM3 GM5 GM3
|
||||
|
||||
60
|
||||
63
|
||||
66
|
||||
69
|
||||
72
|
||||
1
src/testdata/single_line_file.txt
vendored
Normal file
1
src/testdata/single_line_file.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
a b c
|
||||
2
src/testdata/two_channels_one_bar.txt
vendored
Normal file
2
src/testdata/two_channels_one_bar.txt
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
2*a 2*b 2*c
|
||||
. a5 . b5 . c5
|
||||
5
src/testdata/two_channels_two_bars.txt
vendored
Normal file
5
src/testdata/two_channels_two_bars.txt
vendored
Normal file
@@ -0,0 +1,5 @@
|
||||
a b c
|
||||
2*a 2*b 2*c
|
||||
|
||||
d e f
|
||||
2*d 2*e 2*f
|
||||
241
src/tone.rs
Normal file
241
src/tone.rs
Normal file
@@ -0,0 +1,241 @@
|
||||
#[derive(Debug, PartialEq, Clone)]
|
||||
pub struct Tone(i32);
|
||||
|
||||
// new parses [a-g][+-]?[1-5]? to tone, sharp or flat, down 2 octave..up 1 octave
|
||||
pub fn new<S: ToString>(s: S) -> Tone {
|
||||
Tone::new(s.to_string())
|
||||
}
|
||||
|
||||
impl Tone {
|
||||
fn new(s: String) -> Tone {
|
||||
let re = regex::Regex::new(r"^(((?<letter>^[a-g])(?<sharpness>[+-]?)|(?<major>[A-G]M[0-6])|(?<minor>[A-G]m[0-6]))(?<octave>[1-5]?)((?<major_third>T)|(?<minor_third>t)|(?<fifth>f))?|(?<numeric>[0-9]+)|(?<rest>[^a-gA-G0-9]))$").unwrap();
|
||||
let captures = re
|
||||
.captures(&s)
|
||||
.expect(format!("tone '{}' does not match regex", s).as_ref());
|
||||
Tone(match captures.name("numeric") {
|
||||
Some(number) => number.as_str().parse::<i32>().unwrap(),
|
||||
None => match captures.name("rest") {
|
||||
Some(_) => 0,
|
||||
_ => {
|
||||
let mut result = match captures.name("letter") {
|
||||
Some(letter) => Tone::char_to_middle_i32(letter.as_str()),
|
||||
None => match captures.name("major") {
|
||||
Some(major) => {
|
||||
Tone::char_to_middle_i32(&major.as_str()[..1])
|
||||
+ match &major.as_str()[2..] {
|
||||
"0" => 0,
|
||||
"1" => 2,
|
||||
"2" => 4,
|
||||
"3" => 5,
|
||||
"4" => 7,
|
||||
"5" => 9,
|
||||
_ => 11,
|
||||
}
|
||||
}
|
||||
None => {
|
||||
let minor = captures.name("minor").unwrap();
|
||||
Tone::char_to_middle_i32(&minor.as_str()[..1])
|
||||
+ match &minor.as_str()[2..] {
|
||||
"0" => 0,
|
||||
"1" => 2,
|
||||
"2" => 3,
|
||||
"3" => 5,
|
||||
"4" => 7,
|
||||
"5" => 8,
|
||||
_ => 10,
|
||||
}
|
||||
}
|
||||
},
|
||||
} as i32;
|
||||
|
||||
result += match captures.name("sharpness") {
|
||||
Some(sharpness) => match sharpness.as_str() {
|
||||
"+" => 1,
|
||||
"-" => -1,
|
||||
_ => 0,
|
||||
},
|
||||
None => 0,
|
||||
} as i32;
|
||||
|
||||
result += match captures.name("octave") {
|
||||
Some(octave) => match octave.as_str() {
|
||||
"" => 0,
|
||||
_ => (octave.as_str().parse::<i32>().unwrap() - 3) * 12,
|
||||
},
|
||||
None => 0,
|
||||
} as i32;
|
||||
|
||||
result += match captures.name("major_third") {
|
||||
Some(_) => 4, // TODO not all are good
|
||||
None => 0,
|
||||
} as i32;
|
||||
|
||||
result += match captures.name("minor_third") {
|
||||
Some(_) => 3, // TODO not all are good
|
||||
None => 0,
|
||||
} as i32;
|
||||
|
||||
result += match captures.name("fifth") {
|
||||
Some(_) => 7,
|
||||
None => 0,
|
||||
} as i32;
|
||||
|
||||
result
|
||||
}
|
||||
},
|
||||
})
|
||||
}
|
||||
|
||||
fn char_to_middle_i32(c: &str) -> i32 {
|
||||
match c.to_lowercase().as_str() {
|
||||
"a" => 57,
|
||||
"b" => 59,
|
||||
"c" => 60,
|
||||
"d" => 62,
|
||||
"e" => 64,
|
||||
"f" => 65,
|
||||
_ => 67,
|
||||
}
|
||||
}
|
||||
|
||||
pub fn i32(&self) -> i32 {
|
||||
self.0
|
||||
}
|
||||
|
||||
pub fn string(&self) -> String {
|
||||
let v = self.i32();
|
||||
let modifier = if v > 0 && v < 57 {
|
||||
"-"
|
||||
} else if v >= 69 {
|
||||
"+"
|
||||
} else {
|
||||
""
|
||||
};
|
||||
modifier.to_string()
|
||||
+ match v {
|
||||
45 | 57 | 69 => "a",
|
||||
46 | 58 | 70 => "a+",
|
||||
47 | 59 | 71 => "b",
|
||||
48 | 60 | 72 => "c",
|
||||
49 | 61 | 73 => "c+",
|
||||
50 | 62 | 74 => "d",
|
||||
51 | 63 | 75 => "d+",
|
||||
52 | 64 | 76 => "e",
|
||||
53 | 65 | 77 => "f",
|
||||
54 | 66 | 78 => "f+",
|
||||
55 | 67 | 79 => "g",
|
||||
56 | 68 | 80 => "g+",
|
||||
0 => " ",
|
||||
_ => "?",
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
#[test]
|
||||
fn test_tone_new() {
|
||||
eprintln!("numeric");
|
||||
assert_eq!(super::new("60").i32(), 60);
|
||||
|
||||
eprintln!("rests");
|
||||
assert_eq!(super::new("-").i32(), 0);
|
||||
assert_eq!(super::new(".").i32(), 0);
|
||||
|
||||
eprintln!("alpha");
|
||||
assert_eq!(super::new("c").i32(), 60);
|
||||
assert_eq!(super::new("e").i32(), 64);
|
||||
assert_eq!(super::new("g").i32(), 67);
|
||||
|
||||
eprintln!("alpha mod");
|
||||
assert_eq!(super::new("c+").i32(), 60 + 1);
|
||||
assert_eq!(super::new("c-").i32(), 60 - 1);
|
||||
|
||||
eprintln!("alpha octave");
|
||||
assert_eq!(super::new("c3").i32(), 60);
|
||||
assert_eq!(super::new("c4").i32(), 60 + 12);
|
||||
|
||||
eprintln!("alpha mod octave");
|
||||
assert_eq!(super::new("c+4").i32(), 60 + 12 + 1);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_new_major_third() {
|
||||
assert_eq!(super::new("cT").i32(), 60 + 4);
|
||||
assert_eq!(super::new("c1T").i32(), 60 + 4 - 24);
|
||||
|
||||
assert_eq!(super::new("c+T").i32(), 60 + 1 + 4);
|
||||
assert_eq!(super::new("c+1T").i32(), 60 + 1 + 4 - 24);
|
||||
|
||||
assert_eq!(super::new("CM0T").i32(), 60 + 4);
|
||||
assert_eq!(super::new("CM01T").i32(), 60 + 4 - 24);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_new_minor_third() {
|
||||
assert_eq!(super::new("ct").i32(), 60 + 3);
|
||||
assert_eq!(super::new("c1t").i32(), 60 + 3 - 24);
|
||||
|
||||
assert_eq!(super::new("c+t").i32(), 60 + 1 + 3);
|
||||
assert_eq!(super::new("c+1t").i32(), 60 + 1 + 3 - 24);
|
||||
|
||||
assert_eq!(super::new("CM0t").i32(), 60 + 3);
|
||||
assert_eq!(super::new("CM01t").i32(), 60 + 3 - 24);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_new_fifth() {
|
||||
assert_eq!(super::new("cf").i32(), 60 + 7);
|
||||
assert_eq!(super::new("c1f").i32(), 60 + 7 - 24);
|
||||
|
||||
assert_eq!(super::new("c+f").i32(), 60 + 1 + 7);
|
||||
assert_eq!(super::new("c+1f").i32(), 60 + 1 + 7 - 24);
|
||||
|
||||
assert_eq!(super::new("CM0f").i32(), 60 + 7);
|
||||
assert_eq!(super::new("CM01f").i32(), 60 + 7 - 24);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_c_major() {
|
||||
assert_eq!(super::new("CM0"), super::new("c"));
|
||||
assert_eq!(super::new("CM1"), super::new("d"));
|
||||
assert_eq!(super::new("CM2"), super::new("e"));
|
||||
assert_eq!(super::new("CM3"), super::new("f"));
|
||||
assert_eq!(super::new("CM4"), super::new("g"));
|
||||
assert_eq!(super::new("CM5"), super::new("a4"));
|
||||
assert_eq!(super::new("CM6"), super::new("b4"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_c_minor() {
|
||||
assert_eq!(super::new("Cm0"), super::new("c"));
|
||||
assert_eq!(super::new("Cm1"), super::new("d"));
|
||||
assert_eq!(super::new("Cm2"), super::new("e-"));
|
||||
assert_eq!(super::new("Cm3"), super::new("f"));
|
||||
assert_eq!(super::new("Cm4"), super::new("g"));
|
||||
assert_eq!(super::new("Cm5"), super::new("a-4"));
|
||||
assert_eq!(super::new("Cm6"), super::new("b-4"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_a_major() {
|
||||
assert_eq!(super::new("AM0"), super::new("a"));
|
||||
assert_eq!(super::new("AM1"), super::new("b"));
|
||||
assert_eq!(super::new("AM2"), super::new("c+"));
|
||||
assert_eq!(super::new("AM3"), super::new("d"));
|
||||
assert_eq!(super::new("AM4"), super::new("e"));
|
||||
assert_eq!(super::new("AM5"), super::new("f+"));
|
||||
assert_eq!(super::new("AM6"), super::new("g+"));
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn test_tone_a_minor() {
|
||||
assert_eq!(super::new("Am0"), super::new("a"));
|
||||
assert_eq!(super::new("Am1"), super::new("b"));
|
||||
assert_eq!(super::new("Am2"), super::new("c"));
|
||||
assert_eq!(super::new("Am3"), super::new("d"));
|
||||
assert_eq!(super::new("Am4"), super::new("e"));
|
||||
assert_eq!(super::new("Am5"), super::new("f"));
|
||||
assert_eq!(super::new("Am6"), super::new("g"));
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user