mirror of
https://codeberg.org/unspeaker/tek.git
synced 2025-12-07 12:16:42 +01:00
wip: reenable standalone sequencer
This commit is contained in:
parent
690a8e8f24
commit
0eb063db1c
7 changed files with 125 additions and 37 deletions
2
Justfile
2
Justfile
|
|
@ -12,3 +12,5 @@ fpush:
|
||||||
git push -fu origin main
|
git push -fu origin main
|
||||||
arranger:
|
arranger:
|
||||||
cargo run --bin tek_arranger
|
cargo run --bin tek_arranger
|
||||||
|
sequencer:
|
||||||
|
cargo run --bin tek_sequencer
|
||||||
|
|
|
||||||
|
|
@ -457,16 +457,7 @@ impl Scene {
|
||||||
}
|
}
|
||||||
/// Returns the pulse length of the longest phrase in the scene
|
/// Returns the pulse length of the longest phrase in the scene
|
||||||
pub fn pulses <E: Engine> (&self, tracks: &[ArrangementTrack<E>]) -> usize {
|
pub fn pulses <E: Engine> (&self, tracks: &[ArrangementTrack<E>]) -> usize {
|
||||||
self.clips.iter().enumerate()
|
self.clips.iter().fold(0, |a, p|a.max(p.as_ref().map(|q|q.read().unwrap().length).unwrap_or(0)))
|
||||||
.filter_map(|(i, c)|c
|
|
||||||
.map(|c|tracks
|
|
||||||
.get(i)
|
|
||||||
.map(|track|track
|
|
||||||
.phrases
|
|
||||||
.get(c))))
|
|
||||||
.filter_map(|p|p)
|
|
||||||
.filter_map(|p|p)
|
|
||||||
.fold(0, |a, p|a.max(p.read().unwrap().length))
|
|
||||||
}
|
}
|
||||||
/// Returns true if all phrases in the scene are currently playing
|
/// Returns true if all phrases in the scene are currently playing
|
||||||
pub fn is_playing <E: Engine> (&self, tracks: &[ArrangementTrack<E>]) -> bool {
|
pub fn is_playing <E: Engine> (&self, tracks: &[ArrangementTrack<E>]) -> bool {
|
||||||
|
|
|
||||||
|
|
@ -36,12 +36,12 @@ impl ArrangerCli {
|
||||||
*arrangement.name.write().unwrap() = name.clone();
|
*arrangement.name.write().unwrap() = name.clone();
|
||||||
}
|
}
|
||||||
for _ in 0..self.tracks {
|
for _ in 0..self.tracks {
|
||||||
let track = arrangement.track_add(None)?;
|
//let track = arrangement.track_add(None)?;
|
||||||
for _ in 0..self.scenes {
|
//for _ in 0..self.scenes {
|
||||||
track.phrases.push(
|
//track.phrases.push(
|
||||||
Arc::new(RwLock::new(Phrase::new("", true, PPQ * 4, None)))
|
//Arc::new(RwLock::new(Phrase::new("", true, PPQ * 4, None)))
|
||||||
);
|
//);
|
||||||
}
|
//}
|
||||||
}
|
}
|
||||||
for _ in 0..self.scenes {
|
for _ in 0..self.scenes {
|
||||||
let _scene = arrangement.scene_add(None)?;
|
let _scene = arrangement.scene_add(None)?;
|
||||||
|
|
|
||||||
|
|
@ -25,6 +25,21 @@ impl Content for Arranger<Tui> {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
/// Focusable items in standalone arranger.
|
||||||
|
impl Focus<3, Tui> for Arranger<Tui> {
|
||||||
|
fn focus (&self) -> usize {
|
||||||
|
self.focus
|
||||||
|
}
|
||||||
|
fn focus_mut (&mut self) -> &mut usize {
|
||||||
|
&mut self.focus
|
||||||
|
}
|
||||||
|
fn focusable (&self) -> [&dyn Focusable<Tui>;3] {
|
||||||
|
focusables!(self.transport, self.arrangement, self.editor)
|
||||||
|
}
|
||||||
|
fn focusable_mut (&mut self) -> [&mut dyn Focusable<Tui>;3] {
|
||||||
|
focusables_mut!(self.transport, self.arrangement, self.editor)
|
||||||
|
}
|
||||||
|
}
|
||||||
/// Handle top-level events in standalone arranger.
|
/// Handle top-level events in standalone arranger.
|
||||||
impl Handle<Tui> for Arranger<Tui> {
|
impl Handle<Tui> for Arranger<Tui> {
|
||||||
fn handle (&mut self, from: &TuiInput) -> Perhaps<bool> {
|
fn handle (&mut self, from: &TuiInput) -> Perhaps<bool> {
|
||||||
|
|
@ -73,21 +88,6 @@ impl Handle<Tui> for Arranger<Tui> {
|
||||||
Ok(Some(true))
|
Ok(Some(true))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/// Focusable items in standalone arranger.
|
|
||||||
impl Focus<3, Tui> for Arranger<Tui> {
|
|
||||||
fn focus (&self) -> usize {
|
|
||||||
self.focus
|
|
||||||
}
|
|
||||||
fn focus_mut (&mut self) -> &mut usize {
|
|
||||||
&mut self.focus
|
|
||||||
}
|
|
||||||
fn focusable (&self) -> [&dyn Focusable<Tui>;3] {
|
|
||||||
focusables!(self.transport, self.arrangement, self.editor)
|
|
||||||
}
|
|
||||||
fn focusable_mut (&mut self) -> [&mut dyn Focusable<Tui>;3] {
|
|
||||||
focusables_mut!(self.transport, self.arrangement, self.editor)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
impl Arranger<Tui> {
|
impl Arranger<Tui> {
|
||||||
pub fn rename_selected (&mut self) {
|
pub fn rename_selected (&mut self) {
|
||||||
let Arrangement { selected, ref name, ref tracks, ref scenes, .. } = self.arrangement;
|
let Arrangement { selected, ref name, ref tracks, ref scenes, .. } = self.arrangement;
|
||||||
|
|
|
||||||
|
|
@ -13,12 +13,16 @@ pub struct Sequencer<E: Engine> {
|
||||||
pub phrases: Arc<RwLock<PhrasePool<E>>>,
|
pub phrases: Arc<RwLock<PhrasePool<E>>>,
|
||||||
/// Phrase editor view
|
/// Phrase editor view
|
||||||
pub editor: PhraseEditor<E>,
|
pub editor: PhraseEditor<E>,
|
||||||
|
/// Which view is focused
|
||||||
|
pub focus: usize
|
||||||
}
|
}
|
||||||
/// Contains all phrases in a project
|
/// Contains all phrases in a project
|
||||||
pub struct PhrasePool<E: Engine> {
|
pub struct PhrasePool<E: Engine> {
|
||||||
_engine: PhantomData<E>,
|
_engine: PhantomData<E>,
|
||||||
/// Phrases in the pool
|
/// Phrases in the pool
|
||||||
pub phrases: Vec<Arc<RwLock<Option<Phrase>>>>,
|
pub phrases: Vec<Arc<RwLock<Option<Phrase>>>>,
|
||||||
|
/// Whether this widget is focused
|
||||||
|
pub focused: bool,
|
||||||
}
|
}
|
||||||
/// A MIDI sequence.
|
/// A MIDI sequence.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
|
@ -93,7 +97,8 @@ impl<E: Engine> PhrasePool<E> {
|
||||||
pub fn new () -> Self {
|
pub fn new () -> Self {
|
||||||
Self {
|
Self {
|
||||||
_engine: Default::default(),
|
_engine: Default::default(),
|
||||||
phrases: vec![Arc::new(RwLock::new(Some(Phrase::default())))]
|
phrases: vec![Arc::new(RwLock::new(Some(Phrase::default())))],
|
||||||
|
focused: false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -19,8 +19,12 @@ pub struct SequencerCli {
|
||||||
impl SequencerCli {
|
impl SequencerCli {
|
||||||
fn run (&self) -> Usually<()> {
|
fn run (&self) -> Usually<()> {
|
||||||
let seq = Sequencer {
|
let seq = Sequencer {
|
||||||
transport: self.transport.unwrap_or(false)
|
focus: 0,
|
||||||
.then_some(Arc::new(RwLock::new(TransportToolbar::new(None)))),
|
editor: PhraseEditor::new(),
|
||||||
|
phrases: Arc::new(RwLock::new(PhrasePool::new())),
|
||||||
|
transport: self.transport.unwrap_or(false).then_some(
|
||||||
|
Arc::new(RwLock::new(TransportToolbar::new(None)))
|
||||||
|
),
|
||||||
};
|
};
|
||||||
if let Some(name) = self.name.as_ref() {
|
if let Some(name) = self.name.as_ref() {
|
||||||
// TODO
|
// TODO
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,72 @@
|
||||||
use crate::*;
|
use crate::*;
|
||||||
|
impl Content for Sequencer<Tui> {
|
||||||
|
type Engine = Tui;
|
||||||
|
fn content (&self) -> impl Widget<Engine = Tui> {
|
||||||
|
Stack::down(move|add|{
|
||||||
|
add(&self.transport)?;
|
||||||
|
add(&self.phrases.clone()
|
||||||
|
.split(Direction::Right, 20, &self.editor as &dyn Widget<Engine = Tui>)
|
||||||
|
.min_y(20)
|
||||||
|
.fill_y())
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/// Focusable items in standalone arranger.
|
||||||
|
impl Focus<3, Tui> for Sequencer<Tui> {
|
||||||
|
fn focus (&self) -> usize {
|
||||||
|
self.focus
|
||||||
|
}
|
||||||
|
fn focus_mut (&mut self) -> &mut usize {
|
||||||
|
&mut self.focus
|
||||||
|
}
|
||||||
|
fn focusable (&self) -> [&dyn Focusable<Tui>;3] {
|
||||||
|
focusables!(self.transport, self.phrases, self.editor)
|
||||||
|
}
|
||||||
|
fn focusable_mut (&mut self) -> [&mut dyn Focusable<Tui>;3] {
|
||||||
|
focusables_mut!(self.transport, self.phrases, self.editor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/// Handle top-level events in standalone arranger.
|
||||||
|
impl Handle<Tui> for Sequencer<Tui> {
|
||||||
|
fn handle (&mut self, from: &TuiInput) -> Perhaps<bool> {
|
||||||
|
let focus = self.focus;
|
||||||
|
let is_first_row = false;
|
||||||
|
let is_last_row = false;
|
||||||
|
match from.event() {
|
||||||
|
key!(KeyCode::Char(' ')) => {
|
||||||
|
if let Some(ref mut transport) = self.transport {
|
||||||
|
transport.write().unwrap().toggle_play()?;
|
||||||
|
} else {
|
||||||
|
return Ok(None)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
key!(KeyCode::Tab) => {
|
||||||
|
self.focus_next();
|
||||||
|
},
|
||||||
|
key!(KeyCode::BackTab) => {
|
||||||
|
self.focus_prev();
|
||||||
|
},
|
||||||
|
key!(KeyCode::Down) => {
|
||||||
|
if focus == 0 {
|
||||||
|
self.focus_next();
|
||||||
|
} else if focus == 1 && is_last_row {
|
||||||
|
self.focus_next();
|
||||||
|
} else {
|
||||||
|
return self.focused_mut().handle(from)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
key!(KeyCode::Up) => {
|
||||||
|
if focus == 1 && is_first_row {
|
||||||
|
self.focus_prev();
|
||||||
|
} else {
|
||||||
|
return self.focused_mut().handle(from)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => return self.focused_mut().handle(from)
|
||||||
|
}
|
||||||
|
Ok(Some(true))
|
||||||
|
}
|
||||||
|
}
|
||||||
// TODO: Display phrases always in order of appearance
|
// TODO: Display phrases always in order of appearance
|
||||||
impl Content for PhrasePool<Tui> {
|
impl Content for PhrasePool<Tui> {
|
||||||
type Engine = Tui;
|
type Engine = Tui;
|
||||||
|
|
@ -14,6 +82,24 @@ impl Content for PhrasePool<Tui> {
|
||||||
.fg(Color::Rgb(70, 80, 50))))
|
.fg(Color::Rgb(70, 80, 50))))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
impl Focusable<Tui> for PhrasePool<Tui> {
|
||||||
|
fn is_focused (&self) -> bool {
|
||||||
|
self.focused
|
||||||
|
}
|
||||||
|
fn set_focused (&mut self, focused: bool) {
|
||||||
|
self.focused = focused
|
||||||
|
}
|
||||||
|
}
|
||||||
|
impl Handle<Tui> for PhrasePool<Tui> {
|
||||||
|
fn handle (&mut self, from: &TuiInput) -> Perhaps<bool> {
|
||||||
|
match from.event() {
|
||||||
|
key!(KeyCode::Up) => todo!(),
|
||||||
|
key!(KeyCode::Down) => todo!(),
|
||||||
|
_ => return Ok(None),
|
||||||
|
}
|
||||||
|
return Ok(Some(true))
|
||||||
|
}
|
||||||
|
}
|
||||||
impl Content for PhraseEditor<Tui> {
|
impl Content for PhraseEditor<Tui> {
|
||||||
type Engine = Tui;
|
type Engine = Tui;
|
||||||
fn content (&self) -> impl Widget<Engine = Tui> {
|
fn content (&self) -> impl Widget<Engine = Tui> {
|
||||||
|
|
@ -21,7 +107,7 @@ impl Content for PhraseEditor<Tui> {
|
||||||
let toolbar = Stack::down(move|add|{
|
let toolbar = Stack::down(move|add|{
|
||||||
//let name = format!("{:>9}", self.name.read().unwrap().as_str());
|
//let name = format!("{:>9}", self.name.read().unwrap().as_str());
|
||||||
//add(&col!("Track:", TuiStyle::bg(name.as_str(), field_bg)))?;
|
//add(&col!("Track:", TuiStyle::bg(name.as_str(), field_bg)))?;
|
||||||
if let Some(phrase) = self.phrase {
|
if let Some(phrase) = &self.phrase {
|
||||||
let phrase = phrase.read().unwrap();
|
let phrase = phrase.read().unwrap();
|
||||||
let length = format!("{}q{}p", phrase.length / PPQ, phrase.length % PPQ);
|
let length = format!("{}q{}p", phrase.length / PPQ, phrase.length % PPQ);
|
||||||
let length = format!("{:>9}", &length);
|
let length = format!("{:>9}", &length);
|
||||||
|
|
@ -49,10 +135,10 @@ impl Content for PhraseEditor<Tui> {
|
||||||
}).fill_y(),
|
}).fill_y(),
|
||||||
// playhead
|
// playhead
|
||||||
CustomWidget::new(|_|Ok(Some([32,2])), |to: &mut TuiOutput|{
|
CustomWidget::new(|_|Ok(Some([32,2])), |to: &mut TuiOutput|{
|
||||||
if let Some(phrase) = self.phrase {
|
if let Some(phrase) = &self.phrase {
|
||||||
let time_0 = self.time_axis.start;
|
let time_0 = self.time_axis.start;
|
||||||
let time_z = self.time_axis.scale;
|
let time_z = self.time_axis.scale;
|
||||||
let now = self.now % phrase.read().unwrap().length;
|
let now = 0; // TODO FIXME: self.now % phrase.read().unwrap().length;
|
||||||
let [x, y, width, _] = to.area();
|
let [x, y, width, _] = to.area();
|
||||||
let x2 = x as usize + Self::H_KEYS_OFFSET;
|
let x2 = x as usize + Self::H_KEYS_OFFSET;
|
||||||
let x3 = x as usize + width as usize;
|
let x3 = x as usize + width as usize;
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue