mirror of
https://codeberg.org/unspeaker/tek.git
synced 2025-12-08 20:56:43 +01:00
wip: refactor pt.21: api traits
This commit is contained in:
parent
b8708d6b2d
commit
029614631e
10 changed files with 626 additions and 490 deletions
|
|
@ -1,19 +1,104 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
|
||||||
|
pub trait ArrangerModelApi: JackModelApi + ClockModelApi {
|
||||||
|
fn name (&self) -> &Arc<RwLock<String>>;
|
||||||
|
fn phrases (&self) -> &Arc<RwLock<PhrasePool>>;
|
||||||
|
fn tracks (&self) -> &Vec<ArrangerTrack>;
|
||||||
|
fn tracks_mut (&mut self) -> &mut Vec<ArrangerTrack>;
|
||||||
|
fn scenes (&self) -> &Vec<ArrangerScene>;
|
||||||
|
fn scenes_mut (&mut self) -> &mut Vec<ArrangerScene>;
|
||||||
|
|
||||||
|
fn track_default_name (&self) -> String {
|
||||||
|
format!("Track {}", self.tracks().len() + 1)
|
||||||
|
}
|
||||||
|
fn track_add (
|
||||||
|
&mut self, name: Option<&str>, color: Option<ItemColor>
|
||||||
|
) -> Usually<&mut ArrangerTrack> {
|
||||||
|
let name = name.map_or_else(||self.track_default_name(), |x|x.to_string());
|
||||||
|
let track = ArrangerTrack {
|
||||||
|
width: name.len() + 2,
|
||||||
|
color: color.unwrap_or_else(||ItemColor::random()),
|
||||||
|
player: MIDIPlayer::new(&self.jack(), &self.clock(), name.as_str())?,
|
||||||
|
name: Arc::new(name.into()),
|
||||||
|
};
|
||||||
|
self.tracks_mut().push(track);
|
||||||
|
let index = self.tracks().len() - 1;
|
||||||
|
Ok(&mut self.tracks_mut()[index])
|
||||||
|
}
|
||||||
|
fn track_del (&mut self, index: usize) {
|
||||||
|
self.tracks_mut().remove(index);
|
||||||
|
for scene in self.scenes_mut().iter_mut() {
|
||||||
|
scene.clips.remove(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn scene_default_name (&self) -> String {
|
||||||
|
format!("Scene {}", self.scenes().len() + 1)
|
||||||
|
}
|
||||||
|
fn scene_add (
|
||||||
|
&mut self, name: Option<&str>, color: Option<ItemColor>
|
||||||
|
) -> Usually<&mut ArrangerScene> {
|
||||||
|
let name = name.map_or_else(||self.scene_default_name(), |x|x.to_string());
|
||||||
|
let scene = ArrangerScene {
|
||||||
|
name: Arc::new(name.into()),
|
||||||
|
clips: vec![None;self.tracks().len()],
|
||||||
|
color: color.unwrap_or_else(||ItemColor::random()),
|
||||||
|
};
|
||||||
|
self.scenes_mut().push(scene);
|
||||||
|
let index = self.scenes().len() - 1;
|
||||||
|
Ok(&mut self.scenes_mut()[index])
|
||||||
|
}
|
||||||
|
fn scene_del (&mut self, index: usize) {
|
||||||
|
self.scenes_mut().remove(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JackModelApi for ArrangerModel {
|
||||||
|
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
||||||
|
&self.jack
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClockModelApi for ArrangerModel {
|
||||||
|
fn clock (&self) -> &Arc<Clock> {
|
||||||
|
&self.clock
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ArrangerModelApi for ArrangerModel {
|
||||||
|
fn name (&self) -> &Arc<RwLock<String>> {
|
||||||
|
&self.name
|
||||||
|
}
|
||||||
|
fn phrases (&self) -> &Arc<RwLock<PhrasePool>> {
|
||||||
|
&self.phrases
|
||||||
|
}
|
||||||
|
fn tracks (&self) -> &Vec<ArrangerTrack> {
|
||||||
|
&self.tracks
|
||||||
|
}
|
||||||
|
fn tracks_mut (&mut self) -> &mut Vec<ArrangerTrack> {
|
||||||
|
&mut self.tracks
|
||||||
|
}
|
||||||
|
fn scenes (&self) -> &Vec<ArrangerScene> {
|
||||||
|
&self.scenes
|
||||||
|
}
|
||||||
|
fn scenes_mut (&mut self) -> &mut Vec<ArrangerScene> {
|
||||||
|
&mut self.scenes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct ArrangerModel {
|
pub struct ArrangerModel {
|
||||||
/// JACK client handle (needs to not be dropped for standalone mode to work).
|
/// JACK client handle (needs to not be dropped for standalone mode to work).
|
||||||
pub jack: Arc<RwLock<JackClient>>,
|
jack: Arc<RwLock<JackClient>>,
|
||||||
/// Global timebase
|
/// Global timebase
|
||||||
pub clock: Arc<Clock>,
|
clock: Arc<Clock>,
|
||||||
/// Name of arranger
|
/// Name of arranger
|
||||||
pub name: Arc<RwLock<String>>,
|
name: Arc<RwLock<String>>,
|
||||||
/// Collection of phrases.
|
/// Collection of phrases.
|
||||||
pub phrases: Arc<RwLock<PhrasePool>>,
|
phrases: Arc<RwLock<PhrasePool>>,
|
||||||
/// Collection of tracks.
|
/// Collection of tracks.
|
||||||
pub tracks: Vec<ArrangerTrack>,
|
tracks: Vec<ArrangerTrack>,
|
||||||
/// Collection of scenes.
|
/// Collection of scenes.
|
||||||
pub scenes: Vec<ArrangerScene>,
|
scenes: Vec<ArrangerScene>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
|
@ -38,49 +123,20 @@ pub struct ArrangerScene {
|
||||||
pub color: ItemColor,
|
pub color: ItemColor,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ArrangerModel {
|
impl ArrangerTrack {
|
||||||
pub fn is_stopped (&self) -> bool {
|
pub fn longest_name (tracks: &[Self]) -> usize {
|
||||||
*self.clock.playing.read().unwrap() == Some(TransportState::Stopped)
|
tracks.iter().map(|s|s.name.read().unwrap().len()).fold(0, usize::max)
|
||||||
}
|
}
|
||||||
pub fn track_default_name (&self) -> String {
|
|
||||||
format!("Track {}", self.tracks.len() + 1)
|
pub const MIN_WIDTH: usize = 3;
|
||||||
|
|
||||||
|
pub fn width_inc (&mut self) {
|
||||||
|
self.width += 1;
|
||||||
}
|
}
|
||||||
pub fn scene_default_name (&self) -> String {
|
pub fn width_dec (&mut self) {
|
||||||
format!("Scene {}", self.scenes.len() + 1)
|
if self.width > Self::MIN_WIDTH {
|
||||||
|
self.width -= 1;
|
||||||
}
|
}
|
||||||
pub fn track_add (
|
|
||||||
&mut self, name: Option<&str>, color: Option<ItemColor>
|
|
||||||
) -> Usually<&mut ArrangerTrack> {
|
|
||||||
let name = name.map_or_else(||self.track_default_name(), |x|x.to_string());
|
|
||||||
self.tracks.push(ArrangerTrack {
|
|
||||||
width: name.len() + 2,
|
|
||||||
color: color.unwrap_or_else(||ItemColor::random()),
|
|
||||||
player: MIDIPlayer::new(&self.jack, &self.clock, name.as_str())?,
|
|
||||||
name: Arc::new(name.into()),
|
|
||||||
});
|
|
||||||
let index = self.tracks.len() - 1;
|
|
||||||
Ok(&mut self.tracks[index])
|
|
||||||
}
|
|
||||||
pub fn scene_add (
|
|
||||||
&mut self, name: Option<&str>, color: Option<ItemColor>
|
|
||||||
) -> Usually<&mut ArrangerScene> {
|
|
||||||
let name = name.map_or_else(||self.scene_default_name(), |x|x.to_string());
|
|
||||||
self.scenes.push(ArrangerScene {
|
|
||||||
name: Arc::new(name.into()),
|
|
||||||
clips: vec![None;self.tracks.len()],
|
|
||||||
color: color.unwrap_or_else(||ItemColor::random()),
|
|
||||||
});
|
|
||||||
let index = self.scenes.len() - 1;
|
|
||||||
Ok(&mut self.scenes[index])
|
|
||||||
}
|
|
||||||
pub fn track_del (&mut self, index: usize) {
|
|
||||||
self.tracks.remove(index);
|
|
||||||
for scene in self.scenes.iter_mut() {
|
|
||||||
scene.clips.remove(index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn scene_del (&mut self, index: usize) {
|
|
||||||
self.scenes.remove(index);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -160,19 +216,3 @@ impl ArrangerScene {
|
||||||
//})
|
//})
|
||||||
//}
|
//}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ArrangerTrack {
|
|
||||||
pub fn longest_name (tracks: &[Self]) -> usize {
|
|
||||||
tracks.iter().map(|s|s.name.read().unwrap().len()).fold(0, usize::max)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub const MIN_WIDTH: usize = 3;
|
|
||||||
pub fn width_inc (&mut self) {
|
|
||||||
self.width += 1;
|
|
||||||
}
|
|
||||||
pub fn width_dec (&mut self) {
|
|
||||||
if self.width > Self::MIN_WIDTH {
|
|
||||||
self.width -= 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -38,3 +38,15 @@ submod! {
|
||||||
transport
|
transport
|
||||||
transport_cmd
|
transport_cmd
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait JackModelApi {
|
||||||
|
fn jack (&self) -> &Arc<RwLock<JackClient>>;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait ClockModelApi {
|
||||||
|
fn clock (&self) -> &Arc<Clock>;
|
||||||
|
fn is_stopped (&self) -> bool {
|
||||||
|
*self.clock().playing.read().unwrap() == Some(TransportState::Stopped)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -41,6 +41,7 @@ impl Sampler {
|
||||||
},
|
},
|
||||||
_ => panic!("unexpected in sampler {name}: {edn:?}")
|
_ => panic!("unexpected in sampler {name}: {edn:?}")
|
||||||
});
|
});
|
||||||
|
let midi_in = jack.read().unwrap().client().register_port("in", MidiIn::default())?;
|
||||||
Ok(Sampler {
|
Ok(Sampler {
|
||||||
jack: jack.clone(),
|
jack: jack.clone(),
|
||||||
name: name.into(),
|
name: name.into(),
|
||||||
|
|
@ -48,7 +49,7 @@ impl Sampler {
|
||||||
unmapped: Default::default(),
|
unmapped: Default::default(),
|
||||||
voices: Default::default(),
|
voices: Default::default(),
|
||||||
buffer: Default::default(),
|
buffer: Default::default(),
|
||||||
midi_in: jack.read().unwrap().register_port("in", MidiIn::default())?,
|
midi_in: midi_in,
|
||||||
audio_outs: vec![],
|
audio_outs: vec![],
|
||||||
output_gain: 0.
|
output_gain: 0.
|
||||||
})
|
})
|
||||||
|
|
|
||||||
|
|
@ -1,5 +1,49 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
|
||||||
|
pub trait SequencerModelApi: JackModelApi + ClockModelApi {
|
||||||
|
fn phrases (&self) -> &PhrasePool;
|
||||||
|
fn player (&self) -> &MIDIPlayer;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JackModelApi for SequencerModel {
|
||||||
|
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
||||||
|
self.transport.jack()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClockModelApi for SequencerModel {
|
||||||
|
fn clock (&self) -> &Arc<Clock> {
|
||||||
|
self.transport.clock()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TransportModelApi for SequencerModel {
|
||||||
|
fn transport (&self) -> &jack::Transport {
|
||||||
|
&self.transport.transport()
|
||||||
|
}
|
||||||
|
fn metronome (&self) -> bool {
|
||||||
|
self.transport.metronome()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SequencerModelApi for SequencerModel {
|
||||||
|
fn phrases (&self) -> &PhrasePool {
|
||||||
|
&self.phrases
|
||||||
|
}
|
||||||
|
fn player (&self) -> &MIDIPlayer {
|
||||||
|
&self.player
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct SequencerModel {
|
||||||
|
/// State of the JACK transport.
|
||||||
|
transport: TransportModel,
|
||||||
|
/// State of the phrase pool.
|
||||||
|
phrases: PhrasePool,
|
||||||
|
/// State of the phrase player.
|
||||||
|
player: MIDIPlayer,
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct MIDIPlayer {
|
pub struct MIDIPlayer {
|
||||||
/// Global timebase
|
/// Global timebase
|
||||||
|
|
|
||||||
|
|
@ -1,16 +1,42 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
|
||||||
pub struct Transport {
|
pub trait TransportModelApi: JackModelApi + ClockModelApi {
|
||||||
pub jack: Arc<RwLock<JackClient>>,
|
fn transport (&self) -> &jack::Transport;
|
||||||
/// JACK transport handle.
|
fn metronome (&self) -> bool;
|
||||||
pub transport: jack::Transport,
|
|
||||||
/// Current sample rate, tempo, and PPQ.
|
|
||||||
pub clock: Arc<Clock>,
|
|
||||||
/// Enable metronome?
|
|
||||||
pub metronome: bool,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Debug for Transport {
|
impl JackModelApi for TransportModel {
|
||||||
|
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
||||||
|
&self.jack
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ClockModelApi for TransportModel {
|
||||||
|
fn clock (&self) -> &Arc<Clock> {
|
||||||
|
&self.clock
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TransportModelApi for TransportModel {
|
||||||
|
fn transport (&self) -> &jack::Transport {
|
||||||
|
&self.transport
|
||||||
|
}
|
||||||
|
fn metronome (&self) -> bool {
|
||||||
|
self.metronome
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct TransportModel {
|
||||||
|
jack: Arc<RwLock<JackClient>>,
|
||||||
|
/// Current sample rate, tempo, and PPQ.
|
||||||
|
clock: Arc<Clock>,
|
||||||
|
/// JACK transport handle.
|
||||||
|
transport: jack::Transport,
|
||||||
|
/// Enable metronome?
|
||||||
|
metronome: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Debug for TransportModel {
|
||||||
fn fmt (&self, f: &mut Formatter<'_>) -> std::result::Result<(), Error> {
|
fn fmt (&self, f: &mut Formatter<'_>) -> std::result::Result<(), Error> {
|
||||||
f.debug_struct("transport")
|
f.debug_struct("transport")
|
||||||
.field("jack", &self.jack)
|
.field("jack", &self.jack)
|
||||||
|
|
@ -21,7 +47,7 @@ impl Debug for Transport {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Transport {
|
impl TransportModel {
|
||||||
pub fn toggle_play (&mut self) -> Usually<()> {
|
pub fn toggle_play (&mut self) -> Usually<()> {
|
||||||
let playing = self.clock.playing.read().unwrap().expect("1st sample has not been processed yet");
|
let playing = self.clock.playing.read().unwrap().expect("1st sample has not been processed yet");
|
||||||
let playing = match playing {
|
let playing = match playing {
|
||||||
|
|
|
||||||
|
|
@ -1,36 +1,32 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
use jack::*;
|
use jack::*;
|
||||||
|
|
||||||
/// Trait for things that have a JACK process callback.
|
#[derive(Debug)]
|
||||||
pub trait Audio: Send + Sync {
|
/// Event enum for JACK events.
|
||||||
fn process(&mut self, _: &Client, _: &ProcessScope) -> Control {
|
pub enum JackEvent {
|
||||||
Control::Continue
|
ThreadInit,
|
||||||
}
|
Shutdown(ClientStatus, String),
|
||||||
fn callback(
|
Freewheel(bool),
|
||||||
state: &Arc<RwLock<Self>>, client: &Client, scope: &ProcessScope
|
SampleRate(Frames),
|
||||||
) -> Control where Self: Sized {
|
ClientRegistration(String, bool),
|
||||||
if let Ok(mut state) = state.write() {
|
PortRegistration(PortId, bool),
|
||||||
state.process(client, scope)
|
PortRename(PortId, String, String),
|
||||||
} else {
|
PortsConnected(PortId, PortId, bool),
|
||||||
Control::Quit
|
GraphReorder,
|
||||||
}
|
XRun,
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A UI component that may be associated with a JACK client by the `Jack` factory.
|
/// Wraps [Client] or [DynamicAsyncClient] in place.
|
||||||
pub trait AudioComponent<E: Engine>: Component<E> + Audio {
|
#[derive(Debug)]
|
||||||
/// Perform type erasure for collecting heterogeneous devices.
|
pub enum JackClient {
|
||||||
fn boxed(self) -> Box<dyn AudioComponent<E>>
|
/// Before activation.
|
||||||
where
|
Inactive(Client),
|
||||||
Self: Sized + 'static,
|
/// During activation.
|
||||||
{
|
Activating,
|
||||||
Box::new(self)
|
/// After activation. Must not be dropped for JACK thread to persist.
|
||||||
}
|
Active(DynamicAsyncClient),
|
||||||
}
|
}
|
||||||
|
|
||||||
/// All things that implement the required traits can be treated as `AudioComponent`.
|
|
||||||
impl<E: Engine, W: Component<E> + Audio> AudioComponent<E> for W {}
|
|
||||||
|
|
||||||
/// Trait for things that wrap a JACK client.
|
/// Trait for things that wrap a JACK client.
|
||||||
pub trait AudioEngine {
|
pub trait AudioEngine {
|
||||||
|
|
||||||
|
|
@ -82,64 +78,6 @@ pub trait AudioEngine {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Wraps [Client] or [DynamicAsyncClient] in place.
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub enum JackClient {
|
|
||||||
/// Before activation.
|
|
||||||
Inactive(Client),
|
|
||||||
/// During activation.
|
|
||||||
Activating,
|
|
||||||
/// After activation. Must not be dropped for JACK thread to persist.
|
|
||||||
Active(DynamicAsyncClient),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub type DynamicAsyncClient = AsyncClient<DynamicNotifications, DynamicAudioHandler>;
|
|
||||||
|
|
||||||
pub type DynamicAudioHandler = contrib::ClosureProcessHandler<(), BoxedAudioHandler>;
|
|
||||||
|
|
||||||
pub type BoxedAudioHandler = Box<dyn FnMut(&Client, &ProcessScope) -> Control + Send>;
|
|
||||||
|
|
||||||
impl JackClient {
|
|
||||||
pub fn new (name: &str) -> Usually<Self> {
|
|
||||||
let (client, _) = Client::new(name, ClientOptions::NO_START_SERVER)?;
|
|
||||||
Ok(Self::Inactive(client))
|
|
||||||
}
|
|
||||||
pub fn activate_with <T: Audio + 'static> (
|
|
||||||
self,
|
|
||||||
init: impl FnOnce(&Arc<RwLock<JackClient>>)->Usually<T>
|
|
||||||
)
|
|
||||||
-> Usually<Arc<RwLock<T>>>
|
|
||||||
{
|
|
||||||
let client = Arc::new(RwLock::new(self));
|
|
||||||
let target = Arc::new(RwLock::new(init(&client)?));
|
|
||||||
let event = Box::new(move|_|{/*TODO*/}) as Box<dyn Fn(JackEvent) + Send + Sync>;
|
|
||||||
let events = Notifications(event);
|
|
||||||
let frame = Box::new({
|
|
||||||
let target = target.clone();
|
|
||||||
move|c: &_, s: &_|if let Ok(mut target) = target.write() {
|
|
||||||
target.process(c, s)
|
|
||||||
} else {
|
|
||||||
Control::Quit
|
|
||||||
}
|
|
||||||
});
|
|
||||||
let frames = contrib::ClosureProcessHandler::new(frame as BoxedAudioHandler);
|
|
||||||
let mut buffer = Self::Activating;
|
|
||||||
std::mem::swap(&mut*client.write().unwrap(), &mut buffer);
|
|
||||||
*client.write().unwrap() = Self::Active(Client::from(buffer).activate_async(events, frames)?);
|
|
||||||
Ok(target)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<JackClient> for Client {
|
|
||||||
fn from (jack: JackClient) -> Client {
|
|
||||||
match jack {
|
|
||||||
JackClient::Inactive(client) => client,
|
|
||||||
JackClient::Activating => panic!("jack client still activating"),
|
|
||||||
JackClient::Active(_) => panic!("jack client already activated"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AudioEngine for JackClient {
|
impl AudioEngine for JackClient {
|
||||||
fn client(&self) -> &Client {
|
fn client(&self) -> &Client {
|
||||||
match self {
|
match self {
|
||||||
|
|
@ -166,57 +104,27 @@ impl AudioEngine for JackClient {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
pub type DynamicAsyncClient = AsyncClient<DynamicNotifications, DynamicAudioHandler>;
|
||||||
/// Event enum for JACK events.
|
|
||||||
pub enum JackEvent {
|
|
||||||
ThreadInit,
|
|
||||||
Shutdown(ClientStatus, String),
|
|
||||||
Freewheel(bool),
|
|
||||||
SampleRate(Frames),
|
|
||||||
ClientRegistration(String, bool),
|
|
||||||
PortRegistration(PortId, bool),
|
|
||||||
PortRename(PortId, String, String),
|
|
||||||
PortsConnected(PortId, PortId, bool),
|
|
||||||
GraphReorder,
|
|
||||||
XRun,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Trait for things that may expose JACK ports.
|
pub type DynamicAudioHandler = contrib::ClosureProcessHandler<(), BoxedAudioHandler>;
|
||||||
pub trait Ports {
|
|
||||||
fn audio_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
pub type BoxedAudioHandler = Box<dyn FnMut(&Client, &ProcessScope) -> Control + Send>;
|
||||||
Ok(vec![])
|
|
||||||
}
|
impl JackClient {
|
||||||
fn audio_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
pub fn new (name: &str) -> Usually<Self> {
|
||||||
Ok(vec![])
|
let (client, _) = Client::new(name, ClientOptions::NO_START_SERVER)?;
|
||||||
}
|
Ok(Self::Inactive(client))
|
||||||
fn midi_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
|
||||||
Ok(vec![])
|
|
||||||
}
|
|
||||||
fn midi_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
|
||||||
Ok(vec![])
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn register_ports<T: PortSpec + Copy>(
|
impl From<JackClient> for Client {
|
||||||
client: &Client,
|
fn from (jack: JackClient) -> Client {
|
||||||
names: Vec<String>,
|
match jack {
|
||||||
spec: T,
|
JackClient::Inactive(client) => client,
|
||||||
) -> Usually<BTreeMap<String, Port<T>>> {
|
JackClient::Activating => panic!("jack client still activating"),
|
||||||
names
|
JackClient::Active(_) => panic!("jack client already activated"),
|
||||||
.into_iter()
|
}
|
||||||
.try_fold(BTreeMap::new(), |mut ports, name| {
|
}
|
||||||
let port = client.register_port(&name, spec)?;
|
|
||||||
ports.insert(name, port);
|
|
||||||
Ok(ports)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn query_ports(client: &Client, names: Vec<String>) -> BTreeMap<String, Port<Unowned>> {
|
|
||||||
names.into_iter().fold(BTreeMap::new(), |mut ports, name| {
|
|
||||||
let port = client.port_by_name(&name).unwrap();
|
|
||||||
ports.insert(name, port);
|
|
||||||
ports
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Notification handler used by the [Jack] factory
|
/// Notification handler used by the [Jack] factory
|
||||||
|
|
@ -271,264 +179,3 @@ impl<T: Fn(JackEvent) + Send> NotificationHandler for Notifications<T> {
|
||||||
Control::Continue
|
Control::Continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///// A [AudioComponent] bound to a JACK client and a set of ports.
|
|
||||||
//pub struct JackDevice<E: Engine> {
|
|
||||||
///// The active JACK client of this device.
|
|
||||||
//pub client: DynamicAsyncClient,
|
|
||||||
///// The device state, encapsulated for sharing between threads.
|
|
||||||
//pub state: Arc<RwLock<Box<dyn AudioComponent<E>>>>,
|
|
||||||
///// Unowned copies of the device's JACK ports, for connecting to the device.
|
|
||||||
///// The "real" readable/writable `Port`s are owned by the `state`.
|
|
||||||
//pub ports: UnownedJackPorts,
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl<E: Engine> std::fmt::Debug for JackDevice<E> {
|
|
||||||
//fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
||||||
//f.debug_struct("JackDevice")
|
|
||||||
//.field("ports", &self.ports)
|
|
||||||
//.finish()
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl<E: Engine> Widget for JackDevice<E> {
|
|
||||||
//type Engine = E;
|
|
||||||
//fn layout(&self, to: E::Size) -> Perhaps<E::Size> {
|
|
||||||
//self.state.read().unwrap().layout(to)
|
|
||||||
//}
|
|
||||||
//fn render(&self, to: &mut E::Output) -> Usually<()> {
|
|
||||||
//self.state.read().unwrap().render(to)
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl<E: Engine> Handle<E> for JackDevice<E> {
|
|
||||||
//fn handle(&mut self, from: &E::Input) -> Perhaps<E::Handled> {
|
|
||||||
//self.state.write().unwrap().handle(from)
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl<E: Engine> Ports for JackDevice<E> {
|
|
||||||
//fn audio_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
|
||||||
//Ok(self.ports.audio_ins.values().collect())
|
|
||||||
//}
|
|
||||||
//fn audio_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
|
||||||
//Ok(self.ports.audio_outs.values().collect())
|
|
||||||
//}
|
|
||||||
//fn midi_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
|
||||||
//Ok(self.ports.midi_ins.values().collect())
|
|
||||||
//}
|
|
||||||
//fn midi_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
|
||||||
//Ok(self.ports.midi_outs.values().collect())
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl<E: Engine> JackDevice<E> {
|
|
||||||
///// Returns a locked mutex of the state's contents.
|
|
||||||
//pub fn state(&self) -> LockResult<RwLockReadGuard<Box<dyn AudioComponent<E>>>> {
|
|
||||||
//self.state.read()
|
|
||||||
//}
|
|
||||||
///// Returns a locked mutex of the state's contents.
|
|
||||||
//pub fn state_mut(&self) -> LockResult<RwLockWriteGuard<Box<dyn AudioComponent<E>>>> {
|
|
||||||
//self.state.write()
|
|
||||||
//}
|
|
||||||
//pub fn connect_midi_in(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
|
||||||
//Ok(self
|
|
||||||
//.client
|
|
||||||
//.as_client()
|
|
||||||
//.connect_ports(port, self.midi_ins()?[index])?)
|
|
||||||
//}
|
|
||||||
//pub fn connect_midi_out(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
|
||||||
//Ok(self
|
|
||||||
//.client
|
|
||||||
//.as_client()
|
|
||||||
//.connect_ports(self.midi_outs()?[index], port)?)
|
|
||||||
//}
|
|
||||||
//pub fn connect_audio_in(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
|
||||||
//Ok(self
|
|
||||||
//.client
|
|
||||||
//.as_client()
|
|
||||||
//.connect_ports(port, self.audio_ins()?[index])?)
|
|
||||||
//}
|
|
||||||
//pub fn connect_audio_out(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
|
||||||
//Ok(self
|
|
||||||
//.client
|
|
||||||
//.as_client()
|
|
||||||
//.connect_ports(self.audio_outs()?[index], port)?)
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
///// Collection of JACK ports as [AudioIn]/[AudioOut]/[MidiIn]/[MidiOut].
|
|
||||||
//#[derive(Default, Debug)]
|
|
||||||
//pub struct JackPorts {
|
|
||||||
//pub audio_ins: BTreeMap<String, Port<AudioIn>>,
|
|
||||||
//pub midi_ins: BTreeMap<String, Port<MidiIn>>,
|
|
||||||
//pub audio_outs: BTreeMap<String, Port<AudioOut>>,
|
|
||||||
//pub midi_outs: BTreeMap<String, Port<MidiOut>>,
|
|
||||||
//}
|
|
||||||
|
|
||||||
///// Collection of JACK ports as [Unowned].
|
|
||||||
//#[derive(Default, Debug)]
|
|
||||||
//pub struct UnownedJackPorts {
|
|
||||||
//pub audio_ins: BTreeMap<String, Port<Unowned>>,
|
|
||||||
//pub midi_ins: BTreeMap<String, Port<Unowned>>,
|
|
||||||
//pub audio_outs: BTreeMap<String, Port<Unowned>>,
|
|
||||||
//pub midi_outs: BTreeMap<String, Port<Unowned>>,
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl JackPorts {
|
|
||||||
//pub fn clone_unowned(&self) -> UnownedJackPorts {
|
|
||||||
//let mut unowned = UnownedJackPorts::default();
|
|
||||||
//for (name, port) in self.midi_ins.iter() {
|
|
||||||
//unowned.midi_ins.insert(name.clone(), port.clone_unowned());
|
|
||||||
//}
|
|
||||||
//for (name, port) in self.midi_outs.iter() {
|
|
||||||
//unowned.midi_outs.insert(name.clone(), port.clone_unowned());
|
|
||||||
//}
|
|
||||||
//for (name, port) in self.audio_ins.iter() {
|
|
||||||
//unowned.audio_ins.insert(name.clone(), port.clone_unowned());
|
|
||||||
//}
|
|
||||||
//for (name, port) in self.audio_outs.iter() {
|
|
||||||
//unowned
|
|
||||||
//.audio_outs
|
|
||||||
//.insert(name.clone(), port.clone_unowned());
|
|
||||||
//}
|
|
||||||
//unowned
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
///// Implement the `Ports` trait.
|
|
||||||
//#[macro_export]
|
|
||||||
//macro_rules! ports {
|
|
||||||
//($T:ty $({ $(audio: {
|
|
||||||
//$(ins: |$ai_arg:ident|$ai_impl:expr,)?
|
|
||||||
//$(outs: |$ao_arg:ident|$ao_impl:expr,)?
|
|
||||||
//})? $(midi: {
|
|
||||||
//$(ins: |$mi_arg:ident|$mi_impl:expr,)?
|
|
||||||
//$(outs: |$mo_arg:ident|$mo_impl:expr,)?
|
|
||||||
//})?})?) => {
|
|
||||||
//impl Ports for $T {$(
|
|
||||||
//$(
|
|
||||||
//$(fn audio_ins <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
|
||||||
//let cb = |$ai_arg:&'a Self|$ai_impl;
|
|
||||||
//cb(self)
|
|
||||||
//})?
|
|
||||||
//)?
|
|
||||||
//$(
|
|
||||||
//$(fn audio_outs <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
|
||||||
//let cb = (|$ao_arg:&'a Self|$ao_impl);
|
|
||||||
//cb(self)
|
|
||||||
//})?
|
|
||||||
//)?
|
|
||||||
//)? $(
|
|
||||||
//$(
|
|
||||||
//$(fn midi_ins <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
|
||||||
//let cb = (|$mi_arg:&'a Self|$mi_impl);
|
|
||||||
//cb(self)
|
|
||||||
//})?
|
|
||||||
//)?
|
|
||||||
//$(
|
|
||||||
//$(fn midi_outs <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
|
||||||
//let cb = (|$mo_arg:&'a Self|$mo_impl);
|
|
||||||
//cb(self)
|
|
||||||
//})?
|
|
||||||
//)?
|
|
||||||
//)?}
|
|
||||||
//};
|
|
||||||
//}
|
|
||||||
|
|
||||||
///// `JackDevice` factory. Creates JACK `Client`s, performs port registration
|
|
||||||
///// and activation, and encapsulates a `AudioComponent` into a `JackDevice`.
|
|
||||||
//pub struct Jack {
|
|
||||||
//pub client: Client,
|
|
||||||
//pub midi_ins: Vec<String>,
|
|
||||||
//pub audio_ins: Vec<String>,
|
|
||||||
//pub midi_outs: Vec<String>,
|
|
||||||
//pub audio_outs: Vec<String>,
|
|
||||||
//}
|
|
||||||
|
|
||||||
//impl Jack {
|
|
||||||
//pub fn new(name: &str) -> Usually<Self> {
|
|
||||||
//Ok(Self {
|
|
||||||
//midi_ins: vec![],
|
|
||||||
//audio_ins: vec![],
|
|
||||||
//midi_outs: vec![],
|
|
||||||
//audio_outs: vec![],
|
|
||||||
//client: Client::new(name, ClientOptions::NO_START_SERVER)?.0,
|
|
||||||
//})
|
|
||||||
//}
|
|
||||||
//pub fn run<'a: 'static, D, E>(
|
|
||||||
//self,
|
|
||||||
//state: impl FnOnce(JackPorts) -> Box<D>,
|
|
||||||
//) -> Usually<JackDevice<E>>
|
|
||||||
//where
|
|
||||||
//D: AudioComponent<E> + Sized + 'static,
|
|
||||||
//E: Engine + 'static,
|
|
||||||
//{
|
|
||||||
//let owned_ports = JackPorts {
|
|
||||||
//audio_ins: register_ports(&self.client, self.audio_ins, AudioIn::default())?,
|
|
||||||
//audio_outs: register_ports(&self.client, self.audio_outs, AudioOut::default())?,
|
|
||||||
//midi_ins: register_ports(&self.client, self.midi_ins, MidiIn::default())?,
|
|
||||||
//midi_outs: register_ports(&self.client, self.midi_outs, MidiOut::default())?,
|
|
||||||
//};
|
|
||||||
//let midi_outs = owned_ports
|
|
||||||
//.midi_outs
|
|
||||||
//.values()
|
|
||||||
//.map(|p| Ok(p.name()?))
|
|
||||||
//.collect::<Usually<Vec<_>>>()?;
|
|
||||||
//let midi_ins = owned_ports
|
|
||||||
//.midi_ins
|
|
||||||
//.values()
|
|
||||||
//.map(|p| Ok(p.name()?))
|
|
||||||
//.collect::<Usually<Vec<_>>>()?;
|
|
||||||
//let audio_outs = owned_ports
|
|
||||||
//.audio_outs
|
|
||||||
//.values()
|
|
||||||
//.map(|p| Ok(p.name()?))
|
|
||||||
//.collect::<Usually<Vec<_>>>()?;
|
|
||||||
//let audio_ins = owned_ports
|
|
||||||
//.audio_ins
|
|
||||||
//.values()
|
|
||||||
//.map(|p| Ok(p.name()?))
|
|
||||||
//.collect::<Usually<Vec<_>>>()?;
|
|
||||||
//let state = Arc::new(RwLock::new(state(owned_ports) as Box<dyn AudioComponent<E>>));
|
|
||||||
//let client = self.client.activate_async(
|
|
||||||
//Notifications(Box::new({
|
|
||||||
//let _state = state.clone();
|
|
||||||
//move |_event| {
|
|
||||||
//// FIXME: this deadlocks
|
|
||||||
////state.lock().unwrap().handle(&event).unwrap();
|
|
||||||
//}
|
|
||||||
//}) as Box<dyn Fn(JackEvent) + Send + Sync>),
|
|
||||||
//contrib::ClosureProcessHandler::new(Box::new({
|
|
||||||
//let state = state.clone();
|
|
||||||
//move |c: &Client, s: &ProcessScope| state.write().unwrap().process(c, s)
|
|
||||||
//}) as BoxedAudioHandler),
|
|
||||||
//)?;
|
|
||||||
//Ok(JackDevice {
|
|
||||||
//ports: UnownedJackPorts {
|
|
||||||
//audio_ins: query_ports(&client.as_client(), audio_ins),
|
|
||||||
//audio_outs: query_ports(&client.as_client(), audio_outs),
|
|
||||||
//midi_ins: query_ports(&client.as_client(), midi_ins),
|
|
||||||
//midi_outs: query_ports(&client.as_client(), midi_outs),
|
|
||||||
//},
|
|
||||||
//client,
|
|
||||||
//state,
|
|
||||||
//})
|
|
||||||
//}
|
|
||||||
//pub fn audio_in(mut self, name: &str) -> Self {
|
|
||||||
//self.audio_ins.push(name.to_string());
|
|
||||||
//self
|
|
||||||
//}
|
|
||||||
//pub fn audio_out(mut self, name: &str) -> Self {
|
|
||||||
//self.audio_outs.push(name.to_string());
|
|
||||||
//self
|
|
||||||
//}
|
|
||||||
//pub fn midi_in(mut self, name: &str) -> Self {
|
|
||||||
//self.midi_ins.push(name.to_string());
|
|
||||||
//self
|
|
||||||
//}
|
|
||||||
//pub fn midi_out(mut self, name: &str) -> Self {
|
|
||||||
//self.midi_outs.push(name.to_string());
|
|
||||||
//self
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
|
||||||
|
|
@ -10,3 +10,367 @@ submod! {
|
||||||
snd_sequencer
|
snd_sequencer
|
||||||
snd_transport
|
snd_transport
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub trait JackActivate: Sized {
|
||||||
|
fn activate_with <T: Audio + 'static> (
|
||||||
|
self,
|
||||||
|
init: impl FnOnce(&Arc<RwLock<JackClient>>)->Usually<T>
|
||||||
|
)
|
||||||
|
-> Usually<Arc<RwLock<T>>>;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl JackActivate for JackClient {
|
||||||
|
fn activate_with <T: Audio + 'static> (
|
||||||
|
self,
|
||||||
|
init: impl FnOnce(&Arc<RwLock<JackClient>>)->Usually<T>
|
||||||
|
)
|
||||||
|
-> Usually<Arc<RwLock<T>>>
|
||||||
|
{
|
||||||
|
let client = Arc::new(RwLock::new(self));
|
||||||
|
let target = Arc::new(RwLock::new(init(&client)?));
|
||||||
|
let event = Box::new(move|_|{/*TODO*/}) as Box<dyn Fn(JackEvent) + Send + Sync>;
|
||||||
|
let events = Notifications(event);
|
||||||
|
let frame = Box::new({
|
||||||
|
let target = target.clone();
|
||||||
|
move|c: &_, s: &_|if let Ok(mut target) = target.write() {
|
||||||
|
target.process(c, s)
|
||||||
|
} else {
|
||||||
|
Control::Quit
|
||||||
|
}
|
||||||
|
});
|
||||||
|
let frames = contrib::ClosureProcessHandler::new(frame as BoxedAudioHandler);
|
||||||
|
let mut buffer = Self::Activating;
|
||||||
|
std::mem::swap(&mut*client.write().unwrap(), &mut buffer);
|
||||||
|
*client.write().unwrap() = Self::Active(Client::from(buffer).activate_async(events, frames)?);
|
||||||
|
Ok(target)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Trait for things that have a JACK process callback.
|
||||||
|
pub trait Audio: Send + Sync {
|
||||||
|
fn process(&mut self, _: &Client, _: &ProcessScope) -> Control {
|
||||||
|
Control::Continue
|
||||||
|
}
|
||||||
|
fn callback(
|
||||||
|
state: &Arc<RwLock<Self>>, client: &Client, scope: &ProcessScope
|
||||||
|
) -> Control where Self: Sized {
|
||||||
|
if let Ok(mut state) = state.write() {
|
||||||
|
state.process(client, scope)
|
||||||
|
} else {
|
||||||
|
Control::Quit
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A UI component that may be associated with a JACK client by the `Jack` factory.
|
||||||
|
pub trait AudioComponent<E: Engine>: Component<E> + Audio {
|
||||||
|
/// Perform type erasure for collecting heterogeneous devices.
|
||||||
|
fn boxed(self) -> Box<dyn AudioComponent<E>>
|
||||||
|
where
|
||||||
|
Self: Sized + 'static,
|
||||||
|
{
|
||||||
|
Box::new(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// All things that implement the required traits can be treated as `AudioComponent`.
|
||||||
|
impl<E: Engine, W: Component<E> + Audio> AudioComponent<E> for W {}
|
||||||
|
|
||||||
|
/// Trait for things that may expose JACK ports.
|
||||||
|
pub trait Ports {
|
||||||
|
fn audio_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
Ok(vec![])
|
||||||
|
}
|
||||||
|
fn audio_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
Ok(vec![])
|
||||||
|
}
|
||||||
|
fn midi_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
Ok(vec![])
|
||||||
|
}
|
||||||
|
fn midi_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
Ok(vec![])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn register_ports<T: PortSpec + Copy>(
|
||||||
|
client: &Client,
|
||||||
|
names: Vec<String>,
|
||||||
|
spec: T,
|
||||||
|
) -> Usually<BTreeMap<String, Port<T>>> {
|
||||||
|
names
|
||||||
|
.into_iter()
|
||||||
|
.try_fold(BTreeMap::new(), |mut ports, name| {
|
||||||
|
let port = client.register_port(&name, spec)?;
|
||||||
|
ports.insert(name, port);
|
||||||
|
Ok(ports)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn query_ports(client: &Client, names: Vec<String>) -> BTreeMap<String, Port<Unowned>> {
|
||||||
|
names.into_iter().fold(BTreeMap::new(), |mut ports, name| {
|
||||||
|
let port = client.port_by_name(&name).unwrap();
|
||||||
|
ports.insert(name, port);
|
||||||
|
ports
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
///// A [AudioComponent] bound to a JACK client and a set of ports.
|
||||||
|
//pub struct JackDevice<E: Engine> {
|
||||||
|
///// The active JACK client of this device.
|
||||||
|
//pub client: DynamicAsyncClient,
|
||||||
|
///// The device state, encapsulated for sharing between threads.
|
||||||
|
//pub state: Arc<RwLock<Box<dyn AudioComponent<E>>>>,
|
||||||
|
///// Unowned copies of the device's JACK ports, for connecting to the device.
|
||||||
|
///// The "real" readable/writable `Port`s are owned by the `state`.
|
||||||
|
//pub ports: UnownedJackPorts,
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl<E: Engine> std::fmt::Debug for JackDevice<E> {
|
||||||
|
//fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
|
//f.debug_struct("JackDevice")
|
||||||
|
//.field("ports", &self.ports)
|
||||||
|
//.finish()
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl<E: Engine> Widget for JackDevice<E> {
|
||||||
|
//type Engine = E;
|
||||||
|
//fn layout(&self, to: E::Size) -> Perhaps<E::Size> {
|
||||||
|
//self.state.read().unwrap().layout(to)
|
||||||
|
//}
|
||||||
|
//fn render(&self, to: &mut E::Output) -> Usually<()> {
|
||||||
|
//self.state.read().unwrap().render(to)
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl<E: Engine> Handle<E> for JackDevice<E> {
|
||||||
|
//fn handle(&mut self, from: &E::Input) -> Perhaps<E::Handled> {
|
||||||
|
//self.state.write().unwrap().handle(from)
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl<E: Engine> Ports for JackDevice<E> {
|
||||||
|
//fn audio_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
//Ok(self.ports.audio_ins.values().collect())
|
||||||
|
//}
|
||||||
|
//fn audio_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
//Ok(self.ports.audio_outs.values().collect())
|
||||||
|
//}
|
||||||
|
//fn midi_ins(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
//Ok(self.ports.midi_ins.values().collect())
|
||||||
|
//}
|
||||||
|
//fn midi_outs(&self) -> Usually<Vec<&Port<Unowned>>> {
|
||||||
|
//Ok(self.ports.midi_outs.values().collect())
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl<E: Engine> JackDevice<E> {
|
||||||
|
///// Returns a locked mutex of the state's contents.
|
||||||
|
//pub fn state(&self) -> LockResult<RwLockReadGuard<Box<dyn AudioComponent<E>>>> {
|
||||||
|
//self.state.read()
|
||||||
|
//}
|
||||||
|
///// Returns a locked mutex of the state's contents.
|
||||||
|
//pub fn state_mut(&self) -> LockResult<RwLockWriteGuard<Box<dyn AudioComponent<E>>>> {
|
||||||
|
//self.state.write()
|
||||||
|
//}
|
||||||
|
//pub fn connect_midi_in(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
||||||
|
//Ok(self
|
||||||
|
//.client
|
||||||
|
//.as_client()
|
||||||
|
//.connect_ports(port, self.midi_ins()?[index])?)
|
||||||
|
//}
|
||||||
|
//pub fn connect_midi_out(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
||||||
|
//Ok(self
|
||||||
|
//.client
|
||||||
|
//.as_client()
|
||||||
|
//.connect_ports(self.midi_outs()?[index], port)?)
|
||||||
|
//}
|
||||||
|
//pub fn connect_audio_in(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
||||||
|
//Ok(self
|
||||||
|
//.client
|
||||||
|
//.as_client()
|
||||||
|
//.connect_ports(port, self.audio_ins()?[index])?)
|
||||||
|
//}
|
||||||
|
//pub fn connect_audio_out(&self, index: usize, port: &Port<Unowned>) -> Usually<()> {
|
||||||
|
//Ok(self
|
||||||
|
//.client
|
||||||
|
//.as_client()
|
||||||
|
//.connect_ports(self.audio_outs()?[index], port)?)
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
///// Collection of JACK ports as [AudioIn]/[AudioOut]/[MidiIn]/[MidiOut].
|
||||||
|
//#[derive(Default, Debug)]
|
||||||
|
//pub struct JackPorts {
|
||||||
|
//pub audio_ins: BTreeMap<String, Port<AudioIn>>,
|
||||||
|
//pub midi_ins: BTreeMap<String, Port<MidiIn>>,
|
||||||
|
//pub audio_outs: BTreeMap<String, Port<AudioOut>>,
|
||||||
|
//pub midi_outs: BTreeMap<String, Port<MidiOut>>,
|
||||||
|
//}
|
||||||
|
|
||||||
|
///// Collection of JACK ports as [Unowned].
|
||||||
|
//#[derive(Default, Debug)]
|
||||||
|
//pub struct UnownedJackPorts {
|
||||||
|
//pub audio_ins: BTreeMap<String, Port<Unowned>>,
|
||||||
|
//pub midi_ins: BTreeMap<String, Port<Unowned>>,
|
||||||
|
//pub audio_outs: BTreeMap<String, Port<Unowned>>,
|
||||||
|
//pub midi_outs: BTreeMap<String, Port<Unowned>>,
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl JackPorts {
|
||||||
|
//pub fn clone_unowned(&self) -> UnownedJackPorts {
|
||||||
|
//let mut unowned = UnownedJackPorts::default();
|
||||||
|
//for (name, port) in self.midi_ins.iter() {
|
||||||
|
//unowned.midi_ins.insert(name.clone(), port.clone_unowned());
|
||||||
|
//}
|
||||||
|
//for (name, port) in self.midi_outs.iter() {
|
||||||
|
//unowned.midi_outs.insert(name.clone(), port.clone_unowned());
|
||||||
|
//}
|
||||||
|
//for (name, port) in self.audio_ins.iter() {
|
||||||
|
//unowned.audio_ins.insert(name.clone(), port.clone_unowned());
|
||||||
|
//}
|
||||||
|
//for (name, port) in self.audio_outs.iter() {
|
||||||
|
//unowned
|
||||||
|
//.audio_outs
|
||||||
|
//.insert(name.clone(), port.clone_unowned());
|
||||||
|
//}
|
||||||
|
//unowned
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
///// Implement the `Ports` trait.
|
||||||
|
//#[macro_export]
|
||||||
|
//macro_rules! ports {
|
||||||
|
//($T:ty $({ $(audio: {
|
||||||
|
//$(ins: |$ai_arg:ident|$ai_impl:expr,)?
|
||||||
|
//$(outs: |$ao_arg:ident|$ao_impl:expr,)?
|
||||||
|
//})? $(midi: {
|
||||||
|
//$(ins: |$mi_arg:ident|$mi_impl:expr,)?
|
||||||
|
//$(outs: |$mo_arg:ident|$mo_impl:expr,)?
|
||||||
|
//})?})?) => {
|
||||||
|
//impl Ports for $T {$(
|
||||||
|
//$(
|
||||||
|
//$(fn audio_ins <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
||||||
|
//let cb = |$ai_arg:&'a Self|$ai_impl;
|
||||||
|
//cb(self)
|
||||||
|
//})?
|
||||||
|
//)?
|
||||||
|
//$(
|
||||||
|
//$(fn audio_outs <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
||||||
|
//let cb = (|$ao_arg:&'a Self|$ao_impl);
|
||||||
|
//cb(self)
|
||||||
|
//})?
|
||||||
|
//)?
|
||||||
|
//)? $(
|
||||||
|
//$(
|
||||||
|
//$(fn midi_ins <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
||||||
|
//let cb = (|$mi_arg:&'a Self|$mi_impl);
|
||||||
|
//cb(self)
|
||||||
|
//})?
|
||||||
|
//)?
|
||||||
|
//$(
|
||||||
|
//$(fn midi_outs <'a> (&'a self) -> Usually<Vec<&'a Port<Unowned>>> {
|
||||||
|
//let cb = (|$mo_arg:&'a Self|$mo_impl);
|
||||||
|
//cb(self)
|
||||||
|
//})?
|
||||||
|
//)?
|
||||||
|
//)?}
|
||||||
|
//};
|
||||||
|
//}
|
||||||
|
|
||||||
|
///// `JackDevice` factory. Creates JACK `Client`s, performs port registration
|
||||||
|
///// and activation, and encapsulates a `AudioComponent` into a `JackDevice`.
|
||||||
|
//pub struct Jack {
|
||||||
|
//pub client: Client,
|
||||||
|
//pub midi_ins: Vec<String>,
|
||||||
|
//pub audio_ins: Vec<String>,
|
||||||
|
//pub midi_outs: Vec<String>,
|
||||||
|
//pub audio_outs: Vec<String>,
|
||||||
|
//}
|
||||||
|
|
||||||
|
//impl Jack {
|
||||||
|
//pub fn new(name: &str) -> Usually<Self> {
|
||||||
|
//Ok(Self {
|
||||||
|
//midi_ins: vec![],
|
||||||
|
//audio_ins: vec![],
|
||||||
|
//midi_outs: vec![],
|
||||||
|
//audio_outs: vec![],
|
||||||
|
//client: Client::new(name, ClientOptions::NO_START_SERVER)?.0,
|
||||||
|
//})
|
||||||
|
//}
|
||||||
|
//pub fn run<'a: 'static, D, E>(
|
||||||
|
//self,
|
||||||
|
//state: impl FnOnce(JackPorts) -> Box<D>,
|
||||||
|
//) -> Usually<JackDevice<E>>
|
||||||
|
//where
|
||||||
|
//D: AudioComponent<E> + Sized + 'static,
|
||||||
|
//E: Engine + 'static,
|
||||||
|
//{
|
||||||
|
//let owned_ports = JackPorts {
|
||||||
|
//audio_ins: register_ports(&self.client, self.audio_ins, AudioIn::default())?,
|
||||||
|
//audio_outs: register_ports(&self.client, self.audio_outs, AudioOut::default())?,
|
||||||
|
//midi_ins: register_ports(&self.client, self.midi_ins, MidiIn::default())?,
|
||||||
|
//midi_outs: register_ports(&self.client, self.midi_outs, MidiOut::default())?,
|
||||||
|
//};
|
||||||
|
//let midi_outs = owned_ports
|
||||||
|
//.midi_outs
|
||||||
|
//.values()
|
||||||
|
//.map(|p| Ok(p.name()?))
|
||||||
|
//.collect::<Usually<Vec<_>>>()?;
|
||||||
|
//let midi_ins = owned_ports
|
||||||
|
//.midi_ins
|
||||||
|
//.values()
|
||||||
|
//.map(|p| Ok(p.name()?))
|
||||||
|
//.collect::<Usually<Vec<_>>>()?;
|
||||||
|
//let audio_outs = owned_ports
|
||||||
|
//.audio_outs
|
||||||
|
//.values()
|
||||||
|
//.map(|p| Ok(p.name()?))
|
||||||
|
//.collect::<Usually<Vec<_>>>()?;
|
||||||
|
//let audio_ins = owned_ports
|
||||||
|
//.audio_ins
|
||||||
|
//.values()
|
||||||
|
//.map(|p| Ok(p.name()?))
|
||||||
|
//.collect::<Usually<Vec<_>>>()?;
|
||||||
|
//let state = Arc::new(RwLock::new(state(owned_ports) as Box<dyn AudioComponent<E>>));
|
||||||
|
//let client = self.client.activate_async(
|
||||||
|
//Notifications(Box::new({
|
||||||
|
//let _state = state.clone();
|
||||||
|
//move |_event| {
|
||||||
|
//// FIXME: this deadlocks
|
||||||
|
////state.lock().unwrap().handle(&event).unwrap();
|
||||||
|
//}
|
||||||
|
//}) as Box<dyn Fn(JackEvent) + Send + Sync>),
|
||||||
|
//contrib::ClosureProcessHandler::new(Box::new({
|
||||||
|
//let state = state.clone();
|
||||||
|
//move |c: &Client, s: &ProcessScope| state.write().unwrap().process(c, s)
|
||||||
|
//}) as BoxedAudioHandler),
|
||||||
|
//)?;
|
||||||
|
//Ok(JackDevice {
|
||||||
|
//ports: UnownedJackPorts {
|
||||||
|
//audio_ins: query_ports(&client.as_client(), audio_ins),
|
||||||
|
//audio_outs: query_ports(&client.as_client(), audio_outs),
|
||||||
|
//midi_ins: query_ports(&client.as_client(), midi_ins),
|
||||||
|
//midi_outs: query_ports(&client.as_client(), midi_outs),
|
||||||
|
//},
|
||||||
|
//client,
|
||||||
|
//state,
|
||||||
|
//})
|
||||||
|
//}
|
||||||
|
//pub fn audio_in(mut self, name: &str) -> Self {
|
||||||
|
//self.audio_ins.push(name.to_string());
|
||||||
|
//self
|
||||||
|
//}
|
||||||
|
//pub fn audio_out(mut self, name: &str) -> Self {
|
||||||
|
//self.audio_outs.push(name.to_string());
|
||||||
|
//self
|
||||||
|
//}
|
||||||
|
//pub fn midi_in(mut self, name: &str) -> Self {
|
||||||
|
//self.midi_ins.push(name.to_string());
|
||||||
|
//self
|
||||||
|
//}
|
||||||
|
//pub fn midi_out(mut self, name: &str) -> Self {
|
||||||
|
//self.midi_outs.push(name.to_string());
|
||||||
|
//self
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
|
||||||
|
|
@ -1,18 +1,8 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
|
||||||
pub struct ArrangerAudio(pub Arc<RwLock<ArrangerModel>>);
|
impl<T: ArrangerModelApi + Send + Sync> Audio for T {
|
||||||
|
|
||||||
impl Audio for ArrangerAudio {
|
|
||||||
#[inline] fn process (&mut self, client: &Client, scope: &ProcessScope) -> Control {
|
#[inline] fn process (&mut self, client: &Client, scope: &ProcessScope) -> Control {
|
||||||
ArrangerRefAudio(&mut*self.0.write().unwrap()).process(client, scope)
|
for track in self.tracks_mut().iter_mut() {
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct ArrangerRefAudio<'a>(pub &'a mut ArrangerModel);
|
|
||||||
|
|
||||||
impl<'a> Audio for ArrangerRefAudio<'a> {
|
|
||||||
#[inline] fn process (&mut self, client: &Client, scope: &ProcessScope) -> Control {
|
|
||||||
for track in self.0.tracks.iter_mut() {
|
|
||||||
if MIDIPlayerAudio::from(&mut track.player).process(client, scope) == Control::Quit {
|
if MIDIPlayerAudio::from(&mut track.player).process(client, scope) == Control::Quit {
|
||||||
return Control::Quit
|
return Control::Quit
|
||||||
}
|
}
|
||||||
|
|
@ -20,3 +10,24 @@ impl<'a> Audio for ArrangerRefAudio<'a> {
|
||||||
Control::Continue
|
Control::Continue
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//pub struct ArrangerAudio(pub Arc<RwLock<ArrangerModel>>);
|
||||||
|
|
||||||
|
//impl Audio for ArrangerAudio {
|
||||||
|
//#[inline] fn process (&mut self, client: &Client, scope: &ProcessScope) -> Control {
|
||||||
|
//ArrangerRefAudio(&mut*self.0.write().unwrap()).process(client, scope)
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
||||||
|
//pub struct ArrangerRefAudio<'a>(pub &'a mut ArrangerModel);
|
||||||
|
|
||||||
|
//impl<'a> Audio for ArrangerRefAudio<'a> {
|
||||||
|
//#[inline] fn process (&mut self, client: &Client, scope: &ProcessScope) -> Control {
|
||||||
|
//for track in self.0.tracks.iter_mut() {
|
||||||
|
//if MIDIPlayerAudio::from(&mut track.player).process(client, scope) == Control::Quit {
|
||||||
|
//return Control::Quit
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
//Control::Continue
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
|
||||||
|
|
@ -43,15 +43,6 @@ impl TryFrom<&Arc<RwLock<JackClient>>> for SequencerApp {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct SequencerModel {
|
|
||||||
/// State of the JACK transport.
|
|
||||||
pub transport: Arc<RwLock<tek_api::Transport>>,
|
|
||||||
/// State of the phrase pool.
|
|
||||||
pub phrases: Arc<RwLock<PhrasePool>>,
|
|
||||||
/// State of the phrase player.
|
|
||||||
pub player: Arc<RwLock<MIDIPlayer>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<E: Engine> From<&Arc<RwLock<SequencerModel>>> for SequencerView<E> {
|
impl<E: Engine> From<&Arc<RwLock<SequencerModel>>> for SequencerView<E> {
|
||||||
fn from (model: &Arc<RwLock<SequencerModel>>) -> Self {
|
fn from (model: &Arc<RwLock<SequencerModel>>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue