mirror of
https://codeberg.org/unspeaker/tek.git
synced 2025-12-06 11:46:41 +01:00
define midi ins and audio outs in project
This commit is contained in:
parent
58cd51dfbf
commit
c4c1271c32
5 changed files with 56 additions and 35 deletions
|
|
@ -10,7 +10,7 @@ See `demos/project.edn` for the initial contents of the session.
|
||||||
* Rust toolchain
|
* Rust toolchain
|
||||||
* JACK or Pipewire
|
* JACK or Pipewire
|
||||||
|
|
||||||
## Recommended
|
### Recommended
|
||||||
|
|
||||||
* MIDI controller
|
* MIDI controller
|
||||||
* Samples at ~/Lab/Music/pak
|
* Samples at ~/Lab/Music/pak
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,8 @@
|
||||||
(bpm 150)
|
(bpm 150)
|
||||||
|
|
||||||
|
(midi-in "nanoKEY Studio.*capture.*")
|
||||||
|
(audio-out "Komplete.+:playback_FL", "Komplete.+:playback_FR")
|
||||||
|
|
||||||
(scene { :name "Intro" } _ 0 _ _)
|
(scene { :name "Intro" } _ 0 _ _)
|
||||||
(scene { :name "Hook" } 0 1 0 _)
|
(scene { :name "Hook" } 0 1 0 _)
|
||||||
(scene { :name "Verse" } 1 2 1 _)
|
(scene { :name "Verse" } 1 2 1 _)
|
||||||
|
|
|
||||||
22
src/edn.rs
22
src/edn.rs
|
|
@ -59,6 +59,28 @@ impl App {
|
||||||
Some(Edn::Symbol("track")) => {
|
Some(Edn::Symbol("track")) => {
|
||||||
Track::load_edn(self, &items[1..])?;
|
Track::load_edn(self, &items[1..])?;
|
||||||
},
|
},
|
||||||
|
Some(Edn::Symbol("midi-in")) => {
|
||||||
|
self.midi_ins = items[1..].iter().map(|x|match x {
|
||||||
|
Edn::Str(n) => n.to_string(),
|
||||||
|
_ => panic!("unexpected midi-in")
|
||||||
|
}).collect::<Vec<_>>();
|
||||||
|
},
|
||||||
|
Some(Edn::Symbol("audio-out")) => {
|
||||||
|
let client = self.client();
|
||||||
|
self.audio_outs = items[1..].iter().map(|x|match x {
|
||||||
|
Edn::Str(n) => n.to_string(),
|
||||||
|
_ => panic!("unexpected midi-in")
|
||||||
|
}).collect::<Vec<_>>()
|
||||||
|
.iter()
|
||||||
|
.map(|name|client
|
||||||
|
.ports(Some(name), None, PortFlags::empty())
|
||||||
|
.get(0)
|
||||||
|
.map(|name|client.port_by_name(name)))
|
||||||
|
.flatten()
|
||||||
|
.filter_map(|x|x)
|
||||||
|
.map(Arc::new)
|
||||||
|
.collect();
|
||||||
|
},
|
||||||
_ => panic!("unexpected edn: {:?}", items.get(0))
|
_ => panic!("unexpected edn: {:?}", items.get(0))
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
|
||||||
60
src/main.rs
60
src/main.rs
|
|
@ -15,44 +15,40 @@ use crate::{core::*, model::*};
|
||||||
|
|
||||||
/// Application entrypoint.
|
/// Application entrypoint.
|
||||||
pub fn main () -> Usually<()> {
|
pub fn main () -> Usually<()> {
|
||||||
let app = App::from_edn(include_str!("../demos/project.edn"))?
|
run(App::from_edn(include_str!("../demos/project.edn"))?.activate(Some(|app: &Arc<RwLock<App>>| {
|
||||||
.with_midi_ins(&["nanoKEY Studio.*capture.*"])?
|
|
||||||
.with_audio_outs(&["Komplete.+:playback_FL", "Komplete.+:playback_FR"])?
|
|
||||||
.activate(Some(|app: &Arc<RwLock<App>>| {
|
|
||||||
|
|
||||||
let (midi_in, mut midi_outs) = {
|
let (midi_in, mut midi_outs) = {
|
||||||
let app = app.read().unwrap();
|
let app = app.read().unwrap();
|
||||||
let jack = app.jack.as_ref().unwrap();
|
let jack = app.jack.as_ref().unwrap();
|
||||||
let midi_in = jack.register_port("midi-in", MidiIn)?;
|
let midi_in = jack.register_port("midi-in", MidiIn)?;
|
||||||
let midi_outs = app.tracks.iter()
|
let midi_outs = app.tracks.iter()
|
||||||
.map(|t|Some(jack.register_port(&t.name, MidiOut).unwrap()))
|
.map(|t|Some(jack.register_port(&t.name, MidiOut).unwrap()))
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
(midi_in, midi_outs)
|
(midi_in, midi_outs)
|
||||||
};
|
};
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut app = app.write().unwrap();
|
let mut app = app.write().unwrap();
|
||||||
let jack = app.jack.as_ref().unwrap();
|
let jack = app.jack.as_ref().unwrap();
|
||||||
for name in app.midi_ins.iter() {
|
for name in app.midi_ins.iter() {
|
||||||
for port in jack.client().ports(Some(name), None, PortFlags::empty()).iter() {
|
for port in jack.client().ports(Some(name), None, PortFlags::empty()).iter() {
|
||||||
if let Some(port) = jack.client().port_by_name(port) {
|
if let Some(port) = jack.client().port_by_name(port) {
|
||||||
jack.client().connect_ports(&port, &midi_in)?;
|
jack.client().connect_ports(&port, &midi_in)?;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
app.midi_in = Some(Arc::new(midi_in));
|
|
||||||
for (index, track) in app.tracks.iter_mut().enumerate() {
|
|
||||||
track.midi_out = midi_outs[index].take();
|
|
||||||
}
|
|
||||||
for track in app.tracks.iter() {
|
|
||||||
track.connect_first_device()?;
|
|
||||||
track.connect_last_device(&app)?;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
app.midi_in = Some(Arc::new(midi_in));
|
||||||
|
for (index, track) in app.tracks.iter_mut().enumerate() {
|
||||||
|
track.midi_out = midi_outs[index].take();
|
||||||
|
}
|
||||||
|
for track in app.tracks.iter() {
|
||||||
|
track.connect_first_device()?;
|
||||||
|
track.connect_last_device(&app)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
|
||||||
}))?;
|
}))?)?;
|
||||||
run(app)?;
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -56,10 +56,10 @@ pub struct App {
|
||||||
/// Paths to user directories
|
/// Paths to user directories
|
||||||
xdg: Option<Arc<XdgApp>>,
|
xdg: Option<Arc<XdgApp>>,
|
||||||
/// Main audio outputs.
|
/// Main audio outputs.
|
||||||
audio_outs: Vec<Arc<Port<Unowned>>>,
|
pub audio_outs: Vec<Arc<Port<Unowned>>>,
|
||||||
/// Number of frames requested by process callback
|
/// Number of frames requested by process callback
|
||||||
chunk_size: usize,
|
chunk_size: usize,
|
||||||
|
/// Transport model and view.
|
||||||
pub transport: TransportToolbar,
|
pub transport: TransportToolbar,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue