use std::cmp::max;
use std::cmp::min;
use std::sync::Arc;
use std::sync::Mutex;
use std::time::Duration;
use crossterm::event::Event;
use crossterm::event::KeyCode;
use crossterm::event::KeyEventKind;
use itertools::Itertools;
use neptune_cash::application::rpc::auth;
use neptune_cash::protocol::consensus::block::block_height::BlockHeight;
use neptune_cash::protocol::consensus::type_scripts::native_currency_amount::NativeCurrencyAmount;
use neptune_cash::protocol::proof_abstractions::timestamp::Timestamp;
use num_traits::Zero;
use ratatui::layout::Constraint;
use ratatui::layout::Margin;
use ratatui::style::Color;
use ratatui::style::Modifier;
use ratatui::style::Style;
use ratatui::widgets::Block;
use ratatui::widgets::Borders;
use ratatui::widgets::Cell;
use ratatui::widgets::Row;
use ratatui::widgets::StatefulWidget;
use ratatui::widgets::Table;
use ratatui::widgets::TableState;
use ratatui::widgets::Widget;
use tarpc::context;
use tokio::select;
use tokio::task::JoinHandle;
use tokio::time::sleep;
use unicode_width::UnicodeWidthStr;
use super::dashboard_app::DashboardEvent;
use super::screen::Screen;
use crate::dashboard_rpc_client::DashboardRpcClient;
type BalanceUpdate = (
BlockHeight,
Timestamp,
NativeCurrencyAmount,
NativeCurrencyAmount,
);
type BalanceUpdateArc = Arc<std::sync::Mutex<Vec<BalanceUpdate>>>;
type DashboardEventArc = Arc<std::sync::Mutex<Option<DashboardEvent>>>;
type JoinHandleArc = Arc<Mutex<JoinHandle<()>>>;
#[derive(Debug, Clone)]
struct Events {
items: BalanceUpdateArc,
state: TableState,
}
impl From<BalanceUpdateArc> for Events {
fn from(items: BalanceUpdateArc) -> Self {
Events {
items,
state: Default::default(),
}
}
}
impl Events {
const TABLE_HEADER_ROWS: usize = 3;
pub fn next(&mut self) {
let offset = Self::TABLE_HEADER_ROWS;
let i = match self.state.selected() {
Some(i) => {
if i >= self.items.lock().unwrap().len() + offset - 1 {
i } else {
i + 1
}
}
None => offset,
};
self.state.select(Some(i));
}
pub fn previous(&mut self) {
let offset = Self::TABLE_HEADER_ROWS;
let i = match self.state.selected() {
Some(i) => {
if i == offset {
i } else {
i - 1
}
}
None => offset,
};
self.state.select(Some(i));
}
}
#[derive(Debug, Clone)]
pub struct HistoryScreen {
active: bool,
fg: Color,
bg: Color,
in_focus: bool,
data: BalanceUpdateArc,
server: Arc<DashboardRpcClient>,
poll_task: Option<JoinHandleArc>,
escalatable_event: DashboardEventArc,
events: Events,
token: auth::Token,
}
impl HistoryScreen {
pub fn new(rpc_server: Arc<DashboardRpcClient>, token: auth::Token) -> Self {
let data = Arc::new(Mutex::new(vec![]));
Self {
active: false,
fg: Color::Gray,
bg: Color::Black,
in_focus: false,
data: data.clone(),
server: rpc_server,
poll_task: None,
escalatable_event: Arc::new(std::sync::Mutex::new(None)),
events: data.into(),
token,
}
}
async fn run_polling_loop(
rpc_client: Arc<DashboardRpcClient>,
token: auth::Token,
balance_updates: BalanceUpdateArc,
escalatable_event: DashboardEventArc,
) -> ! {
macro_rules! setup_poller {
($name: ident) => {
let $name = sleep(Duration::from_millis(1));
tokio::pin!($name);
};
}
macro_rules! reset_poller {
($name: ident, $period: expr) => {
$name.as_mut().reset(tokio::time::Instant::now() + $period);
};
}
setup_poller!(balance_history);
loop {
select! {
_ = &mut balance_history => {
let bh = rpc_client.history(context::current(), token).await.unwrap().unwrap();
let mut history_builder = Vec::with_capacity(bh.len());
let initial_balance = NativeCurrencyAmount::zero();
let updates = bh.iter().map(|(_,_,_, delta)| *delta);
let balances = NativeCurrencyAmount::scan_balance(&updates, initial_balance);
for ((_, block_height, timestamp, amount), balance) in bh.iter().zip(balances) {
history_builder.push((*block_height, *timestamp, *amount, balance));
}
*balance_updates.lock().unwrap() = history_builder;
*escalatable_event.lock().unwrap() = Some(DashboardEvent::RefreshScreen);
reset_poller!(balance_history, Duration::from_secs(10));
},
}
}
}
pub fn handle(&mut self, event: DashboardEvent) -> Option<DashboardEvent> {
let mut escalate_event = None;
if self.in_focus {
if let DashboardEvent::ConsoleEvent(Event::Key(key)) = event {
if key.kind == KeyEventKind::Press {
match key.code {
KeyCode::Down => self.events.next(),
KeyCode::Up => self.events.previous(),
_ => {
escalate_event = Some(event);
}
}
}
}
}
escalate_event
}
}
impl Screen for HistoryScreen {
fn activate(&mut self) {
self.active = true;
let server_arc = self.server.clone();
let token = self.token;
let data_arc = self.data.clone();
let escalatable_event_arc = self.escalatable_event.clone();
self.poll_task = Some(Arc::new(Mutex::new(tokio::spawn(async move {
HistoryScreen::run_polling_loop(server_arc, token, data_arc, escalatable_event_arc)
.await;
}))));
}
fn deactivate(&mut self) {
self.active = false;
if let Some(task_handle) = &self.poll_task {
(*task_handle.lock().unwrap()).abort();
}
}
fn focus(&mut self) {
self.fg = Color::White;
self.in_focus = true;
}
fn unfocus(&mut self) {
self.fg = Color::Gray;
self.in_focus = false;
}
fn escalatable_event(&self) -> DashboardEventArc {
self.escalatable_event.clone()
}
}
impl Widget for HistoryScreen {
fn render(mut self, area: ratatui::layout::Rect, buf: &mut ratatui::buffer::Buffer) {
let style: Style = if self.in_focus {
Style::default().fg(Color::LightCyan).bg(self.bg)
} else {
Style::default().fg(Color::Gray).bg(self.bg)
};
Block::default()
.borders(Borders::ALL)
.title("History")
.style(style)
.render(area, buf);
let mut table_canvas = area.inner(Margin {
vertical: 2,
horizontal: 2,
});
let style = Style::default().fg(self.fg).bg(self.bg);
let selected_style = style.add_modifier(Modifier::REVERSED);
let header = vec!["height", "date", " ", "amount", "balance after"];
let matrix = self
.data
.lock()
.unwrap()
.iter()
.rev()
.map(|bu| {
let (height, timestamp, amount, balance) = *bu;
vec![
height.to_string(),
timestamp.standard_format(),
if amount.is_negative() { "⭷" } else { "⭸" }.to_string(),
amount.to_string(),
balance.to_string(),
]
})
.collect_vec();
let ncols = header.len();
let mut widths: Vec<usize> = header.iter().map(|h| h.width()).collect();
for (i, w) in widths.iter_mut().enumerate() {
if let Some(body_max) = matrix.iter().map(|row| row[i].width()).max() {
*w = max(*w, body_max);
}
}
let mut widths_with_bars = vec![1];
widths_with_bars.append(
&mut widths
.iter()
.zip(vec![1; ncols].iter())
.map(|(w, o)| vec![*w, *o])
.collect_vec()
.concat(),
);
let mut header_with_bars = vec!["│".to_string()];
header_with_bars.append(
&mut header
.clone()
.into_iter()
.zip(vec!["│"; ncols].iter())
.map(|(h, &b)| vec![h.to_string(), b.to_string()])
.collect_vec()
.concat(),
);
let mut top_with_bars = widths_with_bars
.iter()
.enumerate()
.map(|(i, w)| {
if i % 2 == 0 {
"┬".to_string()
} else {
"─".to_string().repeat(*w)
}
})
.collect_vec();
top_with_bars[0] = "╭".to_string();
*top_with_bars.last_mut().unwrap() = "╮".to_string();
let mut separator_with_bars = widths_with_bars
.iter()
.enumerate()
.map(|(i, w)| {
if i % 2 == 0 {
"┼".to_string()
} else {
"─".to_string().repeat(*w)
}
})
.collect_vec();
separator_with_bars[0] = "├".to_string();
*separator_with_bars.last_mut().unwrap() = "┤".to_string();
let mut footer_with_bars = widths_with_bars
.iter()
.enumerate()
.map(|(i, w)| {
if i % 2 == 0 {
"┴".to_string()
} else {
"─".to_string().repeat(*w)
}
})
.collect_vec();
footer_with_bars[0] = "╰".to_string();
*footer_with_bars.last_mut().unwrap() = "╯".to_string();
let mut body = matrix
.iter()
.map(|row| {
let mut row_with_bars = vec!["│".to_string()];
row_with_bars.append(
&mut row
.iter()
.zip(vec!["│"; ncols].iter())
.map(|(r, &b)| vec![r.to_string(), b.to_string()])
.collect_vec()
.concat(),
);
row_with_bars
})
.map(|row| Row::new(row.iter().map(|c| Cell::from(c.to_string()))))
.collect_vec();
for (i, item) in body.iter_mut().enumerate() {
if i % 2 == 0 {
*item = item.clone().style(Style::default().bg(Color::DarkGray));
}
}
let mut rows: Vec<Row> = vec![
Row::new(top_with_bars),
Row::new(header_with_bars),
Row::new(separator_with_bars),
];
rows.append(&mut body);
rows.push(Row::new(footer_with_bars));
let width_constraints = widths_with_bars
.iter()
.map(|w| Constraint::Length(*w as u16))
.collect_vec();
let table = Table::new(rows, width_constraints)
.style(style)
.row_highlight_style(selected_style);
table_canvas.width = min(
table_canvas.width,
widths.iter().sum::<usize>() as u16 + 3 * widths.len() as u16 + 1,
);
StatefulWidget::render(table, table_canvas, buf, &mut self.events.state);
}
}