veilid/veilid-cli/src/ui.rs

953 lines
32 KiB
Rust
Raw Normal View History

2021-11-22 16:28:30 +00:00
use crate::command_processor::*;
2022-09-06 20:49:43 +00:00
use crate::peers_table_view::*;
2021-11-22 16:28:30 +00:00
use crate::settings::Settings;
2023-06-08 18:07:09 +00:00
use crate::tools::*;
2021-11-22 16:28:30 +00:00
use crossbeam_channel::Sender;
use cursive::align::*;
use cursive::event::*;
use cursive::theme::*;
use cursive::traits::*;
use cursive::utils::markup::StyledString;
use cursive::views::*;
use cursive::Cursive;
use cursive::CursiveRunnable;
use cursive_flexi_logger_view::{CursiveLogWriter, FlexiLoggerView};
2022-09-06 20:49:43 +00:00
//use cursive_multiplex::*;
2021-11-22 16:28:30 +00:00
use std::collections::{HashMap, VecDeque};
2022-06-26 21:00:05 +00:00
use thiserror::Error;
2021-12-14 14:48:33 +00:00
2021-11-22 16:28:30 +00:00
//////////////////////////////////////////////////////////////
///
struct Dirty<T> {
2022-09-06 20:49:43 +00:00
value: T,
2021-11-22 16:28:30 +00:00
dirty: bool,
}
impl<T> Dirty<T> {
pub fn new(value: T) -> Self {
2021-12-04 18:18:44 +00:00
Self { value, dirty: true }
2021-11-22 16:28:30 +00:00
}
pub fn set(&mut self, value: T) {
self.value = value;
self.dirty = true;
}
pub fn get(&self) -> &T {
&self.value
}
// pub fn get_mut(&mut self) -> &mut T {
// &mut self.value
// }
pub fn take_dirty(&mut self) -> bool {
let is_dirty = self.dirty;
self.dirty = false;
is_dirty
}
}
2021-12-11 01:14:33 +00:00
pub type UICallback = Box<dyn Fn(&mut Cursive) + Send>;
2021-11-22 16:28:30 +00:00
struct UIState {
2023-06-08 01:55:23 +00:00
attachment_state: Dirty<String>,
2022-12-26 21:33:48 +00:00
public_internet_ready: Dirty<bool>,
local_network_ready: Dirty<bool>,
2022-05-16 15:52:48 +00:00
network_started: Dirty<bool>,
network_down_up: Dirty<(f32, f32)>,
2021-11-22 16:28:30 +00:00
connection_state: Dirty<ConnectionState>,
2023-06-08 01:55:23 +00:00
peers_state: Dirty<Vec<json::JsonValue>>,
2022-11-16 17:49:53 +00:00
node_id: Dirty<String>,
2021-11-22 16:28:30 +00:00
}
impl UIState {
pub fn new() -> Self {
Self {
2023-06-08 01:55:23 +00:00
attachment_state: Dirty::new("Detached".to_owned()),
2022-12-26 21:33:48 +00:00
public_internet_ready: Dirty::new(false),
local_network_ready: Dirty::new(false),
2022-05-16 15:52:48 +00:00
network_started: Dirty::new(false),
network_down_up: Dirty::new((0.0, 0.0)),
2021-11-22 16:28:30 +00:00
connection_state: Dirty::new(ConnectionState::Disconnected),
2022-09-06 20:49:43 +00:00
peers_state: Dirty::new(Vec::new()),
2022-11-16 17:49:53 +00:00
node_id: Dirty::new("".to_owned()),
2021-11-22 16:28:30 +00:00
}
}
}
2021-12-14 14:48:33 +00:00
//#[derive(Error, Debug)]
//#[error("???")]
//struct UIError;
2021-11-22 16:28:30 +00:00
pub struct UIInner {
ui_state: UIState,
log_colors: HashMap<Level, cursive::theme::Color>,
cmdproc: Option<CommandProcessor>,
cmd_history: VecDeque<String>,
cmd_history_position: usize,
cmd_history_max_size: usize,
connection_dialog_state: Option<ConnectionState>,
}
pub struct UI {
2023-06-08 18:07:09 +00:00
siv: CursiveRunnable,
inner: Arc<Mutex<UIInner>>,
2021-11-22 16:28:30 +00:00
}
2022-06-26 21:00:05 +00:00
#[derive(Error, Debug)]
pub enum DumbError {
// #[error("{0}")]
// Message(String),
}
2021-11-22 16:28:30 +00:00
impl UI {
2021-12-11 01:14:33 +00:00
/////////////////////////////////////////////////////////////////////////////////////
// Private functions
2021-11-22 16:28:30 +00:00
fn command_processor(s: &mut Cursive) -> CommandProcessor {
let inner = Self::inner(s);
inner.cmdproc.as_ref().unwrap().clone()
}
2023-06-08 18:07:09 +00:00
fn inner(s: &mut Cursive) -> MutexGuard<'_, UIInner> {
s.user_data::<Arc<Mutex<UIInner>>>().unwrap().lock()
2021-11-22 16:28:30 +00:00
}
2023-06-08 18:07:09 +00:00
fn inner_mut(s: &mut Cursive) -> MutexGuard<'_, UIInner> {
s.user_data::<Arc<Mutex<UIInner>>>().unwrap().lock()
2021-11-22 16:28:30 +00:00
}
fn setup_colors(siv: &mut CursiveRunnable, inner: &mut UIInner, settings: &Settings) {
// Make colors
let mut theme = cursive::theme::load_default();
theme.shadow = settings.interface.theme.shadow;
theme.borders = BorderStyle::from(&settings.interface.theme.borders);
theme.palette.set_color(
"background",
Color::parse(settings.interface.theme.colors.background.as_str()).unwrap(),
);
theme.palette.set_color(
"shadow",
Color::parse(settings.interface.theme.colors.shadow.as_str()).unwrap(),
);
theme.palette.set_color(
"view",
Color::parse(settings.interface.theme.colors.view.as_str()).unwrap(),
);
theme.palette.set_color(
"primary",
Color::parse(settings.interface.theme.colors.primary.as_str()).unwrap(),
);
theme.palette.set_color(
"secondary",
Color::parse(settings.interface.theme.colors.secondary.as_str()).unwrap(),
);
theme.palette.set_color(
"tertiary",
Color::parse(settings.interface.theme.colors.tertiary.as_str()).unwrap(),
);
theme.palette.set_color(
"title_primary",
Color::parse(settings.interface.theme.colors.title_primary.as_str()).unwrap(),
);
theme.palette.set_color(
"title_secondary",
Color::parse(settings.interface.theme.colors.title_secondary.as_str()).unwrap(),
);
theme.palette.set_color(
"highlight",
Color::parse(settings.interface.theme.colors.highlight.as_str()).unwrap(),
);
theme.palette.set_color(
"highlight_inactive",
Color::parse(settings.interface.theme.colors.highlight_inactive.as_str()).unwrap(),
);
theme.palette.set_color(
"highlight_text",
Color::parse(settings.interface.theme.colors.highlight_text.as_str()).unwrap(),
);
siv.set_theme(theme);
// Make log colors
let mut colors = HashMap::<Level, cursive::theme::Color>::new();
colors.insert(
Level::Trace,
Color::parse(settings.interface.theme.log_colors.trace.as_str()).unwrap(),
);
colors.insert(
Level::Debug,
Color::parse(settings.interface.theme.log_colors.debug.as_str()).unwrap(),
);
colors.insert(
Level::Info,
Color::parse(settings.interface.theme.log_colors.info.as_str()).unwrap(),
);
colors.insert(
Level::Warn,
Color::parse(settings.interface.theme.log_colors.warn.as_str()).unwrap(),
);
colors.insert(
Level::Error,
Color::parse(settings.interface.theme.log_colors.error.as_str()).unwrap(),
);
inner.log_colors = colors;
}
fn setup_quit_handler(siv: &mut Cursive) {
siv.clear_global_callbacks(cursive::event::Event::CtrlChar('c'));
siv.set_on_pre_event(cursive::event::Event::CtrlChar('c'), UI::quit_handler);
siv.set_global_callback(cursive::event::Event::Key(Key::Esc), UI::quit_handler);
}
fn quit_handler(siv: &mut Cursive) {
siv.add_layer(
Dialog::text("Do you want to exit?")
.button("Yes", |s| s.quit())
2021-12-04 18:18:44 +00:00
.button("No", |s| {
2021-11-22 16:28:30 +00:00
s.pop_layer();
2021-12-04 18:18:44 +00:00
UI::setup_quit_handler(s);
2021-11-22 16:28:30 +00:00
}),
);
siv.set_on_pre_event(cursive::event::Event::CtrlChar('c'), |s| {
s.quit();
});
2021-12-04 18:18:44 +00:00
siv.set_global_callback(cursive::event::Event::Key(Key::Esc), |s| {
2021-11-22 16:28:30 +00:00
s.pop_layer();
2021-12-04 18:18:44 +00:00
UI::setup_quit_handler(s);
2021-11-22 16:28:30 +00:00
});
}
2022-11-22 01:21:46 +00:00
fn clear_handler(siv: &mut Cursive) {
cursive_flexi_logger_view::clear_log();
UI::update_cb(siv);
}
2022-11-22 01:55:10 +00:00
fn node_events_panel(s: &mut Cursive) -> ViewRef<Panel<ScrollView<FlexiLoggerView>>> {
2022-11-16 17:49:53 +00:00
s.find_name("node-events-panel").unwrap()
}
2021-11-22 16:28:30 +00:00
fn command_line(s: &mut Cursive) -> ViewRef<EditView> {
s.find_name("command-line").unwrap()
}
fn button_attach(s: &mut Cursive) -> ViewRef<Button> {
s.find_name("button-attach").unwrap()
}
fn status_bar(s: &mut Cursive) -> ViewRef<TextView> {
s.find_name("status-bar").unwrap()
}
2022-09-06 20:49:43 +00:00
fn peers(s: &mut Cursive) -> ViewRef<PeersTableView> {
s.find_name("peers").unwrap()
}
2022-12-26 21:33:48 +00:00
fn render_attachment_state(inner: &mut UIInner) -> String {
2023-06-08 01:55:23 +00:00
let att = match inner.ui_state.attachment_state.get().as_str() {
"Detached" => "[----]",
"Attaching" => "[/ ]",
"AttachedWeak" => "[| ]",
"AttachedGood" => "[|| ]",
"AttachedStrong" => "[||| ]",
"FullyAttached" => "[||||]",
"OverAttached" => "[++++]",
"Detaching" => "[////]",
_ => "[????]",
2022-12-26 21:33:48 +00:00
};
let pi = if *inner.ui_state.public_internet_ready.get() {
"+P"
} else {
"-p"
};
let ln = if *inner.ui_state.local_network_ready.get() {
"+L"
} else {
"-l"
};
format!("{}{}{}", att, pi, ln)
2021-11-22 16:28:30 +00:00
}
2022-05-16 15:52:48 +00:00
fn render_network_status(inner: &mut UIInner) -> String {
match inner.ui_state.network_started.get() {
false => "Down: ----KB/s Up: ----KB/s".to_owned(),
true => {
let (d, u) = inner.ui_state.network_down_up.get();
format!("Down: {:.2}KB/s Up: {:.2}KB/s", d, u)
}
}
}
2021-11-22 16:28:30 +00:00
fn render_button_attach<'a>(inner: &mut UIInner) -> (&'a str, bool) {
if let ConnectionState::Connected(_, _) = inner.ui_state.connection_state.get() {
2023-06-08 01:55:23 +00:00
match inner.ui_state.attachment_state.get().as_str() {
"Detached" => ("Attach", true),
"Attaching" => ("Detach", true),
"AttachedWeak" => ("Detach", true),
"AttachedGood" => ("Detach", true),
"AttachedStrong" => ("Detach", true),
"FullyAttached" => ("Detach", true),
"OverAttached" => ("Detach", true),
"Detaching" => ("Detach", false),
_ => ("???", false),
2021-11-22 16:28:30 +00:00
}
} else {
(" ---- ", false)
}
}
fn on_command_line_edit(s: &mut Cursive, text: &str, _pos: usize) {
let mut inner = Self::inner_mut(s);
// save edited command to newest history slot
let hlen = inner.cmd_history.len();
inner.cmd_history_position = hlen - 1;
inner.cmd_history[hlen - 1] = text.to_owned();
}
2021-12-11 01:14:33 +00:00
fn enable_command_ui(s: &mut Cursive, enabled: bool) {
2021-11-22 16:28:30 +00:00
Self::command_line(s).set_enabled(enabled);
Self::button_attach(s).set_enabled(enabled);
}
2021-12-11 01:14:33 +00:00
fn display_string_dialog_cb(
s: &mut Cursive,
title: String,
contents: String,
close_cb: UICallback,
) {
// Creates a dialog around some text with a single button
2021-12-14 14:48:33 +00:00
let close_cb = Rc::new(close_cb);
let close_cb2 = close_cb.clone();
2021-12-11 01:14:33 +00:00
s.add_layer(
2021-12-14 14:48:33 +00:00
Dialog::around(TextView::new(contents).scrollable())
2021-12-11 01:14:33 +00:00
.title(title)
.button("Close", move |s| {
s.pop_layer();
close_cb(s);
2022-06-08 13:33:41 +00:00
}), //.wrap_with(CircularFocus::new)
//.wrap_tab(),
2021-12-11 01:14:33 +00:00
);
2021-12-14 14:48:33 +00:00
s.set_global_callback(cursive::event::Event::Key(Key::Esc), move |s| {
s.set_global_callback(cursive::event::Event::Key(Key::Esc), UI::quit_handler);
s.pop_layer();
close_cb2(s);
});
2021-12-11 01:14:33 +00:00
}
2021-11-22 16:28:30 +00:00
fn run_command(s: &mut Cursive, text: &str) -> Result<(), String> {
// disable ui
Self::enable_command_ui(s, false);
2022-11-03 15:28:29 +00:00
2021-11-22 16:28:30 +00:00
// run command
2022-11-03 15:28:29 +00:00
s.set_global_callback(cursive::event::Event::Key(Key::Esc), |s| {
let cmdproc = Self::command_processor(s);
cmdproc.cancel_command();
});
2021-11-22 16:28:30 +00:00
let cmdproc = Self::command_processor(s);
cmdproc.run_command(
text,
2021-12-11 01:14:33 +00:00
Box::new(|s| {
2022-11-03 15:28:29 +00:00
s.set_global_callback(cursive::event::Event::Key(Key::Esc), UI::quit_handler);
2021-12-11 01:14:33 +00:00
Self::enable_command_ui(s, true);
2021-11-22 16:28:30 +00:00
}),
)
}
fn on_command_line_entered(s: &mut Cursive, text: &str) {
if text.trim().is_empty() {
return;
}
// run command
cursive_flexi_logger_view::push_to_log(StyledString::styled(
format!("> {}", text),
ColorStyle::primary(),
));
match Self::run_command(s, text) {
Ok(_) => {}
Err(e) => {
2021-12-04 18:18:44 +00:00
let color = *Self::inner_mut(s).log_colors.get(&Level::Error).unwrap();
2021-11-22 16:28:30 +00:00
cursive_flexi_logger_view::push_to_log(StyledString::styled(
format!(" Error: {}", e),
color,
));
}
}
// save to history unless it's a duplicate
{
let mut inner = Self::inner_mut(s);
let hlen = inner.cmd_history.len();
inner.cmd_history[hlen - 1] = text.to_owned();
if hlen >= 2 && inner.cmd_history[hlen - 1] == inner.cmd_history[hlen - 2] {
inner.cmd_history[hlen - 1] = "".to_string();
} else {
if hlen == inner.cmd_history_max_size {
inner.cmd_history.pop_front();
}
inner.cmd_history.push_back("".to_string());
}
let hlen = inner.cmd_history.len();
inner.cmd_history_position = hlen - 1;
}
// Clear the edit field
let mut cmdline = Self::command_line(s);
cmdline.set_content("");
}
fn on_command_line_history(s: &mut Cursive, dir: bool) {
let mut cmdline = Self::command_line(s);
let mut inner = Self::inner_mut(s);
// if at top of buffer or end of buffer, ignore
if (!dir && inner.cmd_history_position == 0)
|| (dir && inner.cmd_history_position == (inner.cmd_history.len() - 1))
{
return;
}
// move the history position
if dir {
inner.cmd_history_position += 1;
} else {
inner.cmd_history_position -= 1;
}
// replace text with current line
let hlen = inner.cmd_history_position;
cmdline.set_content(inner.cmd_history[hlen].as_str());
}
fn on_button_attach_pressed(s: &mut Cursive) {
2023-06-08 01:55:23 +00:00
let action: Option<bool> = match Self::inner_mut(s).ui_state.attachment_state.get().as_str()
{
"Detached" => Some(true),
"Attaching" => Some(false),
"AttachedWeak" => Some(false),
"AttachedGood" => Some(false),
"AttachedStrong" => Some(false),
"FullyAttached" => Some(false),
"OverAttached" => Some(false),
"Detaching" => None,
_ => None,
2021-11-22 16:28:30 +00:00
};
2023-06-08 18:07:09 +00:00
let cmdproc = Self::command_processor(s);
2021-11-22 16:28:30 +00:00
if let Some(a) = action {
if a {
cmdproc.attach();
} else {
cmdproc.detach();
}
}
}
fn refresh_button_attach(s: &mut Cursive) {
let mut button_attach = UI::button_attach(s);
let mut inner = Self::inner_mut(s);
let (button_text, button_enable) = UI::render_button_attach(&mut inner);
button_attach.set_label(button_text);
button_attach.set_enabled(button_enable);
}
fn submit_connection_address(s: &mut Cursive) {
let edit = s.find_name::<EditView>("connection-address").unwrap();
let addr = (*edit.get_content()).clone();
let sa = match addr.parse::<std::net::SocketAddr>() {
Ok(sa) => Some(sa),
Err(_) => {
s.add_layer(Dialog::text("Invalid address").button("Close", |s| {
s.pop_layer();
}));
return;
}
};
Self::command_processor(s).set_server_address(sa);
Self::command_processor(s).start_connection();
}
fn show_connection_dialog(s: &mut Cursive, state: ConnectionState) -> bool {
let mut inner = Self::inner_mut(s);
let mut show: bool = false;
let mut hide: bool = false;
let mut reset: bool = false;
match state {
ConnectionState::Disconnected => {
if inner.connection_dialog_state == None
|| inner
.connection_dialog_state
.as_ref()
.unwrap()
.is_connected()
{
show = true;
} else if inner
.connection_dialog_state
.as_ref()
.unwrap()
.is_retrying()
{
reset = true;
}
}
ConnectionState::Connected(_, _) => {
if inner.connection_dialog_state != None
&& !inner
.connection_dialog_state
.as_ref()
.unwrap()
.is_connected()
{
hide = true;
}
}
ConnectionState::Retrying(_, _) => {
if inner.connection_dialog_state == None
|| inner
.connection_dialog_state
.as_ref()
.unwrap()
.is_connected()
{
show = true;
} else if inner
.connection_dialog_state
.as_ref()
.unwrap()
.is_disconnected()
{
reset = true;
}
}
}
inner.connection_dialog_state = Some(state);
drop(inner);
if hide {
2021-12-14 19:20:05 +00:00
s.pop_layer();
2021-11-22 16:28:30 +00:00
s.pop_layer();
return true;
}
if show {
2021-12-14 19:20:05 +00:00
s.add_fullscreen_layer(Layer::with_color(
ResizedView::with_full_screen(DummyView {}),
ColorStyle::new(PaletteColor::Background, PaletteColor::Background),
));
2021-11-22 16:28:30 +00:00
s.add_layer(
Dialog::around(
LinearLayout::vertical().child(
LinearLayout::horizontal()
.child(TextView::new("Address:"))
.child(
EditView::new()
.on_submit(|s, _| Self::submit_connection_address(s))
.with_name("connection-address")
.fixed_height(1)
.min_width(40),
),
),
)
.title("Connect to server")
.with_name("connection-dialog"),
);
return true;
}
if reset {
let mut dlg = s.find_name::<Dialog>("connection-dialog").unwrap();
dlg.clear_buttons();
return true;
}
2021-11-28 02:31:01 +00:00
false
2021-11-22 16:28:30 +00:00
}
fn refresh_connection_dialog(s: &mut Cursive) {
let new_state = Self::inner(s).ui_state.connection_state.get().clone();
if !Self::show_connection_dialog(s, new_state.clone()) {
return;
}
match new_state {
ConnectionState::Disconnected => {
let addr = match Self::command_processor(s).get_server_address() {
None => "".to_owned(),
Some(addr) => addr.to_string(),
};
debug!("address is {}", addr);
let mut edit = s.find_name::<EditView>("connection-address").unwrap();
2021-11-28 02:31:01 +00:00
edit.set_content(addr);
2021-11-22 16:28:30 +00:00
edit.set_enabled(true);
let mut dlg = s.find_name::<Dialog>("connection-dialog").unwrap();
dlg.add_button("Connect", Self::submit_connection_address);
}
2021-12-04 18:18:44 +00:00
ConnectionState::Connected(_, _) => {}
2021-11-22 16:28:30 +00:00
ConnectionState::Retrying(addr, _) => {
//
let mut edit = s.find_name::<EditView>("connection-address").unwrap();
debug!("address is {}", addr);
edit.set_content(addr.to_string());
edit.set_enabled(false);
let mut dlg = s.find_name::<Dialog>("connection-dialog").unwrap();
dlg.add_button("Cancel", |s| {
Self::command_processor(s).cancel_reconnect();
});
}
}
}
2022-11-16 17:49:53 +00:00
fn refresh_main_titlebar(s: &mut Cursive) {
let mut main_window = UI::node_events_panel(s);
let inner = Self::inner_mut(s);
main_window.set_title(format!("Node: {}", inner.ui_state.node_id.get()));
}
2021-11-22 16:28:30 +00:00
fn refresh_statusbar(s: &mut Cursive) {
let mut statusbar = UI::status_bar(s);
let mut inner = Self::inner_mut(s);
let mut status = StyledString::new();
match inner.ui_state.connection_state.get() {
ConnectionState::Disconnected => {
2021-12-04 18:18:44 +00:00
status.append_styled(
"Disconnected ".to_string(),
ColorStyle::highlight_inactive(),
);
2021-11-22 16:28:30 +00:00
status.append_styled("|", ColorStyle::highlight_inactive());
}
ConnectionState::Retrying(addr, _) => {
status.append_styled(
2022-03-13 16:45:36 +00:00
format!("Reconnecting to {} ", addr),
2021-11-22 16:28:30 +00:00
ColorStyle::highlight_inactive(),
);
status.append_styled("|", ColorStyle::highlight_inactive());
}
ConnectionState::Connected(addr, _) => {
status.append_styled(
2022-03-13 16:45:36 +00:00
format!("Connected to {} ", addr),
2021-11-22 16:28:30 +00:00
ColorStyle::highlight_inactive(),
);
status.append_styled("|", ColorStyle::highlight_inactive());
// Add attachment state
status.append_styled(
format!(" {} ", UI::render_attachment_state(&mut inner)),
ColorStyle::highlight_inactive(),
);
status.append_styled("|", ColorStyle::highlight_inactive());
// Add bandwidth status
status.append_styled(
2022-05-16 15:52:48 +00:00
format!(" {} ", UI::render_network_status(&mut inner)),
2021-11-22 16:28:30 +00:00
ColorStyle::highlight_inactive(),
);
status.append_styled("|", ColorStyle::highlight_inactive());
// Add tunnel status
status.append_styled(" No Tunnels ", ColorStyle::highlight_inactive());
status.append_styled("|", ColorStyle::highlight_inactive());
}
};
statusbar.set_content(status);
}
2022-09-06 20:49:43 +00:00
fn refresh_peers(s: &mut Cursive) {
let mut peers = UI::peers(s);
let inner = Self::inner_mut(s);
peers.set_items_stable(inner.ui_state.peers_state.get().clone());
}
2021-11-22 16:28:30 +00:00
fn update_cb(s: &mut Cursive) {
let mut inner = Self::inner_mut(s);
let mut refresh_statusbar = false;
let mut refresh_button_attach = false;
let mut refresh_connection_dialog = false;
2022-09-06 20:49:43 +00:00
let mut refresh_peers = false;
2022-11-16 17:49:53 +00:00
let mut refresh_main_titlebar = false;
2021-11-22 16:28:30 +00:00
if inner.ui_state.attachment_state.take_dirty() {
refresh_statusbar = true;
refresh_button_attach = true;
2022-09-06 20:49:43 +00:00
refresh_peers = true;
2021-11-22 16:28:30 +00:00
}
2022-05-16 15:52:48 +00:00
if inner.ui_state.network_started.take_dirty() {
refresh_statusbar = true;
}
if inner.ui_state.network_down_up.take_dirty() {
refresh_statusbar = true;
}
2021-11-22 16:28:30 +00:00
if inner.ui_state.connection_state.take_dirty() {
refresh_statusbar = true;
refresh_button_attach = true;
refresh_connection_dialog = true;
2022-09-06 20:49:43 +00:00
refresh_peers = true;
}
if inner.ui_state.peers_state.take_dirty() {
refresh_peers = true;
2021-11-22 16:28:30 +00:00
}
2022-11-16 17:49:53 +00:00
if inner.ui_state.node_id.take_dirty() {
refresh_main_titlebar = true;
}
2021-11-22 16:28:30 +00:00
drop(inner);
if refresh_statusbar {
Self::refresh_statusbar(s);
}
if refresh_button_attach {
Self::refresh_button_attach(s);
}
if refresh_connection_dialog {
Self::refresh_connection_dialog(s);
}
2022-09-06 20:49:43 +00:00
if refresh_peers {
Self::refresh_peers(s);
}
2022-11-16 17:49:53 +00:00
if refresh_main_titlebar {
Self::refresh_main_titlebar(s);
}
2021-11-22 16:28:30 +00:00
}
2021-12-11 01:14:33 +00:00
////////////////////////////////////////////////////////////////////////////
// Public functions
2023-06-08 18:07:09 +00:00
pub fn new(node_log_scrollback: usize, settings: &Settings) -> (Self, UISender) {
2021-12-11 01:14:33 +00:00
cursive_flexi_logger_view::resize(node_log_scrollback);
// Instantiate the cursive runnable
2022-07-02 15:41:25 +00:00
let runnable = CursiveRunnable::new(
2023-05-29 22:12:02 +00:00
|| -> Result<Box<dyn cursive::backend::Backend>, Box<DumbError>> {
2023-05-30 00:13:06 +00:00
#[cfg(feature = "macos")]
2023-05-29 22:12:02 +00:00
let backend = cursive::backends::curses::n::Backend::init().unwrap();
2023-05-30 00:13:06 +00:00
#[cfg(not(feature = "macos"))]
2022-07-02 15:41:25 +00:00
let backend = cursive::backends::crossterm::Backend::init().unwrap();
let buffered_backend = cursive_buffered_backend::BufferedBackend::new(backend);
Ok(Box::new(buffered_backend))
},
);
2022-06-26 21:00:05 +00:00
2021-12-11 01:14:33 +00:00
// Make the callback mechanism easily reachable
let cb_sink = runnable.cb_sink().clone();
// Create the UI object
2023-06-08 18:07:09 +00:00
let mut this = Self {
siv: runnable,
inner: Arc::new(Mutex::new(UIInner {
2021-12-11 01:14:33 +00:00
ui_state: UIState::new(),
log_colors: Default::default(),
cmdproc: None,
cmd_history: {
let mut vd = VecDeque::new();
vd.push_back("".to_string());
vd
},
cmd_history_position: 0,
cmd_history_max_size: settings.interface.command_line.history_size,
connection_dialog_state: None,
})),
};
2023-06-08 18:07:09 +00:00
let mut inner = this.inner.lock();
2021-12-11 01:14:33 +00:00
// Make the inner object accessible in callbacks easily
2023-06-08 18:07:09 +00:00
this.siv.set_user_data(this.inner.clone());
2021-12-11 01:14:33 +00:00
// Create layouts
2022-09-06 20:49:43 +00:00
2022-11-22 01:55:10 +00:00
let node_events_view = Panel::new(FlexiLoggerView::new_scrollable())
.title_position(HAlign::Left)
.title("Node Events")
.with_name("node-events-panel")
.full_screen();
2022-09-06 20:49:43 +00:00
let peers_table_view = PeersTableView::new()
2023-03-14 21:44:32 +00:00
.column(PeerTableColumn::NodeId, "Node Id", |c| c.width(48))
2022-09-06 20:49:43 +00:00
.column(PeerTableColumn::Address, "Address", |c| c)
.column(PeerTableColumn::LatencyAvg, "Ping", |c| c.width(8))
.column(PeerTableColumn::TransferDownAvg, "Down", |c| c.width(8))
.column(PeerTableColumn::TransferUpAvg, "Up", |c| c.width(8))
.with_name("peers")
.full_width()
.min_height(8);
// attempt at using Mux. Mux has bugs, like resizing problems.
// let mut mux = Mux::new();
// let node_node_events_view = mux
// .add_below(node_events_view, mux.root().build().unwrap())
// .unwrap();
// let node_peers_table_view = mux
// .add_below(peers_table_view, node_node_events_view)
// .unwrap();
// mux.set_container_split_ratio(node_peers_table_view, 0.75)
// .unwrap();
// let mut mainlayout = LinearLayout::vertical();
// mainlayout.add_child(mux);
// Back to fixed layout
let mut mainlayout = LinearLayout::vertical();
mainlayout.add_child(node_events_view);
mainlayout.add_child(peers_table_view);
// ^^^ fixed layout
2021-12-11 01:14:33 +00:00
let mut command = StyledString::new();
command.append_styled("Command> ", ColorStyle::title_primary());
//
2022-09-06 20:49:43 +00:00
mainlayout.add_child(
2021-12-11 01:14:33 +00:00
LinearLayout::horizontal()
.child(TextView::new(command))
.child(
EditView::new()
.on_submit(UI::on_command_line_entered)
.on_edit(UI::on_command_line_edit)
.on_up_down(UI::on_command_line_history)
.style(ColorStyle::new(
PaletteColor::Background,
PaletteColor::Secondary,
))
.with_name("command-line")
.full_screen()
.fixed_height(1),
)
.child(
Button::new("Attach", |s| {
UI::on_button_attach_pressed(s);
})
.with_name("button-attach"),
),
);
let mut version = StyledString::new();
version.append_styled(
concat!(" | veilid-cli v", env!("CARGO_PKG_VERSION")),
ColorStyle::highlight_inactive(),
);
2022-09-06 20:49:43 +00:00
mainlayout.add_child(
2021-12-11 01:14:33 +00:00
LinearLayout::horizontal()
.color(Some(ColorStyle::highlight_inactive()))
.child(
TextView::new("")
.with_name("status-bar")
.full_screen()
.fixed_height(1),
)
.child(TextView::new(version)),
);
2023-06-08 18:07:09 +00:00
this.siv.add_fullscreen_layer(mainlayout);
2021-12-11 01:14:33 +00:00
2023-06-08 18:07:09 +00:00
UI::setup_colors(&mut this.siv, &mut inner, settings);
UI::setup_quit_handler(&mut this.siv);
this.siv
.set_global_callback(cursive::event::Event::CtrlChar('k'), UI::clear_handler);
2021-12-11 01:14:33 +00:00
drop(inner);
2023-06-08 18:07:09 +00:00
let inner = this.inner.clone();
(this, UISender { inner, cb_sink })
2021-12-11 01:14:33 +00:00
}
pub fn cursive_flexi_logger(&self) -> Box<CursiveLogWriter> {
2023-06-08 18:07:09 +00:00
let mut flv = cursive_flexi_logger_view::cursive_flexi_logger(self.siv.cb_sink().clone());
flv.set_colors(self.inner.lock().log_colors.clone());
2021-12-11 01:14:33 +00:00
flv
}
pub fn set_command_processor(&mut self, cmdproc: CommandProcessor) {
2023-06-08 18:07:09 +00:00
let mut inner = self.inner.lock();
2021-12-11 01:14:33 +00:00
inner.cmdproc = Some(cmdproc);
}
2022-12-26 21:33:48 +00:00
2023-06-08 18:07:09 +00:00
// Note: Cursive is not re-entrant, can't borrow_mut self.siv again after this
pub async fn run_async(&mut self) {
self.siv.run_async().await;
2021-12-11 01:14:33 +00:00
}
2023-06-08 18:07:09 +00:00
// pub fn run(&mut self) {
// self.siv.run();
// }
}
2022-09-06 20:49:43 +00:00
2023-06-08 18:07:09 +00:00
#[derive(Clone)]
pub struct UISender {
inner: Arc<Mutex<UIInner>>,
cb_sink: Sender<Box<dyn FnOnce(&mut Cursive) + 'static + Send>>,
}
2021-12-11 01:14:33 +00:00
2023-06-08 18:07:09 +00:00
impl UISender {
2021-12-11 01:14:33 +00:00
pub fn display_string_dialog<T: ToString, S: ToString>(
&self,
title: T,
text: S,
close_cb: UICallback,
) {
let title = title.to_string();
let text = text.to_string();
2023-06-08 18:07:09 +00:00
let _ = self.cb_sink.send(Box::new(move |s| {
2021-12-11 01:14:33 +00:00
UI::display_string_dialog_cb(s, title, text, close_cb)
}));
}
pub fn quit(&self) {
2023-06-08 18:07:09 +00:00
let _ = self.cb_sink.send(Box::new(|s| {
2021-12-11 01:14:33 +00:00
s.quit();
}));
}
pub fn send_callback(&self, callback: UICallback) {
2023-06-08 18:07:09 +00:00
let _ = self.cb_sink.send(Box::new(move |s| callback(s)));
2021-12-11 01:14:33 +00:00
}
2023-06-08 18:07:09 +00:00
pub fn set_attachment_state(
&mut self,
state: String,
public_internet_ready: bool,
local_network_ready: bool,
) {
{
let mut inner = self.inner.lock();
inner.ui_state.attachment_state.set(state);
inner
.ui_state
.public_internet_ready
.set(public_internet_ready);
inner.ui_state.local_network_ready.set(local_network_ready);
}
2021-12-11 01:14:33 +00:00
2023-06-08 18:07:09 +00:00
let _ = self.cb_sink.send(Box::new(UI::update_cb));
}
pub fn set_network_status(
&mut self,
started: bool,
bps_down: u64,
bps_up: u64,
peers: Vec<json::JsonValue>,
) {
{
let mut inner = self.inner.lock();
inner.ui_state.network_started.set(started);
inner.ui_state.network_down_up.set((
((bps_down as f64) / 1000.0f64) as f32,
((bps_up as f64) / 1000.0f64) as f32,
));
inner.ui_state.peers_state.set(peers);
}
let _ = self.cb_sink.send(Box::new(UI::update_cb));
}
pub fn set_config(&mut self, config: &json::JsonValue) {
let mut inner = self.inner.lock();
inner
.ui_state
.node_id
.set(config["network"]["routing_table"]["node_id"].to_string());
}
pub fn set_connection_state(&mut self, state: ConnectionState) {
{
let mut inner = self.inner.lock();
inner.ui_state.connection_state.set(state);
}
let _ = self.cb_sink.send(Box::new(UI::update_cb));
}
pub fn add_node_event(&self, event: String) {
{
let inner = self.inner.lock();
let color = *inner.log_colors.get(&Level::Info).unwrap();
let mut starting_style: Style = color.into();
for line in event.lines() {
let (spanned_string, end_style) =
cursive::utils::markup::ansi::parse_with_starting_style(starting_style, line);
cursive_flexi_logger_view::push_to_log(spanned_string);
starting_style = end_style;
}
}
let _ = self.cb_sink.send(Box::new(UI::update_cb));
2021-12-11 01:14:33 +00:00
}
2021-11-22 16:28:30 +00:00
}