use super::*;
impl NavState {
pub(crate) fn toggle_mute(&mut self) {
if let Some(t) = self.current_track_mut() {
t.muted = !t.muted;
t.sync_to_audio();
}
}
pub(crate) fn toggle_solo(&mut self) {
if let Some(t) = self.current_track_mut() {
t.soloed = !t.soloed;
t.sync_to_audio();
}
}
pub(crate) fn toggle_arm(&mut self) {
if let Some(t) = self.current_track_mut() {
t.armed = !t.armed;
t.sync_to_audio();
}
}
pub(crate) fn digit_input(&mut self, ch: char) {
if self.focused_pane == Pane::Tracks && self.track_selected {
self.number_buf.push_digit(ch);
}
}
pub(crate) fn tick(&mut self) {
if let Some(clip_num) = self.number_buf.check_timeout() {
self.jump_to_clip(clip_num);
}
}
pub(crate) fn jump_to_clip(&mut self, clip_number: usize) {
if let Some(track) = self.current_track() {
if let Some(idx) = track.clips.iter().position(|c| c.number == clip_number) {
self.track_element = TrackElement::Clip(idx);
self.open_clip_view(self.track_cursor, idx);
}
}
}
pub(crate) fn activate_element(&mut self) {
match self.track_element {
TrackElement::Mute => self.toggle_mute(),
TrackElement::Solo => self.toggle_solo(),
TrackElement::RecordArm => self.toggle_arm(),
TrackElement::Fx => {
self.fx_menu.open = true;
self.fx_menu.cursor = 0;
}
TrackElement::Volume => { }
TrackElement::Clip(idx) => {
self.open_clip_view(self.track_cursor, idx);
self.clip_view.clip_tab = ClipTab::PianoRoll;
self.clip_view.focus = ClipViewFocus::PianoRoll;
}
_ => {}
}
}
pub(crate) fn add_instrument_track(
&mut self,
instrument: InstrumentType,
mixer_id: usize,
handle: std::sync::Arc<phosphor_core::project::TrackHandle>,
) {
let name = match instrument {
InstrumentType::Synth => "synth",
InstrumentType::DrumRack => "drums",
InstrumentType::DX7 => "dx7",
InstrumentType::Jupiter8 => "jup8",
InstrumentType::Odyssey => "odyss",
InstrumentType::Juno60 => "juno",
InstrumentType::Sampler => "smplr",
};
let insert_pos = self.tracks.iter().position(|t| {
matches!(t.kind, TrackKind::SendA | TrackKind::SendB | TrackKind::Master)
}).unwrap_or(self.tracks.len());
let color = insert_pos % 8;
let mut track = TrackState::new(name, color, true, TrackKind::Instrument, vec![]);
track.mixer_id = Some(mixer_id);
track.handle = Some(handle);
track.instrument_type = Some(instrument);
track.synth_params = match instrument {
InstrumentType::Synth | InstrumentType::Sampler => {
phosphor_dsp::synth::PARAM_DEFAULTS.to_vec()
}
InstrumentType::DrumRack => {
phosphor_dsp::drum_rack::PARAM_DEFAULTS.to_vec()
}
InstrumentType::DX7 => {
phosphor_dsp::dx7::PARAM_DEFAULTS.to_vec()
}
InstrumentType::Jupiter8 => {
phosphor_dsp::jupiter::PARAM_DEFAULTS.to_vec()
}
InstrumentType::Odyssey => {
phosphor_dsp::odyssey::PARAM_DEFAULTS.to_vec()
}
InstrumentType::Juno60 => {
phosphor_dsp::juno::PARAM_DEFAULTS.to_vec()
}
};
track.sync_to_audio();
self.tracks.insert(insert_pos, track);
self.track_cursor = insert_pos;
if self.track_cursor >= self.track_scroll + MAX_VISIBLE_TRACKS {
self.track_scroll = self.track_cursor + 1 - MAX_VISIBLE_TRACKS;
}
self.track_selected = true;
self.track_element = TrackElement::Label;
self.show_current_track_controls();
}
pub(crate) fn open_clip_view(&mut self, track_idx: usize, clip_idx: usize) {
self.clip_view_visible = true;
self.clip_view_target = Some((track_idx, clip_idx));
self.clip_view.fx_cursor = 0;
}
pub(crate) fn fx_menu_select(&mut self) {
if let Some(fx_type) = FxType::ALL.get(self.fx_menu.cursor) {
let inst = FxInstance::new(*fx_type);
if let Some(t) = self.current_track_mut() {
t.fx_chain.push(inst);
}
}
self.fx_menu.open = false;
}
pub(crate) fn active_fx_chain_len(&self) -> usize {
match self.clip_view.fx_panel_tab {
FxPanelTab::TrackFx | FxPanelTab::Synth => {
self.current_track().map(|t| t.fx_chain.len().max(1)).unwrap_or(1)
}
}
}
pub(crate) fn receive_clip_snapshot(&mut self, snap: phosphor_core::clip::ClipSnapshot) {
crate::debug_log::system(&format!(
"clip received: track={} events={} notes={} ticks={}..{}",
snap.track_id, snap.event_count, snap.notes.len(),
snap.start_tick, snap.start_tick + snap.length_ticks,
));
if let Some(track) = self.tracks.iter_mut().find(|t| t.mixer_id == Some(snap.track_id)) {
let ppq = phosphor_core::transport::Transport::PPQ;
let beats = (snap.length_ticks as f64 / ppq as f64).ceil() as u16;
let width = beats.max(2);
if let Some(existing) = track.clips.iter_mut().find(|c| {
c.start_tick < snap.start_tick + snap.length_ticks
&& snap.start_tick < c.start_tick + c.length_ticks
}) {
for n in &snap.notes {
crate::debug_log::system(&format!(
" overdub note: pitch={} frac={:.4} dur={:.4} (snap len={})",
n.note, n.start_frac, n.duration_frac, snap.length_ticks
));
}
if snap.length_ticks != existing.length_ticks && existing.length_ticks > 0 {
let scale = snap.length_ticks as f64 / existing.length_ticks as f64;
let offset = (snap.start_tick - existing.start_tick) as f64 / existing.length_ticks as f64;
let mut adjusted = snap.notes.clone();
for n in &mut adjusted {
n.start_frac = n.start_frac * scale + offset;
n.duration_frac *= scale;
}
existing.notes.extend(adjusted);
crate::debug_log::system(&format!(
" rescaled: scale={:.4} offset={:.4}", scale, offset
));
} else {
existing.notes.extend(snap.notes);
}
existing.has_content = true;
existing.length_ticks = existing.length_ticks.max(snap.length_ticks);
existing.width = width.max(existing.width);
crate::debug_log::system(&format!(
" overdub merged: now {} notes (existing len={})", existing.notes.len(), existing.length_ticks
));
} else {
let clip_number = track.clips.len() + 1;
track.clips.push(Clip {
number: clip_number,
width,
has_content: true,
start_tick: snap.start_tick,
length_ticks: snap.length_ticks,
notes: snap.notes,
});
crate::debug_log::system(" new clip created");
}
}
}
}
pub fn initial_tracks() -> Vec<TrackState> {
vec![
TrackState::new("snd a", 5, false, TrackKind::SendA, vec![]),
TrackState::new("snd b", 6, false, TrackKind::SendB, vec![]),
TrackState::new("mstr", 7, false, TrackKind::Master, vec![]),
]
}