use std::time::Duration; use crate::{ app::{App, AppResult, SelectedTab}, ui::InputMode, }; use crossterm::event::{KeyCode, KeyEvent, KeyModifiers}; use rust_fuzzy_search::{self, fuzzy_search_sorted}; use simple_dmenu::dmenu; pub fn handle_key_events(key_event: KeyEvent, app: &mut App) -> AppResult<()> { if app.inputmode == InputMode::Editing { // Live update let list: Vec<&str> = app .browser .filetree .iter() .map(|(_, f)| f.as_str()) .collect::>(); let res: Vec<(&str, f32)> = fuzzy_search_sorted(&app.search_input, &list); let res = res.iter().map(|(x, _)| *x).collect::>(); for (i, (_, item)) in app.browser.filetree.iter().enumerate() { if item.contains(res.get(0).unwrap()) { app.browser.selected = i; } } match key_event.code { KeyCode::Esc => { app.inputmode = InputMode::Normal; } KeyCode::Char(to_insert) => { app.enter_char(to_insert); } KeyCode::Enter => { let list: Vec<&str> = app .browser .filetree .iter() .map(|(_, f)| f.as_str()) .collect::>(); let res: Vec<(&str, f32)> = fuzzy_search_sorted(&app.search_input, &list); let (res, _) = res.get(0).unwrap(); for (i, (_, item)) in app.browser.filetree.iter().enumerate() { if item.contains(res) { app.browser.selected = i; } } app.search_input.clear(); app.inputmode = InputMode::Normal; app.reset_cursor(); } KeyCode::Backspace => { app.delete_char(); } KeyCode::Left => { app.move_cursor_left(); } KeyCode::Right => { app.move_cursor_right(); } _ => {} } } else { match key_event.code { KeyCode::Char('q') | KeyCode::Esc => app.quit(), KeyCode::Char('c') | KeyCode::Char('C') => { if key_event.modifiers == KeyModifiers::CONTROL { app.quit(); } else { app.conn.conn.clear()?; app.conn.update_status(); } } KeyCode::Char('j') | KeyCode::Down => match app.selected_tab { SelectedTab::DirectoryBrowser => app.browser.next(), SelectedTab::Queue => app.queue_list.next(), SelectedTab::Playlists => app.pl_list.next(), }, KeyCode::Char('k') | KeyCode::Up => match app.selected_tab { SelectedTab::DirectoryBrowser => app.browser.prev(), SelectedTab::Queue => app.queue_list.prev(), SelectedTab::Playlists => app.pl_list.prev(), }, KeyCode::Enter | KeyCode::Char('l') => { // app.update_queue(); match app.selected_tab { SelectedTab::DirectoryBrowser => { app.browser.handle_enter(&mut app.conn)?; } SelectedTab::Queue => { let song = app.conn.get_song_with_only_filename( app.queue_list.list.get(app.queue_list.index).unwrap(), ); app.conn.push(&song)?; } SelectedTab::Playlists => { app.conn .push_playlist(app.pl_list.list.get(app.pl_list.index).unwrap())?; } } app.conn.update_status(); } KeyCode::Char('h') => match app.selected_tab { SelectedTab::DirectoryBrowser => { app.browser.handle_go_back(&mut app.conn)?; } SelectedTab::Queue => {} SelectedTab::Playlists => {} }, // Playback controls // Toggle Pause KeyCode::Char('p') => { app.conn.toggle_pause(); } // Pause KeyCode::Char('s') => { app.conn.pause(); } // Toggle rpeat KeyCode::Char('r') => { app.conn.toggle_repeat(); app.conn.update_status(); } // Toggle random KeyCode::Char('z') => { app.conn.toggle_random(); app.conn.update_status(); } // Dmenu prompt KeyCode::Char('D') => { app.conn.play_dmenu()?; } // add to queue KeyCode::Char('a') => { // let song = app.conn.get_song_with_only_filename( // app.conn.songs_filenames.get(app.song_list.index).unwrap(), // ); let list = app .conn .songs_filenames .iter() .map(|f| f.as_str()) .collect::>(); let (filename, _) = rust_fuzzy_search::fuzzy_search_sorted(&app.browser.path, &list) .get(0) .unwrap() .clone(); let song = app.conn.get_song_with_only_filename(filename); app.conn.conn.push(&song)?; } KeyCode::Right => { app.conn .push_playlist(app.pl_list.list.get(app.pl_list.index).unwrap())?; } KeyCode::Char('f') => { let place = app.conn.conn.status().unwrap().song.unwrap().pos; let (pos, _) = app.conn.conn.status().unwrap().time.unwrap(); let pos = Duration::from_secs(pos.as_secs().wrapping_add(2)); app.conn.conn.seek(place, pos)?; } KeyCode::Char('b') => { let place = app.conn.conn.status().unwrap().song.unwrap().pos; let (pos, _) = app.conn.conn.status().unwrap().time.unwrap(); let pos = Duration::from_secs(pos.as_secs().wrapping_add(2)); app.conn.conn.seek(place, pos)?; } KeyCode::Tab => { app.cycle_tabls(); } KeyCode::Char('1') => { app.selected_tab = SelectedTab::DirectoryBrowser; } KeyCode::Char('2') => { app.selected_tab = SelectedTab::Queue; } KeyCode::Char('3') => { app.selected_tab = SelectedTab::Playlists; } KeyCode::Char('>') => { app.conn.conn.next()?; } KeyCode::Char('<') => { app.conn.conn.prev()?; } // Volume controls KeyCode::Char('=') => { app.conn.inc_volume(2); app.conn.update_status(); } KeyCode::Char('-') => { app.conn.dec_volume(2); app.conn.update_status(); } // Delete highlighted song from the queue KeyCode::Char('d') => { if app.queue_list.index >= app.queue_list.list.len() - 1 { if app.queue_list.index != 0 { app.queue_list.index -= 1; } } app.conn.conn.delete(app.queue_list.index as u32)?; app.update_queue(); } KeyCode::Char('U') => { app.conn.conn.update()?; } KeyCode::Char('L') => { let str = dmenu!(prompt "Search: "); let list = app .conn .songs_filenames .iter() .map(|f| f.as_str()) .collect::>(); let (filename, _) = rust_fuzzy_search::fuzzy_search_sorted(&str, &list) .get(0) .unwrap() .clone(); let song = app.conn.get_song_with_only_filename(filename); app.conn.push(&song)?; } // Search for songs KeyCode::Char('/') => { app.inputmode = InputMode::toggle_editing_states(&app.inputmode); } _ => {} } } Ok(()) }