1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
use std::borrow::Cow;
use std::io::stdout;
use std::sync::Arc;
use std::time::Duration;
use app::{
App,
AppReturn
};
use eyre::Result;
use inputs::events::Events;
use inputs::InputEvent;
use io::IoEvent;
use tui::backend::CrosstermBackend;
use tui::Terminal;
use tui::layout::Rect;
use ui::ui_main;
pub mod app;
pub mod inputs;
pub mod io;
pub mod constants;
pub mod ui;
pub async fn start_ui(app: &Arc<tokio::sync::Mutex<App>>) -> Result<()> {
let stdout = stdout();
crossterm::terminal::enable_raw_mode()?;
let backend = CrosstermBackend::new(stdout);
let mut terminal = Terminal::new(backend)?;
terminal.clear()?;
terminal.hide_cursor()?;
let tick_rate = app.lock().await.config.tickrate;
let mut events = Events::new(Duration::from_millis(tick_rate));
{
let mut app = app.lock().await;
app.dispatch(IoEvent::Initialize).await;
}
loop {
let mut app = app.lock().await;
let mut states = app.state.clone();
terminal.draw(|rect| ui_main::draw(rect, &mut app, &mut states))?;
let result = match events.next().await {
InputEvent::Input(key) => app.do_action(key).await,
InputEvent::Tick => {
AppReturn::Continue
}
};
if result == AppReturn::Exit {
events.close();
break;
}
}
terminal.clear()?;
terminal.set_cursor(0, 0)?;
terminal.show_cursor()?;
crossterm::terminal::disable_raw_mode()?;
Ok(())
}
fn calculate_cursor_position(text: Vec<Cow<str>>, current_cursor_position: usize, view_box: Rect) -> (u16, u16) {
let wrapped_text_iter = text.iter();
let mut cursor_pos = current_cursor_position;
for (i, line) in wrapped_text_iter.enumerate() {
if cursor_pos <= line.len() || i == text.len() - 1 {
let x_pos = view_box.x + 1 + cursor_pos as u16;
let y_pos = view_box.y + 1 + i as u16;
let x_pos = if x_pos > i as u16 {
x_pos - i as u16
} else {
x_pos
};
return (x_pos, y_pos);
}
cursor_pos -= line.len();
}
(view_box.x + 1, view_box.y + 1)
}
fn lerp_between(color_a: (u8, u8, u8), color_b: (u8, u8, u8), time_in_ms: f32) -> (u8, u8, u8) {
let r = (color_a.0 as f32 * (1.0 - time_in_ms) + color_b.0 as f32 * time_in_ms) as u8;
let g = (color_a.1 as f32 * (1.0 - time_in_ms) + color_b.1 as f32 * time_in_ms) as u8;
let b = (color_a.2 as f32 * (1.0 - time_in_ms) + color_b.2 as f32 * time_in_ms) as u8;
(r, g, b)
}