cmdsys: handle entered areas

This commit is contained in:
🪞👃🪞 2024-11-09 02:34:43 +01:00
parent 31c1db8a5c
commit 1085826849
4 changed files with 65 additions and 24 deletions

View file

@ -6,12 +6,15 @@ pub trait FocusGrid {
fn cursor (&self) -> (usize, usize); fn cursor (&self) -> (usize, usize);
fn cursor_mut (&mut self) -> &mut (usize, usize); fn cursor_mut (&mut self) -> &mut (usize, usize);
fn update_focus (&mut self) {} fn update_focus (&mut self) {}
fn focused (&self) -> &Self::Item { fn focus_enter (&mut self) {}
fn focus_exit (&mut self) {}
fn entered (&self) -> Option<Self::Item>;
fn focused (&self) -> Self::Item {
let (x, y) = self.cursor(); let (x, y) = self.cursor();
&self.layout()[y][x] self.layout()[y][x]
} }
fn focus (&mut self, target: Self::Item) { fn focus (&mut self, target: Self::Item) {
while self.focused() != &target { while self.focused() != target {
self.focus_next() self.focus_next()
} }
} }
@ -50,7 +53,7 @@ pub trait FocusGrid {
self.update_focus(); self.update_focus();
} }
fn focus_next (&mut self) { fn focus_next (&mut self) {
let current = *self.focused(); let current = self.focused();
let (x, y) = self.cursor(); let (x, y) = self.cursor();
if x < self.layout()[y].len().saturating_sub(1) { if x < self.layout()[y].len().saturating_sub(1) {
self.focus_right(); self.focus_right();
@ -58,13 +61,13 @@ pub trait FocusGrid {
self.focus_down(); self.focus_down();
self.cursor_mut().0 = 0; self.cursor_mut().0 = 0;
} }
if *self.focused() == current { // FIXME: prevent infinite loop if self.focused() == current { // FIXME: prevent infinite loop
self.focus_next() self.focus_next()
} }
self.update_focus(); self.update_focus();
} }
fn focus_prev (&mut self) { fn focus_prev (&mut self) {
let current = *self.focused(); let current = self.focused();
let (x, _) = self.cursor(); let (x, _) = self.cursor();
if x > 0 { if x > 0 {
self.focus_left(); self.focus_left();
@ -74,13 +77,11 @@ pub trait FocusGrid {
let next_x = self.layout()[y].len().saturating_sub(1); let next_x = self.layout()[y].len().saturating_sub(1);
self.cursor_mut().0 = next_x; self.cursor_mut().0 = next_x;
} }
if *self.focused() == current { // FIXME: prevent infinite loop if self.focused() == current { // FIXME: prevent infinite loop
self.focus_prev() self.focus_prev()
} }
self.update_focus(); self.update_focus();
} }
fn focus_enter (&mut self) {}
fn focus_exit (&mut self) {}
} }
#[derive(Clone, PartialEq)] #[derive(Clone, PartialEq)]

View file

@ -274,10 +274,32 @@ impl<E: Engine> FocusGrid for Arranger<E> {
type Item = ArrangerFocus; type Item = ArrangerFocus;
fn cursor (&self) -> (usize, usize) { self.focus_cursor } fn cursor (&self) -> (usize, usize) { self.focus_cursor }
fn cursor_mut (&mut self) -> &mut (usize, usize) { &mut self.focus_cursor } fn cursor_mut (&mut self) -> &mut (usize, usize) { &mut self.focus_cursor }
fn focus_enter (&mut self) { self.entered = true } fn focus_enter (&mut self) {
fn focus_exit (&mut self) { self.entered = false } let focused = self.focused();
if !self.entered {
self.entered = true;
use ArrangerFocus::*;
if let Some(transport) = self.transport.as_ref() {
//transport.write().unwrap().entered = focused == Transport
}
self.arrangement.entered = focused == Arrangement;
//self.phrases.write().unwrap().entered = focused == PhrasePool;
self.editor.entered = focused == PhraseEditor;
}
}
fn focus_exit (&mut self) {
if self.entered {
self.entered = false;
self.arrangement.entered = false;
self.editor.entered = false;
}
}
fn entered (&self) -> Option<ArrangerFocus> { fn entered (&self) -> Option<ArrangerFocus> {
if self.entered { Some(self.focused()) } else { None } if self.entered {
Some(self.focused())
} else {
None
}
} }
fn layout (&self) -> &[&[ArrangerFocus]] { fn layout (&self) -> &[&[ArrangerFocus]] {
use ArrangerFocus::*; use ArrangerFocus::*;
@ -288,17 +310,14 @@ impl<E: Engine> FocusGrid for Arranger<E> {
] ]
} }
fn update_focus (&mut self) { fn update_focus (&mut self) {
use ArrangerFocus::*;
let focused = self.focused(); let focused = self.focused();
if let Some(transport) = self.transport.as_ref() { if let Some(transport) = self.transport.as_ref() {
transport.write().unwrap().focused = transport.write().unwrap().focused = focused == Transport
focused == ArrangerFocus::Transport
} }
self.arrangement.focused = self.arrangement.focused = focused == Arrangement;
focused == ArrangerFocus::Arrangement; self.phrases.write().unwrap().focused = focused == PhrasePool;
self.phrases.write().unwrap().focused = self.editor.focused = focused == PhraseEditor;
focused == ArrangerFocus::PhrasePool;
self.editor.focused =
focused == ArrangerFocus::PhraseEditor;
self.update_status(); self.update_status();
} }
} }

View file

@ -526,8 +526,19 @@ impl<'a> Content for HorizontalArranger<'a, Tui> {
} }
/// 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, i: &TuiInput) -> Perhaps<bool> {
Ok(if let Some(command) = ArrangerCommand::match_input(self, from) { if let Some(entered) = self.entered() {
use ArrangerFocus::*;
if let Some(true) = match entered {
Transport => self.transport.as_mut().map(|t|t.handle(i)).transpose()?.flatten(),
Arrangement => self.arrangement.handle(i)?,
PhrasePool => self.phrases.write().unwrap().handle(i)?,
PhraseEditor => self.editor.handle(i)?,
} {
return Ok(Some(true))
}
}
Ok(if let Some(command) = ArrangerCommand::match_input(self, i) {
let _undo = command.execute(self)?; let _undo = command.execute(self)?;
Some(true) Some(true)
} else { } else {

View file

@ -11,8 +11,18 @@ impl Content for Sequencer<Tui> {
} }
} }
impl Handle<Tui> for Sequencer<Tui> { impl Handle<Tui> for Sequencer<Tui> {
fn handle (&mut self, from: &TuiInput) -> Perhaps<bool> { fn handle (&mut self, i: &TuiInput) -> Perhaps<bool> {
if let Some(command) = SequencerCommand::match_input(self, from) { if let Some(entered) = self.entered() {
use SequencerFocus::*;
if let Some(true) = match entered {
Transport => self.transport.as_mut().map(|t|t.handle(i)).transpose()?.flatten(),
PhrasePool => self.phrases.write().unwrap().handle(i)?,
PhraseEditor => self.editor.handle(i)?,
} {
return Ok(Some(true))
}
}
if let Some(command) = SequencerCommand::match_input(self, i) {
let _undo = command.execute(self)?; let _undo = command.execute(self)?;
return Ok(Some(true)) return Ok(Some(true))
} }