joecalsend/src/frontend/mod.rs

251 lines
7.5 KiB
Rust
Raw Normal View History

use std::{collections::BTreeMap, net::SocketAddr, sync::OnceLock, time::Duration};
use crossterm::event::{Event, EventStream, KeyCode, KeyEvent, KeyEventKind};
use futures::{FutureExt, StreamExt};
use joecalsend::{
Config, JoecalState, Listeners, TransferEvent, UploadDialog,
error::{LocalSendError, Result},
models::Device,
};
use native_dialog::MessageDialogBuilder;
use ratatui::{
DefaultTerminal,
buffer::Buffer,
layout::Rect,
style::Stylize,
symbols::border,
text::{Line, Text},
widgets::{Block, Paragraph, Widget},
};
use tokio::{
sync::mpsc::{UnboundedReceiver, unbounded_channel},
task::JoinSet,
};
pub mod ui;
pub type Peers = BTreeMap<SocketAddr, (String, String)>;
pub struct App {
2025-07-15 23:27:19 +00:00
pub state: OnceLock<JoecalState>,
pub screen: Vec<CurrentScreen>,
pub events: EventStream,
// addr -> (alias, fingerprint)
pub peers: Peers,
// for getting messages back from the web server or web client about things we've done; the
// other end is held by the state
transfer_event_rx: OnceLock<UnboundedReceiver<TransferEvent>>,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum CurrentScreen {
Main,
Sending,
Receiving,
Stopping,
}
impl Default for App {
fn default() -> Self {
Self::new()
}
}
impl App {
2025-07-15 23:27:19 +00:00
pub fn new() -> Self {
App {
2025-07-15 23:27:19 +00:00
state: Default::default(),
screen: vec![CurrentScreen::Main],
peers: Default::default(),
events: Default::default(),
transfer_event_rx: Default::default(),
}
}
2025-07-15 23:27:19 +00:00
#[tokio::main]
2025-07-16 00:28:40 +00:00
pub async fn start_and_run(
2025-07-15 23:27:19 +00:00
&mut self,
terminal: &mut DefaultTerminal,
config: Config,
device: Device,
) -> Result<()> {
let (transfer_event_tx, transfer_event_rx) = unbounded_channel();
let state = JoecalState::new(device, transfer_event_tx)
2025-07-15 23:27:19 +00:00
.await
.expect("Could not create JoecalState");
let _ = self.transfer_event_rx.set(transfer_event_rx);
2025-07-15 23:27:19 +00:00
let mut handles = JoinSet::new();
state.start(&config, &mut handles).await;
let _ = self.state.set(state);
loop {
terminal.draw(|frame| self.draw(frame))?;
self.handle_events().await?;
if let Some(&top) = self.screen.last()
2025-07-09 05:27:00 +00:00
&& top == CurrentScreen::Stopping
{
2025-07-15 23:27:19 +00:00
self.state.get().unwrap().stop().await;
break;
}
2025-07-15 23:27:19 +00:00
let peers = self.state.get().unwrap().peers.lock().await;
self.peers.clear();
2025-07-15 23:27:19 +00:00
peers.iter().for_each(|(fingerprint, (addr, device))| {
let alias = device.alias.clone();
self.peers
.insert(addr.to_owned(), (alias, fingerprint.to_owned()));
});
}
2025-07-15 23:27:19 +00:00
shutdown(&mut handles).await;
Ok(())
}
async fn handle_events(&mut self) -> Result<()> {
tokio::select! {
event = self.events.next().fuse() => {
if let Some(Ok(evt)) = event {
match evt {
Event::Key(key)
if key.kind == KeyEventKind::Press
=> self.handle_key_event(key),
Event::Mouse(_) => {}
Event::Resize(_, _) => {}
_ => {}
}
}
}
transfer_event = self.transfer_event_rx.get_mut().unwrap().recv() => {
if let Some(event) = transfer_event {
match event {
TransferEvent::UploadRequest { alias, id } => {
let sender =
self
.state
.get()
.unwrap()
2025-07-28 20:46:50 +00:00
.get_upload_request(id)
.await
2025-07-28 20:46:50 +00:00
.ok_or(LocalSendError::SessionInactive)?;
// TODO: replace this with ratatui widget dialog
let upload_confirmed = MessageDialogBuilder::default()
.set_title(&alias)
.set_text("Do you want to receive files from this device?")
.confirm()
.show()
.unwrap();
if upload_confirmed {
let _ = sender.send(UploadDialog::UploadConfirm);
} else {
let _ = sender.send(UploadDialog::UploadDeny);
}
}
TransferEvent::Sent => {}
_ => {}
}
}
}
2025-07-15 22:46:13 +00:00
_ = tokio::time::sleep(Duration::from_millis(200)) => {}
}
Ok(())
}
fn handle_key_event(&mut self, key_event: KeyEvent) {
match key_event.code {
KeyCode::Char('q') => self.exit(),
KeyCode::Char('s') => self.send(),
KeyCode::Char('r') => self.recv(),
2025-07-09 05:27:00 +00:00
KeyCode::Esc => self.pop(),
_ => {}
}
}
fn exit(&mut self) {
self.screen.push(CurrentScreen::Stopping);
}
fn send(&mut self) {
2025-07-15 22:46:13 +00:00
let last = self.screen.last();
match last {
Some(CurrentScreen::Sending) => {}
_ => self.screen.push(CurrentScreen::Sending),
}
}
fn recv(&mut self) {
2025-07-15 22:46:13 +00:00
let last = self.screen.last();
match last {
Some(CurrentScreen::Receiving) => {}
_ => self.screen.push(CurrentScreen::Receiving),
}
2025-07-09 05:27:00 +00:00
}
fn pop(&mut self) {
2025-07-15 22:46:13 +00:00
self.screen.pop();
if self.screen.last().is_none() {
self.screen.push(CurrentScreen::Main);
2025-07-09 05:27:00 +00:00
}
}
}
impl Widget for &App {
fn render(self, area: Rect, buf: &mut Buffer) {
let title = Line::from(" Joecalsend ".bold());
let instructions = Line::from(vec![
" Send ".into(),
"<S>".blue().bold(),
" Receive ".into(),
"<R>".blue().bold(),
" Discover ".into(),
"<D>".blue().bold(),
" Quit ".into(),
"<Q> ".blue().bold(),
]);
let block = Block::bordered()
.title(title.centered())
.title_bottom(instructions.centered())
.border_set(border::THICK);
2025-07-09 05:27:00 +00:00
let current_screen = format!(
"{:?}",
self.screen.last().copied().unwrap_or(CurrentScreen::Main)
2025-07-09 05:27:00 +00:00
);
let text = Text::from(Line::from(current_screen.yellow()));
Paragraph::new(text)
.centered()
.block(block)
.render(area, buf);
}
}
2025-07-16 00:28:40 +00:00
async fn shutdown(handles: &mut JoinSet<Listeners>) {
let mut alarm = tokio::time::interval(tokio::time::Duration::from_secs(5));
alarm.tick().await;
loop {
tokio::select! {
join_result = handles.join_next() => {
match join_result {
Some(handle) => match handle {
Ok(h) => println!("Stopped {h:?}"),
Err(e) => println!("Got error {e:?}"),
}
None => break,
}
}
_ = alarm.tick() => {
println!("Exit timeout reached, aborting all unjoined tasks");
handles.abort_all();
break;
},
}
}
}