use crate::action::Action;
use crate::app::InputMode;
use crossterm::event::{self, Event, KeyCode, KeyEvent, KeyModifiers};
use std::time::Duration;
pub fn poll_action(timeout: Duration, mode: &InputMode) -> Option<Action> {
if event::poll(timeout).ok()? {
if let Event::Key(key) = event::read().ok()? {
return map_key(key, mode);
}
}
None
}
fn map_key(key: KeyEvent, mode: &InputMode) -> Option<Action> {
if key.kind != crossterm::event::KeyEventKind::Press {
return None;
}
match mode {
InputMode::Normal => map_normal(key),
InputMode::Search => map_search(key),
InputMode::TapeFilter => map_tape_filter(key),
InputMode::TapeRename => map_tape_rename(key),
InputMode::TapeMove => map_tape_move(key),
}
}
fn map_normal(key: KeyEvent) -> Option<Action> {
match (key.modifiers, key.code) {
(_, KeyCode::Char('q')) => Some(Action::Quit),
(_, KeyCode::Esc) => Some(Action::Quit),
(mods, KeyCode::Char('c')) if mods.contains(KeyModifiers::CONTROL) => Some(Action::Quit),
(_, KeyCode::Char('/')) => Some(Action::EnterSearch),
(_, KeyCode::Char('t')) => Some(Action::EnterTapeFilter),
(mods, KeyCode::Char('f')) if mods.contains(KeyModifiers::CONTROL) => {
Some(Action::EnterSearch)
}
(_, KeyCode::Up) | (_, KeyCode::Char('k')) => Some(Action::PrevStation),
(_, KeyCode::Down) | (_, KeyCode::Char('j')) => Some(Action::NextStation),
(_, KeyCode::Right) | (_, KeyCode::Char('l')) | (_, KeyCode::Char('d')) => {
Some(Action::StepSettingForward)
}
(_, KeyCode::Left) | (_, KeyCode::Char('a')) => Some(Action::StepSettingBackward),
(_, KeyCode::Enter) => Some(Action::PlaySelected),
(_, KeyCode::Char(' ')) => Some(Action::TogglePause),
(_, KeyCode::Char('s')) => Some(Action::Stop),
(_, KeyCode::Char('+')) | (_, KeyCode::Char('=')) => Some(Action::VolumeUp),
(_, KeyCode::Char('-')) => Some(Action::VolumeDown),
(_, KeyCode::Char('m')) => Some(Action::ToggleMute),
(_, KeyCode::Char('f')) => Some(Action::RemoveLibrarySelection),
(_, KeyCode::Char('u')) => Some(Action::UndoRemoveLibrarySelection),
(_, KeyCode::Tab) => Some(Action::NextGenre),
(_, KeyCode::BackTab) => Some(Action::PrevGenre),
(_, KeyCode::Char('?')) | (_, KeyCode::Char('h')) => Some(Action::ToggleHelp),
(_, KeyCode::Char('b')) => Some(Action::CycleLayout),
(_, KeyCode::Char('p')) => Some(Action::NextDeckPage),
(_, KeyCode::Char('v')) => Some(Action::ToggleVisualizerMode),
(_, KeyCode::Char(',')) => Some(Action::ToggleSettings),
(mods, KeyCode::Char('r')) if mods.contains(KeyModifiers::CONTROL) => {
Some(Action::RefreshTapeArchive)
}
(_, KeyCode::Delete) => Some(Action::DeleteSelectedTape),
(_, KeyCode::Char('o')) => Some(Action::OpenSelectedTapeFolder),
(_, KeyCode::Char('g')) => Some(Action::CycleTapePlaybackMode),
(_, KeyCode::Char('i')) => Some(Action::ToggleTapeDetails),
(_, KeyCode::Char('R')) => Some(Action::EnterTapeRename),
(_, KeyCode::Char('M')) => Some(Action::EnterTapeMove),
(_, KeyCode::Char('y')) => Some(Action::ConfirmDeleteTape),
(_, KeyCode::Char('n')) => Some(Action::CancelDeleteTape),
(_, KeyCode::Char('r')) => Some(Action::ToggleRecording),
(_, KeyCode::Char('K')) => Some(Action::KeepRecordingRecovery),
(_, KeyCode::Char('T')) => Some(Action::TrashRecordingRecovery),
(_, KeyCode::Char('D')) => Some(Action::DismissRecordingRecovery),
_ => None,
}
}
fn map_search(key: KeyEvent) -> Option<Action> {
match (key.modifiers, key.code) {
(_, KeyCode::Esc) => Some(Action::ExitSearch),
(mods, KeyCode::Enter) if mods.contains(KeyModifiers::CONTROL) => {
Some(Action::SearchAudition)
}
(_, KeyCode::Char(' ')) => Some(Action::SearchAudition),
(_, KeyCode::Enter) => Some(Action::SearchConfirm),
(_, KeyCode::Up) => Some(Action::PrevStation),
(_, KeyCode::Down) => Some(Action::NextStation),
(_, KeyCode::Backspace) => Some(Action::SearchBackspace),
(mods, KeyCode::Char('c')) if mods.contains(KeyModifiers::CONTROL) => Some(Action::Quit),
(_, KeyCode::Char('\u{3}')) => Some(Action::Quit),
(mods, KeyCode::Char('-')) if has_search_escape_modifier(mods) => Some(Action::VolumeDown),
(mods, KeyCode::Char('+') | KeyCode::Char('=')) if has_search_escape_modifier(mods) => {
Some(Action::VolumeUp)
}
(mods, KeyCode::Char('m')) if has_search_escape_modifier(mods) => Some(Action::ToggleMute),
(_, KeyCode::Char(c)) => Some(Action::SearchInput(c)),
_ => None,
}
}
fn map_tape_filter(key: KeyEvent) -> Option<Action> {
match (key.modifiers, key.code) {
(_, KeyCode::Esc) => Some(Action::ExitTapeFilter),
(_, KeyCode::Enter) => Some(Action::PlaySelected),
(_, KeyCode::Up) | (_, KeyCode::Char('k')) => Some(Action::PrevStation),
(_, KeyCode::Down) | (_, KeyCode::Char('j')) => Some(Action::NextStation),
(_, KeyCode::Backspace) => Some(Action::TapeFilterBackspace),
(mods, KeyCode::Char('r')) if mods.contains(KeyModifiers::CONTROL) => {
Some(Action::RefreshTapeArchive)
}
(mods, KeyCode::Char('c')) if mods.contains(KeyModifiers::CONTROL) => Some(Action::Quit),
(_, KeyCode::Char(c)) if !c.is_control() => Some(Action::TapeFilterInput(c)),
_ => None,
}
}
fn map_tape_rename(key: KeyEvent) -> Option<Action> {
match (key.modifiers, key.code) {
(_, KeyCode::Esc) => Some(Action::CancelTapeManager),
(_, KeyCode::Enter) => Some(Action::ConfirmTapeRename),
(_, KeyCode::Backspace) => Some(Action::TapeManagerBackspace),
(mods, KeyCode::Char('c')) if mods.contains(KeyModifiers::CONTROL) => Some(Action::Quit),
(_, KeyCode::Char(c)) if !c.is_control() => Some(Action::TapeManagerInput(c)),
_ => None,
}
}
fn map_tape_move(key: KeyEvent) -> Option<Action> {
match (key.modifiers, key.code) {
(_, KeyCode::Esc) => Some(Action::CancelTapeManager),
(_, KeyCode::Enter) => Some(Action::ConfirmTapeMove),
(_, KeyCode::Backspace) => Some(Action::TapeManagerBackspace),
(mods, KeyCode::Char('c')) if mods.contains(KeyModifiers::CONTROL) => Some(Action::Quit),
(_, KeyCode::Char(c)) if !c.is_control() => Some(Action::TapeManagerInput(c)),
_ => None,
}
}
fn has_search_escape_modifier(modifiers: KeyModifiers) -> bool {
modifiers.intersects(KeyModifiers::CONTROL | KeyModifiers::ALT)
}
#[cfg(test)]
mod tests {
use super::*;
fn key(code: KeyCode) -> KeyEvent {
KeyEvent::new(code, KeyModifiers::NONE)
}
fn modified_key(code: KeyCode, modifiers: KeyModifiers) -> KeyEvent {
KeyEvent::new(code, modifiers)
}
#[test]
fn search_mode_treats_plain_f_as_text_input() {
assert_eq!(
map_key(key(KeyCode::Char('f')), &InputMode::Search),
Some(Action::SearchInput('f')),
);
}
#[test]
fn search_mode_treats_plain_a_as_text_input() {
assert_eq!(
map_key(key(KeyCode::Char('a')), &InputMode::Search),
Some(Action::SearchInput('a')),
);
}
#[test]
fn search_mode_treats_plain_u_as_text_input() {
assert_eq!(
map_key(key(KeyCode::Char('u')), &InputMode::Search),
Some(Action::SearchInput('u')),
);
}
#[test]
fn search_mode_f2_does_not_add_selected_result() {
assert_eq!(map_key(key(KeyCode::F(2)), &InputMode::Search), None,);
}
#[test]
fn search_mode_insert_does_not_add_selected_result() {
assert_eq!(map_key(key(KeyCode::Insert), &InputMode::Search), None,);
}
#[test]
fn search_mode_plain_audio_keys_remain_text_input() {
assert_eq!(
map_key(key(KeyCode::Char('m')), &InputMode::Search),
Some(Action::SearchInput('m')),
);
assert_eq!(
map_key(key(KeyCode::Char('-')), &InputMode::Search),
Some(Action::SearchInput('-')),
);
assert_eq!(
map_key(key(KeyCode::Char('=')), &InputMode::Search),
Some(Action::SearchInput('=')),
);
assert_eq!(
map_key(key(KeyCode::Char('+')), &InputMode::Search),
Some(Action::SearchInput('+')),
);
}
#[test]
fn search_mode_ctrl_audio_keys_bypass_text_input() {
assert_eq!(
map_key(
modified_key(KeyCode::Char('-'), KeyModifiers::CONTROL),
&InputMode::Search
),
Some(Action::VolumeDown),
);
assert_eq!(
map_key(
modified_key(KeyCode::Char('='), KeyModifiers::CONTROL),
&InputMode::Search
),
Some(Action::VolumeUp),
);
assert_eq!(
map_key(
modified_key(KeyCode::Char('+'), KeyModifiers::CONTROL),
&InputMode::Search
),
Some(Action::VolumeUp),
);
assert_eq!(
map_key(
modified_key(KeyCode::Char('m'), KeyModifiers::CONTROL),
&InputMode::Search
),
Some(Action::ToggleMute),
);
}
#[test]
fn search_mode_alt_audio_keys_bypass_text_input() {
assert_eq!(
map_key(
modified_key(KeyCode::Char('-'), KeyModifiers::ALT),
&InputMode::Search
),
Some(Action::VolumeDown),
);
assert_eq!(
map_key(
modified_key(KeyCode::Char('='), KeyModifiers::ALT),
&InputMode::Search
),
Some(Action::VolumeUp),
);
assert_eq!(
map_key(
modified_key(KeyCode::Char('m'), KeyModifiers::ALT),
&InputMode::Search
),
Some(Action::ToggleMute),
);
}
#[test]
fn normal_mode_right_steps_setting_forward() {
assert_eq!(
map_key(key(KeyCode::Right), &InputMode::Normal),
Some(Action::StepSettingForward),
);
}
#[test]
fn normal_mode_left_steps_setting_backward() {
assert_eq!(
map_key(key(KeyCode::Left), &InputMode::Normal),
Some(Action::StepSettingBackward),
);
}
#[test]
fn normal_mode_l_and_d_step_setting_forward() {
assert_eq!(
map_key(key(KeyCode::Char('l')), &InputMode::Normal),
Some(Action::StepSettingForward),
);
assert_eq!(
map_key(key(KeyCode::Char('d')), &InputMode::Normal),
Some(Action::StepSettingForward),
);
}
#[test]
fn normal_mode_a_steps_setting_backward() {
assert_eq!(
map_key(key(KeyCode::Char('a')), &InputMode::Normal),
Some(Action::StepSettingBackward),
);
}
#[test]
fn normal_mode_f_removes_library_selection() {
assert_eq!(
map_key(key(KeyCode::Char('f')), &InputMode::Normal),
Some(Action::RemoveLibrarySelection),
);
}
#[test]
fn normal_mode_u_undoes_library_removal() {
assert_eq!(
map_key(key(KeyCode::Char('u')), &InputMode::Normal),
Some(Action::UndoRemoveLibrarySelection),
);
}
#[test]
fn normal_mode_ctrl_r_refreshes_tape_archive() {
assert_eq!(
map_key(
modified_key(KeyCode::Char('r'), KeyModifiers::CONTROL),
&InputMode::Normal
),
Some(Action::RefreshTapeArchive),
);
}
#[test]
fn normal_mode_t_enters_tape_filter_action() {
assert_eq!(
map_key(key(KeyCode::Char('t')), &InputMode::Normal),
Some(Action::EnterTapeFilter),
);
}
#[test]
fn tape_filter_mode_collects_text() {
assert_eq!(
map_key(key(KeyCode::Char('s')), &InputMode::TapeFilter),
Some(Action::TapeFilterInput('s')),
);
}
#[test]
fn tape_filter_mode_escape_exits_filter() {
assert_eq!(
map_key(key(KeyCode::Esc), &InputMode::TapeFilter),
Some(Action::ExitTapeFilter),
);
}
#[test]
fn tape_filter_mode_backspace_edits_filter() {
assert_eq!(
map_key(key(KeyCode::Backspace), &InputMode::TapeFilter),
Some(Action::TapeFilterBackspace),
);
}
#[test]
fn normal_mode_delete_requests_tape_delete() {
assert_eq!(
map_key(key(KeyCode::Delete), &InputMode::Normal),
Some(Action::DeleteSelectedTape),
);
}
#[test]
fn normal_mode_o_opens_selected_tape_folder() {
assert_eq!(
map_key(key(KeyCode::Char('o')), &InputMode::Normal),
Some(Action::OpenSelectedTapeFolder),
);
}
#[test]
fn search_mode_space_auditions_selected_result() {
assert_eq!(
map_key(key(KeyCode::Char(' ')), &InputMode::Search),
Some(Action::SearchAudition),
);
}
#[test]
fn search_mode_ctrl_enter_auditions_selected_result_when_supported() {
assert_eq!(
map_key(
modified_key(KeyCode::Enter, KeyModifiers::CONTROL),
&InputMode::Search
),
Some(Action::SearchAudition),
);
}
#[test]
fn search_mode_enter_adds_and_plays_selected_result() {
assert_eq!(
map_key(key(KeyCode::Enter), &InputMode::Search),
Some(Action::SearchConfirm),
);
}
#[test]
fn normal_mode_shift_k_keeps_recording_recovery() {
assert_eq!(
map_key(key(KeyCode::Char('K')), &InputMode::Normal),
Some(Action::KeepRecordingRecovery),
);
}
#[test]
fn normal_mode_shift_t_trashes_recording_recovery() {
assert_eq!(
map_key(key(KeyCode::Char('T')), &InputMode::Normal),
Some(Action::TrashRecordingRecovery),
);
}
#[test]
fn normal_mode_shift_d_dismisses_recording_recovery() {
assert_eq!(
map_key(key(KeyCode::Char('D')), &InputMode::Normal),
Some(Action::DismissRecordingRecovery),
);
}
#[test]
fn normal_mode_g_cycles_tape_playback_mode() {
assert_eq!(
map_key(key(KeyCode::Char('g')), &InputMode::Normal),
Some(Action::CycleTapePlaybackMode),
);
}
}