mirror of
https://codeberg.org/unspeaker/tek.git
synced 2025-12-07 12:16:42 +01:00
wip: refactor pt.35 (16e) even more traits, even fewer structs
This commit is contained in:
parent
beca1a6ade
commit
7af5bbd02b
12 changed files with 121 additions and 191 deletions
|
|
@ -19,11 +19,12 @@ impl<T: ClockApi> Command<T> for ClockCommand {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait ClockApi {
|
pub trait ClockApi {
|
||||||
fn quant (&self) -> &Quantize;
|
/// Current moment in time
|
||||||
|
|
||||||
fn sync (&self) -> &LaunchSync;
|
|
||||||
|
|
||||||
fn current (&self) -> &Instant;
|
fn current (&self) -> &Instant;
|
||||||
|
/// Note quantization factor
|
||||||
|
fn quant (&self) -> &Quantize;
|
||||||
|
/// Launch quantization factor
|
||||||
|
fn sync (&self) -> &LaunchSync;
|
||||||
|
|
||||||
fn timebase (&self) -> &Timebase {
|
fn timebase (&self) -> &Timebase {
|
||||||
&self.current().timebase
|
&self.current().timebase
|
||||||
|
|
@ -38,19 +39,3 @@ pub trait ClockApi {
|
||||||
&self.timebase().ppq
|
&self.timebase().ppq
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait HasClock {
|
|
||||||
fn clock (&self) -> &impl ClockApi;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<T: HasClock> ClockApi for T {
|
|
||||||
fn quant (&self) -> &Quantize {
|
|
||||||
self.clock().quant()
|
|
||||||
}
|
|
||||||
fn sync (&self) -> &LaunchSync {
|
|
||||||
self.clock().sync()
|
|
||||||
}
|
|
||||||
fn current (&self) -> &Instant {
|
|
||||||
self.clock().current()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -35,7 +35,7 @@ impl<T: PlayheadApi> Command<T> for PlayheadCommand {
|
||||||
|
|
||||||
pub trait PlayheadApi: ClockApi {
|
pub trait PlayheadApi: ClockApi {
|
||||||
|
|
||||||
fn transport (&self) -> &RwLock<Option<TransportState>>;
|
fn transport (&self) -> &jack::Transport;
|
||||||
|
|
||||||
fn playing (&self) -> &RwLock<Option<TransportState>>;
|
fn playing (&self) -> &RwLock<Option<TransportState>>;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -3,26 +3,13 @@ use crate::*;
|
||||||
pub trait HasScenes<S: ArrangerSceneApi> {
|
pub trait HasScenes<S: ArrangerSceneApi> {
|
||||||
fn scenes (&self) -> &Vec<S>;
|
fn scenes (&self) -> &Vec<S>;
|
||||||
fn scenes_mut (&mut self) -> &mut Vec<S>;
|
fn scenes_mut (&mut self) -> &mut Vec<S>;
|
||||||
|
fn scene_add (&mut self, name: Option<&str>, color: Option<ItemColor>) -> Usually<&mut S>;
|
||||||
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 S>
|
|
||||||
{
|
|
||||||
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) {
|
fn scene_del (&mut self, index: usize) {
|
||||||
self.scenes_mut().remove(index);
|
self.scenes_mut().remove(index);
|
||||||
}
|
}
|
||||||
|
fn scene_default_name (&self) -> String {
|
||||||
|
format!("Scene {}", self.scenes().len() + 1)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
|
|
@ -47,9 +34,9 @@ pub enum ArrangerSceneCommand {
|
||||||
//}
|
//}
|
||||||
|
|
||||||
pub trait ArrangerSceneApi: Sized {
|
pub trait ArrangerSceneApi: Sized {
|
||||||
fn name () -> Arc<RwLock<String>>;
|
fn name (&self) -> Arc<RwLock<String>>;
|
||||||
fn clips () -> Vec<Option<Arc<RwLock<Phrase>>>>;
|
fn clips (&self) -> &Vec<Option<Arc<RwLock<Phrase>>>>;
|
||||||
fn color () -> ItemColor;
|
fn color (&self) -> ItemColor;
|
||||||
|
|
||||||
fn ppqs (scenes: &[Self], factor: usize) -> Vec<(usize, usize)> {
|
fn ppqs (scenes: &[Self], factor: usize) -> Vec<(usize, usize)> {
|
||||||
let mut total = 0;
|
let mut total = 0;
|
||||||
|
|
@ -80,7 +67,7 @@ pub trait ArrangerSceneApi: Sized {
|
||||||
/// Returns true if all phrases in the scene are
|
/// Returns true if all phrases in the scene are
|
||||||
/// currently playing on the given collection of tracks.
|
/// currently playing on the given collection of tracks.
|
||||||
fn is_playing (&self, tracks: &[ArrangerTrack]) -> bool {
|
fn is_playing (&self, tracks: &[ArrangerTrack]) -> bool {
|
||||||
self.clips().iter().any(|clip|clip.is_some()) && self.clips.iter().enumerate()
|
self.clips().iter().any(|clip|clip.is_some()) && self.clips().iter().enumerate()
|
||||||
.all(|(track_index, clip)|match clip {
|
.all(|(track_index, clip)|match clip {
|
||||||
Some(clip) => tracks
|
Some(clip) => tracks
|
||||||
.get(track_index)
|
.get(track_index)
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
|
||||||
pub trait HasPlayer: HasJack + HasClock {
|
pub trait HasPlayer: HasJack {
|
||||||
fn player (&self) -> &MIDIPlayer;
|
fn player (&self) -> &MIDIPlayer;
|
||||||
fn player_mut (&mut self) -> &mut MIDIPlayer;
|
fn player_mut (&mut self) -> &mut MIDIPlayer;
|
||||||
}
|
}
|
||||||
|
|
@ -24,7 +24,7 @@ pub trait HasMidiBuffer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait HasPhrase: PlayheadApi + HasClock + HasMidiBuffer {
|
pub trait HasPhrase: ClockApi + PlayheadApi + HasMidiBuffer {
|
||||||
fn phrase (&self) -> &Option<(Instant, Arc<RwLock<Phrase>>)>;
|
fn phrase (&self) -> &Option<(Instant, Arc<RwLock<Phrase>>)>;
|
||||||
fn next_phrase (&self) -> &Option<(Instant, Arc<RwLock<Phrase>>)>;
|
fn next_phrase (&self) -> &Option<(Instant, Arc<RwLock<Phrase>>)>;
|
||||||
fn next_phrase_mut (&mut self) -> &mut Option<(Instant, Arc<RwLock<Phrase>>)>;
|
fn next_phrase_mut (&mut self) -> &mut Option<(Instant, Arc<RwLock<Phrase>>)>;
|
||||||
|
|
@ -34,13 +34,13 @@ pub trait HasPhrase: PlayheadApi + HasClock + HasMidiBuffer {
|
||||||
//let samples = scope.n_frames() as usize;
|
//let samples = scope.n_frames() as usize;
|
||||||
if let Some((start_at, phrase)) = &self.next_phrase() {
|
if let Some((start_at, phrase)) = &self.next_phrase() {
|
||||||
let start = start_at.sample.get() as usize;
|
let start = start_at.sample.get() as usize;
|
||||||
let sample = self.clock().started.read().unwrap().unwrap().0;
|
let sample = self.started().read().unwrap().unwrap().0;
|
||||||
// If it's time to switch to the next phrase:
|
// If it's time to switch to the next phrase:
|
||||||
if start <= sample0.saturating_sub(sample) {
|
if start <= sample0.saturating_sub(sample) {
|
||||||
// Samples elapsed since phrase was supposed to start
|
// Samples elapsed since phrase was supposed to start
|
||||||
let skipped = sample0 - start;
|
let skipped = sample0 - start;
|
||||||
// Switch over to enqueued phrase
|
// Switch over to enqueued phrase
|
||||||
let started = Instant::from_sample(&self.clock().timebase(), start as f64);
|
let started = Instant::from_sample(&self.timebase(), start as f64);
|
||||||
self.phrase = Some((started, phrase.clone()));
|
self.phrase = Some((started, phrase.clone()));
|
||||||
// Unset enqueuement (TODO: where to implement looping?)
|
// Unset enqueuement (TODO: where to implement looping?)
|
||||||
*self.next_phrase_mut() = None
|
*self.next_phrase_mut() = None
|
||||||
|
|
@ -54,9 +54,9 @@ pub trait HasPhrase: PlayheadApi + HasClock + HasMidiBuffer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn enqueue_next (&mut self, phrase: Option<&Arc<RwLock<Phrase>>>) {
|
fn enqueue_next (&mut self, phrase: Option<&Arc<RwLock<Phrase>>>) {
|
||||||
let start = self.clock().next_launch_pulse();
|
let start = self.next_launch_pulse();
|
||||||
self.next_phrase = Some((
|
self.next_phrase = Some((
|
||||||
Instant::from_pulse(&self.clock().timebase(), start as f64),
|
Instant::from_pulse(&self.timebase(), start as f64),
|
||||||
phrase.map(|p|p.clone())
|
phrase.map(|p|p.clone())
|
||||||
));
|
));
|
||||||
self.reset = true;
|
self.reset = true;
|
||||||
|
|
@ -70,7 +70,7 @@ pub trait HasPhrase: PlayheadApi + HasClock + HasMidiBuffer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait MidiInputApi: PlayheadApi + HasMidiBuffer + HasPhrase {
|
pub trait MidiInputApi: ClockApi + PlayheadApi + HasMidiBuffer + HasPhrase {
|
||||||
fn midi_ins (&self) -> &Vec<Port<MidiIn>>;
|
fn midi_ins (&self) -> &Vec<Port<MidiIn>>;
|
||||||
fn midi_ins_mut (&self) -> &mut Vec<Port<MidiIn>>;
|
fn midi_ins_mut (&self) -> &mut Vec<Port<MidiIn>>;
|
||||||
fn has_midi_ins (&self) -> bool {
|
fn has_midi_ins (&self) -> bool {
|
||||||
|
|
@ -147,7 +147,7 @@ pub trait MidiInputApi: PlayheadApi + HasMidiBuffer + HasPhrase {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait MidiOutputApi: PlayheadApi + HasMidiBuffer + HasPhrase + HasClock {
|
pub trait MidiOutputApi: ClockApi + PlayheadApi + HasMidiBuffer + HasPhrase {
|
||||||
fn midi_outs (&self) -> &Vec<Port<MidiOut>>;
|
fn midi_outs (&self) -> &Vec<Port<MidiOut>>;
|
||||||
|
|
||||||
fn midi_outs_mut (&self) -> &mut Vec<Port<MidiOut>>;
|
fn midi_outs_mut (&self) -> &mut Vec<Port<MidiOut>>;
|
||||||
|
|
|
||||||
|
|
@ -3,30 +3,11 @@ use crate::*;
|
||||||
pub trait HasTracks<T: ArrangerTrackApi> {
|
pub trait HasTracks<T: ArrangerTrackApi> {
|
||||||
fn tracks (&self) -> &Vec<T>;
|
fn tracks (&self) -> &Vec<T>;
|
||||||
fn tracks_mut (&mut self) -> &mut Vec<T>;
|
fn tracks_mut (&mut self) -> &mut Vec<T>;
|
||||||
|
fn track_add (&mut self, name: Option<&str>, color: Option<ItemColor>)-> Usually<&mut T>;
|
||||||
|
fn track_del (&mut self, index: usize);
|
||||||
fn track_default_name (&self) -> String {
|
fn track_default_name (&self) -> String {
|
||||||
format!("Track {}", self.tracks().len() + 1)
|
format!("Track {}", self.tracks().len() + 1)
|
||||||
}
|
}
|
||||||
fn track_add (&mut self, name: Option<&str>, color: Option<ItemColor>)
|
|
||||||
-> Usually<&mut T>
|
|
||||||
{
|
|
||||||
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);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
|
|
|
||||||
|
|
@ -30,12 +30,10 @@ submod! {
|
||||||
|
|
||||||
model_scene
|
model_scene
|
||||||
model_track
|
model_track
|
||||||
model_clock
|
//model_clock
|
||||||
model_phrase
|
model_phrase
|
||||||
model_player
|
model_player
|
||||||
model_pool
|
model_pool
|
||||||
model_sequencer
|
|
||||||
model_transport
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//impl Command<ArrangerModel> for ArrangerSceneCommand {
|
//impl Command<ArrangerModel> for ArrangerSceneCommand {
|
||||||
|
|
@ -100,3 +98,15 @@ submod! {
|
||||||
//_ => todo!("arrangement: move forward")
|
//_ => todo!("arrangement: move forward")
|
||||||
//}
|
//}
|
||||||
//}
|
//}
|
||||||
|
|
||||||
|
//impl From<Instant> for Clock {
|
||||||
|
//fn from (current: Instant) -> Self {
|
||||||
|
//Self {
|
||||||
|
//playing: Some(TransportState::Stopped).into(),
|
||||||
|
//started: None.into(),
|
||||||
|
//quant: 24.into(),
|
||||||
|
//sync: (current.timebase.ppq.get() * 4.).into(),
|
||||||
|
//current,
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
//}
|
||||||
|
|
|
||||||
|
|
@ -36,15 +36,3 @@ impl PlayheadApi for Clock {
|
||||||
&self.started
|
&self.started
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Instant> for Clock {
|
|
||||||
fn from (current: Instant) -> Self {
|
|
||||||
Self {
|
|
||||||
playing: Some(TransportState::Stopped).into(),
|
|
||||||
started: None.into(),
|
|
||||||
quant: 24.into(),
|
|
||||||
sync: (current.timebase.ppq.get() * 4.).into(),
|
|
||||||
current,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -1,49 +0,0 @@
|
||||||
use crate::*;
|
|
||||||
|
|
||||||
pub struct SequencerModel {
|
|
||||||
/// State of the JACK transport.
|
|
||||||
transport: TransportModel,
|
|
||||||
/// State of the phrase pool.
|
|
||||||
phrases: Vec<Arc<RwLock<Phrase>>>,
|
|
||||||
/// State of the phrase player.
|
|
||||||
player: MIDIPlayer,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl HasJack for SequencerModel {
|
|
||||||
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
|
||||||
self.transport.jack()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl HasClock 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 HasPhrases for SequencerModel {
|
|
||||||
fn phrases (&self) -> &Vec<Arc<RwLock<Phrase>>> {
|
|
||||||
&self.phrases
|
|
||||||
}
|
|
||||||
fn phrases_mut (&mut self) -> &mut Vec<Arc<RwLock<Phrase>>> {
|
|
||||||
&mut self.phrases
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl HasPlayer for SequencerModel {
|
|
||||||
fn player (&self) -> &MIDIPlayer {
|
|
||||||
&self.player
|
|
||||||
}
|
|
||||||
fn player_mut (&mut self) -> &mut MIDIPlayer {
|
|
||||||
&mut self.player
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -1,43 +0,0 @@
|
||||||
use crate::*;
|
|
||||||
|
|
||||||
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 HasJack for TransportModel {
|
|
||||||
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
|
||||||
&self.jack
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl HasClock 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
|
|
||||||
//}
|
|
||||||
//}
|
|
||||||
|
|
||||||
impl Debug for TransportModel {
|
|
||||||
fn fmt (&self, f: &mut Formatter<'_>) -> std::result::Result<(), Error> {
|
|
||||||
f.debug_struct("transport")
|
|
||||||
.field("jack", &self.jack)
|
|
||||||
.field("transport", &"(JACK transport)")
|
|
||||||
.field("clock", &self.clock)
|
|
||||||
.field("metronome", &self.metronome)
|
|
||||||
.finish()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
@ -290,6 +290,26 @@ impl HasTracks<ArrangerTrack> for ArrangerView<Tui> {
|
||||||
fn tracks_mut (&mut self) -> &mut Vec<ArrangerTrack> {
|
fn tracks_mut (&mut self) -> &mut Vec<ArrangerTrack> {
|
||||||
&mut self.tracks
|
&mut self.tracks
|
||||||
}
|
}
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl HasScenes<ArrangerScene> for ArrangerView<Tui> {
|
impl HasScenes<ArrangerScene> for ArrangerView<Tui> {
|
||||||
|
|
@ -299,6 +319,19 @@ impl HasScenes<ArrangerScene> for ArrangerView<Tui> {
|
||||||
fn scenes_mut (&mut self) -> &mut Vec<ArrangerScene> {
|
fn scenes_mut (&mut self) -> &mut Vec<ArrangerScene> {
|
||||||
&mut self.scenes
|
&mut self.scenes
|
||||||
}
|
}
|
||||||
|
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])
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Display mode of arranger
|
/// Display mode of arranger
|
||||||
|
|
|
||||||
|
|
@ -213,3 +213,27 @@ impl FocusGrid for SequencerApp<Tui> {
|
||||||
// TODO
|
// TODO
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl HasJack for SequencerView {
|
||||||
|
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
||||||
|
self.transport.jack()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HasPhrases for SequencerView {
|
||||||
|
fn phrases (&self) -> &Vec<Arc<RwLock<Phrase>>> {
|
||||||
|
&self.phrases
|
||||||
|
}
|
||||||
|
fn phrases_mut (&mut self) -> &mut Vec<Arc<RwLock<Phrase>>> {
|
||||||
|
&mut self.phrases
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HasPlayer for SequencerView {
|
||||||
|
fn player (&self) -> &MIDIPlayer {
|
||||||
|
&self.player
|
||||||
|
}
|
||||||
|
fn player_mut (&mut self) -> &mut MIDIPlayer {
|
||||||
|
&mut self.player
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -4,11 +4,14 @@ use crate::*;
|
||||||
impl TryFrom<&Arc<RwLock<JackClient>>> for TransportApp<Tui> {
|
impl TryFrom<&Arc<RwLock<JackClient>>> for TransportApp<Tui> {
|
||||||
type Error = Box<dyn std::error::Error>;
|
type Error = Box<dyn std::error::Error>;
|
||||||
fn try_from (jack: &Arc<RwLock<JackClient>>) -> Usually<Self> {
|
fn try_from (jack: &Arc<RwLock<JackClient>>) -> Usually<Self> {
|
||||||
Ok(Self::new(TransportModel {
|
Ok(Self::new(TransportView {
|
||||||
metronome: false,
|
metronome: false,
|
||||||
transport: jack.read().unwrap().transport(),
|
transport: jack.read().unwrap().transport(),
|
||||||
jack: jack.clone(),
|
jack: jack.clone(),
|
||||||
clock: Arc::new(Clock::from(Instant::default()))
|
clock: Arc::new(Clock::from(Instant::default()))
|
||||||
|
focused: false,
|
||||||
|
focus: TransportViewFocus::PlayPause,
|
||||||
|
size: Measure::new(),
|
||||||
}.into(), None, None))
|
}.into(), None, None))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -121,18 +124,6 @@ impl Command<TransportApp<Tui>> for TransportCommand {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<E: Engine> From<TransportModel> for TransportView<E> {
|
|
||||||
fn from (model: TransportModel) -> Self {
|
|
||||||
Self {
|
|
||||||
_engine: Default::default(),
|
|
||||||
focused: false,
|
|
||||||
focus: TransportViewFocus::PlayPause,
|
|
||||||
size: Measure::new(),
|
|
||||||
model
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Stores and displays time-related info.
|
/// Stores and displays time-related info.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub struct TransportView<E: Engine> {
|
pub struct TransportView<E: Engine> {
|
||||||
|
|
@ -298,3 +289,26 @@ impl FocusGrid for TransportApp<Tui> {
|
||||||
// TODO
|
// TODO
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl HasJack for TransportView {
|
||||||
|
fn jack (&self) -> &Arc<RwLock<JackClient>> {
|
||||||
|
&self.jack
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HasClock for TransportView {
|
||||||
|
fn clock (&self) -> &Arc<Clock> {
|
||||||
|
&self.clock
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl std::fmt::Debug for TransportView {
|
||||||
|
fn fmt (&self, f: &mut std::fmt::Formatter<'_>) -> std::result::Result<(), Error> {
|
||||||
|
f.debug_struct("transport")
|
||||||
|
.field("jack", &self.jack)
|
||||||
|
.field("transport", &"(JACK transport)")
|
||||||
|
.field("clock", &self.clock)
|
||||||
|
.field("metronome", &self.metronome)
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue