add keybinds for sending text
This commit is contained in:
parent
01c478c137
commit
95e24d14b3
2 changed files with 87 additions and 76 deletions
|
@ -125,7 +125,7 @@ impl App {
|
||||||
KeyCode::Esc => self.pop(),
|
KeyCode::Esc => self.pop(),
|
||||||
KeyCode::Char('q') => self.exit(),
|
KeyCode::Char('q') => self.exit(),
|
||||||
KeyCode::Tab => *s = SendingScreen::Peers,
|
KeyCode::Tab => *s = SendingScreen::Peers,
|
||||||
KeyCode::Enter => self.send_content().await,
|
KeyCode::Enter => self.chdir_or_send_file().await,
|
||||||
_ => self.file_picker.handle(&event).unwrap_or_default(),
|
_ => self.file_picker.handle(&event).unwrap_or_default(),
|
||||||
},
|
},
|
||||||
SendingScreen::Peers => match key_event.code {
|
SendingScreen::Peers => match key_event.code {
|
||||||
|
@ -137,7 +137,15 @@ impl App {
|
||||||
KeyCode::Down => self.peer_state.select_next(),
|
KeyCode::Down => self.peer_state.select_next(),
|
||||||
_ => {}
|
_ => {}
|
||||||
},
|
},
|
||||||
SendingScreen::Text => {}
|
SendingScreen::Text => match key_event.code {
|
||||||
|
KeyCode::Esc => {
|
||||||
|
self.text = None;
|
||||||
|
*s = SendingScreen::Files;
|
||||||
|
}
|
||||||
|
KeyCode::Tab => *s = SendingScreen::Peers,
|
||||||
|
KeyCode::Enter => self.send_text().await,
|
||||||
|
_ => {}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
_ => match key_event.code {
|
_ => match key_event.code {
|
||||||
KeyCode::Char('q') => self.exit(),
|
KeyCode::Char('q') => self.exit(),
|
||||||
|
@ -230,9 +238,14 @@ impl App {
|
||||||
self.receive_requests.remove(key);
|
self.receive_requests.remove(key);
|
||||||
}
|
}
|
||||||
|
|
||||||
// send content to selected peer, or change directories in the file explorer
|
async fn chdir_or_send_file(&mut self) {
|
||||||
async fn send_content(&mut self) {
|
let file = self.file_picker.current().path().clone();
|
||||||
debug!("sending content");
|
if file.is_dir()
|
||||||
|
&& let Err(e) = self.file_picker.set_cwd(&file)
|
||||||
|
{
|
||||||
|
error!("could not list directory {file:?}: {e}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let Some(peer_idx) = self.peer_state.selected() else {
|
let Some(peer_idx) = self.peer_state.selected() else {
|
||||||
warn!("no peer selected to send to");
|
warn!("no peer selected to send to");
|
||||||
|
@ -243,30 +256,43 @@ impl App {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Some(ref text) = self.text {
|
|
||||||
if let Err(e) = self.service.send_text(&peer.fingerprint, text).await {
|
|
||||||
error!("got error sending \"{text}\" to {}: {e:?}", peer.alias);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
let file = self.file_picker.current().path().clone();
|
|
||||||
if file.is_dir()
|
|
||||||
&& let Err(e) = self.file_picker.set_cwd(&file)
|
|
||||||
{
|
|
||||||
error!("could not list directory {file:?}: {e}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if file.is_file() {
|
if file.is_file() {
|
||||||
debug!("sending {file:?}");
|
debug!("sending {file:?}");
|
||||||
if let Err(e) = self
|
if let Err(e) = self.service.send_file(&peer.fingerprint, file).await {
|
||||||
.service
|
|
||||||
.send_file(&peer.fingerprint, file.clone())
|
|
||||||
.await
|
|
||||||
{
|
|
||||||
error!("got error sending content: {e:?}");
|
error!("got error sending content: {e:?}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// send content to selected peer, or change directories in the file explorer
|
||||||
|
async fn send_text(&mut self) {
|
||||||
|
debug!("sending text");
|
||||||
|
|
||||||
|
let Some(peer_idx) = self.peer_state.selected() else {
|
||||||
|
debug!("no peer selected to send to");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
let Some(peer) = self.peers.get(peer_idx) else {
|
||||||
|
warn!("invalid peer index {peer_idx}");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let Some(text) = &self.text else {
|
||||||
|
debug!("no text to send");
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Err(e) = self.service.send_text(&peer.fingerprint, text).await {
|
||||||
|
error!("got error sending \"{text}\" to {}: {e:?}", peer.alias);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn send_content(&mut self) {
|
||||||
|
if self.text.is_some() {
|
||||||
|
self.send_text().await;
|
||||||
|
} else {
|
||||||
|
self.chdir_or_send_file().await;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -128,12 +128,11 @@ impl Widget for &mut App {
|
||||||
let rx_reqs: Vec<_> = self.receive_requests.values().collect();
|
let rx_reqs: Vec<_> = self.receive_requests.values().collect();
|
||||||
outer_frame(*current_screen, &MAIN_MENU, area, buf);
|
outer_frame(*current_screen, &MAIN_MENU, area, buf);
|
||||||
logger(header_right.inner(header_margin), buf);
|
logger(header_right.inner(header_margin), buf);
|
||||||
let peers = PeersWidget { peers: &self.peers };
|
peers(
|
||||||
ratatui::widgets::StatefulWidget::render(
|
&self.peers,
|
||||||
peers,
|
&mut self.peer_state,
|
||||||
footer_right.inner(footer_margin),
|
footer_right.inner(footer_margin),
|
||||||
buf,
|
buf,
|
||||||
&mut self.peer_state,
|
|
||||||
);
|
);
|
||||||
NetworkInfoWidget.render(footer_left.inner(footer_margin), buf);
|
NetworkInfoWidget.render(footer_left.inner(footer_margin), buf);
|
||||||
receive_requests(
|
receive_requests(
|
||||||
|
@ -169,18 +168,16 @@ impl Widget for &mut App {
|
||||||
SendingScreen::Text => {}
|
SendingScreen::Text => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
let peers = PeersWidget { peers: &self.peers };
|
|
||||||
|
|
||||||
self.file_picker
|
self.file_picker
|
||||||
.widget()
|
.widget()
|
||||||
.render(header_left.inner(header_margin), buf);
|
.render(header_left.inner(header_margin), buf);
|
||||||
logger(header_right.inner(header_margin), buf);
|
logger(header_right.inner(header_margin), buf);
|
||||||
|
|
||||||
ratatui::widgets::StatefulWidget::render(
|
peers(
|
||||||
peers,
|
&self.peers,
|
||||||
|
&mut self.peer_state,
|
||||||
bottom.inner(subscreen_margin),
|
bottom.inner(subscreen_margin),
|
||||||
buf,
|
buf,
|
||||||
&mut self.peer_state,
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
|
@ -272,31 +269,20 @@ fn receive_requests(
|
||||||
ratatui::widgets::StatefulWidget::render(table, area, buf, state);
|
ratatui::widgets::StatefulWidget::render(table, area, buf, state);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
fn peers(peers: &[Peer], state: &mut ListState, area: Rect, buf: &mut Buffer) {
|
||||||
pub struct PeersWidget<'p> {
|
if peers.is_empty() {
|
||||||
pub peers: &'p [Peer],
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'p> ratatui::widgets::StatefulWidget for PeersWidget<'p> {
|
|
||||||
type State = ListState;
|
|
||||||
|
|
||||||
fn render(self, area: Rect, buf: &mut Buffer, state: &mut Self::State)
|
|
||||||
where
|
|
||||||
Self: Sized,
|
|
||||||
{
|
|
||||||
if self.peers.is_empty() {
|
|
||||||
state.select(None);
|
state.select(None);
|
||||||
}
|
}
|
||||||
if state.selected().is_none() {
|
if state.selected().is_none() {
|
||||||
state.select(Some(0));
|
state.select(Some(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut items = Vec::with_capacity(self.peers.len());
|
let mut items = Vec::with_capacity(peers.len());
|
||||||
for Peer {
|
for Peer {
|
||||||
addr,
|
addr,
|
||||||
alias,
|
alias,
|
||||||
fingerprint,
|
fingerprint,
|
||||||
} in self.peers.iter()
|
} in peers
|
||||||
{
|
{
|
||||||
let item = format!("{:?}: {} ({})", addr, alias, fingerprint);
|
let item = format!("{:?}: {} ({})", addr, alias, fingerprint);
|
||||||
let s = Line::from(item.yellow());
|
let s = Line::from(item.yellow());
|
||||||
|
@ -312,7 +298,6 @@ impl<'p> ratatui::widgets::StatefulWidget for PeersWidget<'p> {
|
||||||
.block(block)
|
.block(block)
|
||||||
.highlight_style(Style::new().bg(Color::Rgb(99, 99, 99)));
|
.highlight_style(Style::new().bg(Color::Rgb(99, 99, 99)));
|
||||||
ratatui::widgets::StatefulWidget::render(list, area, buf, state);
|
ratatui::widgets::StatefulWidget::render(list, area, buf, state);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -364,7 +349,7 @@ impl Widget for NetworkInfoWidget {
|
||||||
}
|
}
|
||||||
|
|
||||||
// helpers
|
// helpers
|
||||||
fn centered_rect(area: Rect, width_pct: u16, height_pct: u16) -> Rect {
|
fn _centered_rect(area: Rect, width_pct: u16, height_pct: u16) -> Rect {
|
||||||
let horizontal = Layout::horizontal([Constraint::Percentage(width_pct)]).flex(Flex::Center);
|
let horizontal = Layout::horizontal([Constraint::Percentage(width_pct)]).flex(Flex::Center);
|
||||||
let vertical = Layout::vertical([Constraint::Percentage(height_pct)]).flex(Flex::Center);
|
let vertical = Layout::vertical([Constraint::Percentage(height_pct)]).flex(Flex::Center);
|
||||||
let [area] = vertical.areas(area);
|
let [area] = vertical.areas(area);
|
||||||
|
|
Loading…
Reference in a new issue