wip: full block piano roll

This commit is contained in:
🪞👃🪞 2024-12-03 21:53:34 +01:00
parent 385d95f800
commit 64ac577f4a
2 changed files with 199 additions and 72 deletions

View file

@ -26,6 +26,7 @@ pub struct PhraseEditorModel {
pub(crate) time_scale: AtomicUsize,
pub(crate) edit_mode: PhraseEditMode,
pub(crate) view_mode: PhraseViewMode,
}
impl std::fmt::Debug for PhraseEditorModel {
@ -56,6 +57,7 @@ impl Default for PhraseEditorModel {
now: Pulse::default().into(),
size: Measure::new(),
edit_mode: PhraseEditMode::Scroll,
view_mode: PhraseViewMode::Horizontal,
note_lo: 0.into(),
note_point: 24.into(),
time_start: 0.into(),
@ -109,6 +111,13 @@ pub enum PhraseEditMode {
Scroll,
}
#[derive(Copy, Clone, Debug)]
pub enum PhraseViewMode {
Horizontal,
HorizontalHalf,
Vertical,
}
pub trait HasEditor {
fn editor (&self) -> &PhraseEditorModel;
fn editor_focused (&self) -> bool;

View file

@ -216,88 +216,206 @@ impl<'a> Content for PhraseView<'a> {
}
}
//const NTH_OCTAVE: [&'static str; 11] = [
//"0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "X",
////"-2", "-1", "0", "1", "2", "3", "4", "5", "6", "7", "8",
//];
#[derive(Copy, Clone, Debug)]
pub enum PhraseViewMode {
PianoHorizontal {
time_zoom: usize,
note_zoom: PhraseViewNoteZoom,
},
PianoVertical {
time_zoom: usize,
note_zoom: PhraseViewNoteZoom,
},
}
impl PhraseEditorModel {
pub(crate) fn redraw (phrase: &Phrase) -> BigBuffer {
let mut buf = BigBuffer::new(usize::MAX.min(phrase.length), 65);
Self::fill_seq_bg(&mut buf, phrase.length, phrase.ppq);
Self::fill_seq_fg(&mut buf, &phrase);
buf
#[derive(Copy, Clone, Debug)]
pub enum PhraseViewNoteZoom {
Half,
N(usize),
}
fn fill_seq_bg (buf: &mut BigBuffer, length: usize, ppq: usize) {
for x in 0..buf.width {
// Only fill as far as phrase length
if x as usize >= length { break }
// Fill each row with background characters
for y in 0..buf.height {
if y >= 64 {
break
impl PhraseViewMode {
/// Return a new [BigBuffer] containing a render of the phrase.
fn draw (&self, phrase: &Phrase) -> BigBuffer {
let mut buffer = BigBuffer::new(self.buffer_width(phrase), self.buffer_height(phrase));
match self {
Self::PianoHorizontal { time_zoom, note_zoom } => match note_zoom {
PhraseViewNoteZoom::Half => Self::draw_piano_horizontal_half(
&mut buffer, phrase, *time_zoom
),
PhraseViewNoteZoom::N(_) => Self::draw_piano_horizontal(
&mut buffer, phrase, *time_zoom, 1
),
},
_ => unimplemented!(),
}
buf.get_mut(x, y).map(|cell|{
cell.set_char(if ppq == 0 {
'·'
} else if x % (4 * ppq) == 0 {
'│'
} else if x % ppq == 0 {
'╎'
} else {
'·'
});
cell.set_fg(Color::Rgb(48, 64, 56));
cell.modifier = Modifier::DIM;
});
buffer
}
/// Determine the required width to render the phrase.
fn buffer_width (&self, phrase: &Phrase) -> usize {
match self {
Self::PianoHorizontal { time_zoom, .. } => {
phrase.length / time_zoom
},
Self::PianoVertical { note_zoom, .. } => match note_zoom {
PhraseViewNoteZoom::Half => 64,
PhraseViewNoteZoom::N(n) => 128*n,
},
}
}
}
fn fill_seq_fg (buf: &mut BigBuffer, phrase: &Phrase) {
let mut notes_on = [false;128];
for x in 0..buf.width {
if x as usize >= phrase.length {
break
}
if let Some(notes) = phrase.notes.get(x as usize) {
if phrase.percussive {
for note in notes {
match note {
MidiMessage::NoteOn { key, .. } =>
notes_on[key.as_int() as usize] = true,
_ => {}
/// Determine the required height to render the phrase.
fn buffer_height (&self, phrase: &Phrase) -> usize {
match self {
Self::PianoHorizontal { note_zoom, .. } => match note_zoom {
PhraseViewNoteZoom::Half => 64,
PhraseViewNoteZoom::N(n) => 128*n,
},
Self::PianoVertical { time_zoom, .. } => {
phrase.length / time_zoom
},
}
}
} else {
for note in notes {
match note {
MidiMessage::NoteOn { key, .. } =>
notes_on[key.as_int() as usize] = true,
MidiMessage::NoteOff { key, .. } =>
notes_on[key.as_int() as usize] = false,
_ => {}
}
}
}
for y in 0..buf.height {
if y > 63 {
break
}
let y = 63 - y;
if let Some(block) = half_block(
notes_on[y as usize * 2 + 1],
notes_on[y as usize * 2],
fn draw_piano_horizontal_half (
_: &mut BigBuffer, _: &Phrase, _: usize
) {
buf.get_mut(x, y).map(|cell|{
cell.set_char(block);
cell.set_fg(Color::White);
});
unimplemented!()
}
fn draw_piano_horizontal (
buffer: &mut BigBuffer, phrase: &Phrase, time_zoom: usize, note_zoom: usize
) {
let mut notes_on = [false;128];
for col in 0..buffer.width {
let time_start = time_zoom * col;
let time_end = time_zoom * (col + 1);
for time in time_start..time_end {
for row in 0..buffer.height {
let cell = buffer.get_mut(row, col).unwrap();
if notes_on[row] {
cell.set_char('▄');
} else {
cell.set_char(' ');
}
}
if phrase.percussive {
notes_on.fill(false);
for event in phrase.notes[time].iter() {
match event {
MidiMessage::NoteOn { key, .. } => {
let row = key.as_int() as usize;
buffer.get_mut(row, col).unwrap().set_char('█');
notes_on[row] = true
},
MidiMessage::NoteOff { key, .. } => {
notes_on[key.as_int() as usize] = false
},
_ => {}
}
}
}
}
//for row in 0..buf.height {
//let pitch = 127 - row;
//}
unimplemented!()
}
}
//impl PhraseEditorModel {
//pub(crate) fn redraw (phrase: &Phrase, mode: PhraseViewMode) -> BigBuffer {
//let mut buf = BigBuffer::new(usize::MAX.min(phrase.length), 65);
//Self::fill_seq_bg(mode, &mut buf, phrase.length, phrase.ppq);
//Self::fill_seq_fg(mode, &mut buf, &phrase);
//buf
//}
//fn fill_seq_bg (_mode: PhraseViewMode, buf: &mut BigBuffer, length: usize, ppq: usize) {
//for x in 0..buf.width {
//// Only fill as far as phrase length
//if x as usize >= length { break }
//// Fill each row with background characters
//for y in 0..buf.height {
//if y >= 64 {
//break
//}
//buf.get_mut(x, y).map(|cell|{
//cell.set_char(if ppq == 0 {
//'·'
//} else if x % (4 * ppq) == 0 {
//'│'
//} else if x % ppq == 0 {
//'╎'
//} else {
//'·'
//});
//cell.set_fg(Color::Rgb(48, 64, 56));
//cell.modifier = Modifier::DIM;
//});
//}
//}
//}
//fn fill_seq_fg (mode: PhraseViewMode, buf: &mut BigBuffer, phrase: &Phrase) {
//match mode {
//PhraseViewMode::Horizontal =>
//Self::fill_seq_fg_horizontal(buf, phrase),
//PhraseViewMode::HorizontalHalf =>
//Self::fill_seq_fg_horizontal_half(buf, phrase),
//PhraseViewMode::Vertical =>
//Self::fill_seq_fg_vertical(buf, phrase),
//}
//}
//fn fill_seq_fg_horizontal (buf: &mut BigBuffer, phrase: &Phrase) {
//let mut notes_on = [false;128];
//for x in 0..buf.width {
//}
//}
//fn fill_seq_fg_horizontal_half (buf: &mut BigBuffer, phrase: &Phrase) {
//let mut notes_on = [false;128];
//for x in 0..buf.width {
//if x as usize >= phrase.length {
//break
//}
//if let Some(notes) = phrase.notes.get(x as usize) {
//if phrase.percussive {
//for note in notes {
//match note {
//MidiMessage::NoteOn { key, .. } =>
//notes_on[key.as_int() as usize] = true,
//_ => {}
//}
//}
//} else {
//for note in notes {
//match note {
//MidiMessage::NoteOn { key, .. } =>
//notes_on[key.as_int() as usize] = true,
//MidiMessage::NoteOff { key, .. } =>
//notes_on[key.as_int() as usize] = false,
//_ => {}
//}
//}
//}
//for y in 0..buf.height {
//if y > 63 {
//break
//}
//let y = 63 - y;
//if let Some(block) = half_block(
//notes_on[y as usize * 2 + 1],
//notes_on[y as usize * 2],
//) {
//buf.get_mut(x, y).map(|cell|{
//cell.set_char(block);
//cell.set_fg(Color::White);
//});
//}
//}
//if phrase.percussive {
//notes_on.fill(false);
//}
//}
//}
//}
//}
////const NTH_OCTAVE: [&'static str; 11] = [
////"0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "X",
//////"-2", "-1", "0", "1", "2", "3", "4", "5", "6", "7", "8",
////];