use tabs instead of layouts
This commit is contained in:
parent
044101b290
commit
9e2b51c8e6
29
src/app.rs
29
src/app.rs
|
|
@ -1,7 +1,6 @@
|
||||||
use crate::connection::Connection;
|
use crate::connection::Connection;
|
||||||
use crate::list::ContentList;
|
use crate::list::ContentList;
|
||||||
use mpd::Client;
|
use mpd::Client;
|
||||||
use std::collections::VecDeque;
|
|
||||||
|
|
||||||
// Application result type
|
// Application result type
|
||||||
pub type AppResult<T> = std::result::Result<T, Box<dyn std::error::Error>>;
|
pub type AppResult<T> = std::result::Result<T, Box<dyn std::error::Error>>;
|
||||||
|
|
@ -15,6 +14,25 @@ pub struct App {
|
||||||
pub song_list: ContentList<String>,
|
pub song_list: ContentList<String>,
|
||||||
pub queue_list: ContentList<String>,
|
pub queue_list: ContentList<String>,
|
||||||
pub pl_list: ContentList<String>,
|
pub pl_list: ContentList<String>,
|
||||||
|
pub selected_tab: SelectedTab,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Clone)]
|
||||||
|
pub enum SelectedTab {
|
||||||
|
SongList,
|
||||||
|
Queue,
|
||||||
|
Playlists,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SelectedTab {
|
||||||
|
fn as_usize(&self) {
|
||||||
|
match self {
|
||||||
|
SelectedTab::SongList => 0,
|
||||||
|
SelectedTab::Queue => 1,
|
||||||
|
SelectedTab::Playlists => 2,
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl App {
|
impl App {
|
||||||
|
|
@ -35,6 +53,7 @@ impl App {
|
||||||
song_list,
|
song_list,
|
||||||
queue_list,
|
queue_list,
|
||||||
pl_list,
|
pl_list,
|
||||||
|
selected_tab: SelectedTab::SongList,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -73,4 +92,12 @@ impl App {
|
||||||
Self::get_playlist(&mut self.conn.conn)?;
|
Self::get_playlist(&mut self.conn.conn)?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn cycle_tabls(&mut self) {
|
||||||
|
self.selected_tab = match self.selected_tab {
|
||||||
|
SelectedTab::SongList => SelectedTab::Queue,
|
||||||
|
SelectedTab::Queue => SelectedTab::Playlists,
|
||||||
|
SelectedTab::Playlists=> SelectedTab::SongList,
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -158,15 +158,17 @@ impl Connection {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Volume controls
|
// Volume controls
|
||||||
/// Sets the volume
|
pub fn inc_volume(&mut self, v: i8) {
|
||||||
pub fn set_volume(&mut self, u: String) {
|
|
||||||
let cur = self.conn.status().unwrap().volume;
|
let cur = self.conn.status().unwrap().volume;
|
||||||
let sym = u.get(0..1).unwrap();
|
if cur + v <= 100 {
|
||||||
let u: i8 = u.parse::<i8>().unwrap();
|
self.conn.volume(cur + v).unwrap();
|
||||||
if sym == "+" || sym == "-" {
|
}
|
||||||
self.conn.volume(cur + u).unwrap();
|
}
|
||||||
} else {
|
|
||||||
self.conn.volume(u).unwrap();
|
pub fn dec_volume(&mut self, v: i8) {
|
||||||
|
let cur = self.conn.status().unwrap().volume;
|
||||||
|
if cur - v >= 0 {
|
||||||
|
self.conn.volume(cur - v).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,7 +1,8 @@
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
|
|
||||||
use crate::app::{App, AppResult};
|
use crate::app::{App, AppResult, SelectedTab};
|
||||||
use crossterm::event::{KeyCode, KeyEvent, KeyModifiers};
|
use crossterm::event::{KeyCode, KeyEvent, KeyModifiers};
|
||||||
|
use ratatui::style::Modifier;
|
||||||
|
|
||||||
pub fn handle_key_events(key_event: KeyEvent, app: &mut App) -> AppResult<()> {
|
pub fn handle_key_events(key_event: KeyEvent, app: &mut App) -> AppResult<()> {
|
||||||
match key_event.code {
|
match key_event.code {
|
||||||
|
|
@ -12,36 +13,53 @@ pub fn handle_key_events(key_event: KeyEvent, app: &mut App) -> AppResult<()> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
KeyCode::Char('j') => {
|
KeyCode::Char('j') | KeyCode::Down => match app.selected_tab {
|
||||||
app.song_list.next();
|
SelectedTab::SongList => app.song_list.next(),
|
||||||
}
|
SelectedTab::Queue => app.queue_list.next(),
|
||||||
|
SelectedTab::Playlists => app.pl_list.next(),
|
||||||
|
},
|
||||||
|
|
||||||
KeyCode::Char('k') => {
|
KeyCode::Char('k') | KeyCode::Up => match app.selected_tab {
|
||||||
app.song_list.prev();
|
SelectedTab::SongList => app.song_list.prev(),
|
||||||
}
|
SelectedTab::Queue => app.queue_list.prev(),
|
||||||
|
SelectedTab::Playlists => app.pl_list.prev(),
|
||||||
|
},
|
||||||
|
|
||||||
KeyCode::Enter | KeyCode::Char('l') => {
|
KeyCode::Enter | KeyCode::Char('l') => {
|
||||||
|
// app.update_queue();
|
||||||
|
|
||||||
|
match app.selected_tab {
|
||||||
|
SelectedTab::SongList => {
|
||||||
let song = app.conn.get_song_with_only_filename(
|
let song = app.conn.get_song_with_only_filename(
|
||||||
app.conn.songs_filenames.get(app.song_list.index).unwrap(),
|
app.conn.songs_filenames.get(app.song_list.index).unwrap(),
|
||||||
);
|
);
|
||||||
app.conn.push(&song)?;
|
app.conn.push(&song)?;
|
||||||
// app.update_queue();
|
}
|
||||||
|
SelectedTab::Queue => {}
|
||||||
|
SelectedTab::Playlists => {
|
||||||
|
app.conn
|
||||||
|
.push_playlist(app.pl_list.list.get(app.pl_list.index).unwrap())?;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Playback controls
|
// Playback controls
|
||||||
// Toggle Pause
|
// Toggle Pause
|
||||||
KeyCode::Char('p') => {
|
KeyCode::Char('p') => {
|
||||||
app.conn.toggle_pause();
|
app.conn.toggle_pause();
|
||||||
|
app.conn.update_state();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pause
|
// Pause
|
||||||
KeyCode::Char('s') => {
|
KeyCode::Char('s') => {
|
||||||
app.conn.pause();
|
app.conn.pause();
|
||||||
|
app.conn.update_state();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clearn Queue
|
// Clear Queue
|
||||||
KeyCode::Char('x') => {
|
KeyCode::Char('x') => {
|
||||||
app.conn.conn.clear()?;
|
app.conn.conn.clear()?;
|
||||||
|
app.conn.update_state();
|
||||||
// app.update_queue();
|
// app.update_queue();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -49,22 +67,6 @@ pub fn handle_key_events(key_event: KeyEvent, app: &mut App) -> AppResult<()> {
|
||||||
app.conn.play_dmenu()?;
|
app.conn.play_dmenu()?;
|
||||||
}
|
}
|
||||||
|
|
||||||
KeyCode::Down => {
|
|
||||||
if key_event.modifiers == KeyModifiers::SHIFT {
|
|
||||||
app.queue_list.next();
|
|
||||||
} else {
|
|
||||||
app.pl_list.next();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
KeyCode::Up => {
|
|
||||||
if key_event.modifiers == KeyModifiers::SHIFT {
|
|
||||||
app.queue_list.prev();
|
|
||||||
} else {
|
|
||||||
app.pl_list.prev();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
KeyCode::Right => {
|
KeyCode::Right => {
|
||||||
app.conn
|
app.conn
|
||||||
.push_playlist(app.pl_list.list.get(app.pl_list.index).unwrap())?;
|
.push_playlist(app.pl_list.list.get(app.pl_list.index).unwrap())?;
|
||||||
|
|
@ -83,6 +85,43 @@ pub fn handle_key_events(key_event: KeyEvent, app: &mut App) -> AppResult<()> {
|
||||||
let pos = Duration::from_secs(pos.as_secs().wrapping_add(2));
|
let pos = Duration::from_secs(pos.as_secs().wrapping_add(2));
|
||||||
app.conn.conn.seek(place, pos)?;
|
app.conn.conn.seek(place, pos)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
KeyCode::Tab => {
|
||||||
|
app.cycle_tabls();
|
||||||
|
}
|
||||||
|
|
||||||
|
KeyCode::Char('1') => {
|
||||||
|
app.selected_tab = SelectedTab::SongList;
|
||||||
|
}
|
||||||
|
|
||||||
|
KeyCode::Char('2') => {
|
||||||
|
app.selected_tab = SelectedTab::Queue;
|
||||||
|
}
|
||||||
|
|
||||||
|
KeyCode::Char('3') => {
|
||||||
|
app.selected_tab = SelectedTab::Playlists;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
KeyCode::Char('n') => {
|
||||||
|
app.conn.conn.next()?;
|
||||||
|
}
|
||||||
|
|
||||||
|
KeyCode::Char('N') => {
|
||||||
|
app.conn.conn.prev()?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Volume controls
|
||||||
|
KeyCode::Char('=') => {
|
||||||
|
app.conn.inc_volume(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
KeyCode::Char('-') => {
|
||||||
|
app.conn.dec_volume(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
52
src/ui.rs
52
src/ui.rs
|
|
@ -1,6 +1,7 @@
|
||||||
use crate::app::{App, AppResult};
|
use crate::app::{App, AppResult, SelectedTab};
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
prelude::*,
|
prelude::*,
|
||||||
|
style::palette::tailwind,
|
||||||
widgets::{block::Title, *},
|
widgets::{block::Title, *},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -12,25 +13,45 @@ pub fn render(app: &mut App, frame: &mut Frame) {
|
||||||
// - https://github.com/ratatui-org/ratatui/tree/master/examples
|
// - https://github.com/ratatui-org/ratatui/tree/master/examples
|
||||||
|
|
||||||
// Layout
|
// Layout
|
||||||
let main_layout = Layout::default()
|
let layout = Layout::default()
|
||||||
.direction(Direction::Vertical)
|
.direction(Direction::Vertical)
|
||||||
.constraints(vec![Constraint::Percentage(93), Constraint::Percentage(7)])
|
.constraints(vec![
|
||||||
|
Constraint::Percentage(5),
|
||||||
|
Constraint::Percentage(88),
|
||||||
|
Constraint::Percentage(7),
|
||||||
|
])
|
||||||
.split(frame.size());
|
.split(frame.size());
|
||||||
|
//
|
||||||
|
// let outer_layout = Layout::default()
|
||||||
|
// .direction(Direction::Horizontal)
|
||||||
|
// .constraints(vec![Constraint::Percentage(50), Constraint::Percentage(50)])
|
||||||
|
// .split(main_layout[1]);
|
||||||
|
//
|
||||||
|
// let inner_layout = Layout::default()
|
||||||
|
// .direction(Direction::Vertical)
|
||||||
|
// .constraints(vec![Constraint::Percentage(50), Constraint::Percentage(50)])
|
||||||
|
// .split(outer_layout[1]);
|
||||||
|
|
||||||
let outer_layout = Layout::default()
|
// draw_song_list(frame, app, outer_layout[0]);
|
||||||
.direction(Direction::Horizontal)
|
// draw_queue(frame, app, inner_layout[0]);
|
||||||
.constraints(vec![Constraint::Percentage(50), Constraint::Percentage(50)])
|
// draw_playlists(frame, app, inner_layout[1]);
|
||||||
.split(main_layout[0]);
|
draw_progress_bar(frame, app, layout[2]);
|
||||||
|
|
||||||
let inner_layout = Layout::default()
|
let highlight_style = (Color::default(), tailwind::YELLOW.c700);
|
||||||
.direction(Direction::Vertical)
|
let tab = Tabs::new(vec!["Songs List", "Play Queue", "Playlists"])
|
||||||
.constraints(vec![Constraint::Percentage(50), Constraint::Percentage(50)])
|
.block(Block::default().title("Tabs").borders(Borders::ALL))
|
||||||
.split(outer_layout[1]);
|
.style(Style::default().white())
|
||||||
|
.highlight_style(highlight_style)
|
||||||
|
.divider(" ")
|
||||||
|
.select(app.selected_tab.clone() as usize)
|
||||||
|
.padding("", "");
|
||||||
|
frame.render_widget(tab, layout[0]);
|
||||||
|
|
||||||
draw_song_list(frame, app, outer_layout[0]);
|
match app.selected_tab {
|
||||||
draw_queue(frame, app, inner_layout[0]);
|
SelectedTab::SongList => draw_song_list(frame, app, layout[1]),
|
||||||
draw_playlists(frame, app, inner_layout[1]);
|
SelectedTab::Queue => draw_queue(frame, app, layout[1]),
|
||||||
draw_progress_bar(frame, app, main_layout[1]);
|
SelectedTab::Playlists => draw_playlists(frame, app, layout[1]),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// draws list of songs
|
/// draws list of songs
|
||||||
|
|
@ -98,6 +119,7 @@ fn draw_progress_bar(frame: &mut Frame, app: &mut App, size: Rect) {
|
||||||
let title = Block::default()
|
let title = Block::default()
|
||||||
.title(Title::from(format!("{}: ", state).red().bold()))
|
.title(Title::from(format!("{}: ", state).red().bold()))
|
||||||
.title(Title::from(song.green().bold()));
|
.title(Title::from(song.green().bold()));
|
||||||
|
// .title(Title::from(app.conn.conn.status().unwrap_or_default().volume.to_string().yellow())).title_alignment(Alignment::Right);
|
||||||
let progress_bar = LineGauge::default()
|
let progress_bar = LineGauge::default()
|
||||||
.block(title.borders(Borders::ALL))
|
.block(title.borders(Borders::ALL))
|
||||||
.gauge_style(
|
.gauge_style(
|
||||||
|
|
|
||||||
Reference in New Issue