|
|
|
@ -1,18 +1,17 @@ |
|
|
|
|
mod confirm; |
|
|
|
|
mod main_wnd; |
|
|
|
|
use async_channel::Sender; |
|
|
|
|
pub use confirm::{confirm_wnd, ConfirmWnd}; |
|
|
|
|
use crossbeam::channel::Sender; |
|
|
|
|
pub use main_wnd::MainWnd; |
|
|
|
|
|
|
|
|
|
use crate::tui::{ |
|
|
|
|
get_terminal, impls::Id, teardown, utils::Channel, Backend, Frame, GEvent, RetVal, ReturnValue, |
|
|
|
|
get_terminal, impls::Id, AsyncChannel, Backend, Frame, GEvent, RetVal, ReturnValue, |
|
|
|
|
GENERAL_EVENT_CHANNEL, |
|
|
|
|
}; |
|
|
|
|
use anyhow::Result as AResult; |
|
|
|
|
use crossterm::event::KeyCode; |
|
|
|
|
use once_cell::sync::{Lazy, OnceCell}; |
|
|
|
|
use std::collections::BTreeMap; |
|
|
|
|
use std::process::exit; |
|
|
|
|
use std::sync::Arc; |
|
|
|
|
use std::sync::{Mutex as StdMutex, MutexGuard as StdMutexGuard}; |
|
|
|
|
use tokio::sync::{Mutex, MutexGuard}; |
|
|
|
@ -24,11 +23,10 @@ static WINDOWS: Lazy<Arc<Mutex<WindowsHandler>>> = |
|
|
|
|
|
|
|
|
|
static LAST_WND_ID: OnceCell<StdMutex<WndId>> = OnceCell::new(); |
|
|
|
|
|
|
|
|
|
static WND_EVENT_CHANNEL: Lazy<Channel<WndEvent>> = Lazy::new(|| Channel::new()); |
|
|
|
|
|
|
|
|
|
pub type SharedWnd = Arc<Mutex<dyn Window>>; |
|
|
|
|
|
|
|
|
|
enum WndEvent { |
|
|
|
|
#[derive(Clone, Debug)] |
|
|
|
|
pub enum WndEvent { |
|
|
|
|
Key(KeyCode), |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -36,7 +34,7 @@ enum WndEvent { |
|
|
|
|
pub struct WndId(u64); |
|
|
|
|
|
|
|
|
|
impl WndId { |
|
|
|
|
pub fn last_wid() -> StdMutexGuard<'static, WndId> { |
|
|
|
|
fn last_wid() -> StdMutexGuard<'static, WndId> { |
|
|
|
|
LAST_WND_ID |
|
|
|
|
.get_or_init(|| StdMutex::new(WndId(0))) |
|
|
|
|
.lock() |
|
|
|
@ -54,25 +52,25 @@ impl Default for WndId { |
|
|
|
|
|
|
|
|
|
#[async_trait] |
|
|
|
|
pub trait Window: Id<WndId> + RetVal + Send { |
|
|
|
|
async fn check_wnd_events(&mut self) { |
|
|
|
|
if !WND_EVENT_CHANNEL.is_empty() { |
|
|
|
|
match WND_EVENT_CHANNEL.recv() { |
|
|
|
|
WndEvent::Key(k) => { |
|
|
|
|
self.handle_kbd(k).await.unwrap(); |
|
|
|
|
} |
|
|
|
|
async fn handle_event(&mut self, ev: WndEvent) { |
|
|
|
|
match ev { |
|
|
|
|
WndEvent::Key(k) => { |
|
|
|
|
self.handle_kbd(k).await.unwrap(); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
async fn close(&mut self, force: bool) { |
|
|
|
|
let rv = self.retval(); |
|
|
|
|
ReturnValue::set(rv).await; |
|
|
|
|
ReturnValue::set(rv); |
|
|
|
|
GENERAL_EVENT_CHANNEL.send(GEvent::CloseWnd { |
|
|
|
|
wid: self.id(), |
|
|
|
|
force, |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
fn get_chan(&self) -> &'static AsyncChannel<WndEvent>; |
|
|
|
|
|
|
|
|
|
fn into_shared(self) -> SharedWnd |
|
|
|
|
where |
|
|
|
|
Self: Sized + 'static, |
|
|
|
@ -90,30 +88,35 @@ pub trait Window: Id<WndId> + RetVal + Send { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
pub struct WndLoop { |
|
|
|
|
upd_tx: Sender<()>, |
|
|
|
|
upd_tx: Sender<WndEvent>, |
|
|
|
|
updater: JoinHandle<()>, |
|
|
|
|
window: SharedWnd, |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
impl WndLoop { |
|
|
|
|
pub fn with_wnd(window: SharedWnd) -> Self { |
|
|
|
|
let Channel { tx, rx } = Channel::new(); |
|
|
|
|
pub async fn with_wnd(window: SharedWnd) -> Self { |
|
|
|
|
let wnd = window.clone(); |
|
|
|
|
let AsyncChannel { tx, rx } = wnd.lock().await.get_chan(); |
|
|
|
|
let wnd_loop = async move { |
|
|
|
|
loop { |
|
|
|
|
rx.recv().unwrap(); |
|
|
|
|
wnd.lock().await.check_wnd_events().await; |
|
|
|
|
match rx.recv().await { |
|
|
|
|
Ok(ev) => wnd.lock().await.handle_event(ev).await, |
|
|
|
|
Err(_) => break, |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
}; |
|
|
|
|
WndLoop { |
|
|
|
|
upd_tx: tx, |
|
|
|
|
upd_tx: tx.clone(), |
|
|
|
|
window, |
|
|
|
|
updater: task::spawn(wnd_loop), |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
pub fn send_update(&self) { |
|
|
|
|
self.upd_tx.send(()).unwrap(); |
|
|
|
|
pub async fn send(&self, ev: WndEvent) { |
|
|
|
|
let wnd_id = self.window.lock().await.id(); |
|
|
|
|
let event = ev.clone(); |
|
|
|
|
debug!(?event, ?wnd_id, "sending"); |
|
|
|
|
self.upd_tx.send(ev).await.expect("big zhopa"); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -151,7 +154,8 @@ impl WindowsHandler { |
|
|
|
|
|
|
|
|
|
pub async fn push_dyn(&mut self, window: SharedWnd) -> SharedWnd { |
|
|
|
|
let wid = window.lock().await.id(); |
|
|
|
|
self.queue.insert(wid, WndLoop::with_wnd(window.clone())); |
|
|
|
|
self.queue |
|
|
|
|
.insert(wid, WndLoop::with_wnd(window.clone()).await); |
|
|
|
|
window |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -160,19 +164,12 @@ impl WindowsHandler { |
|
|
|
|
Ok(()) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
pub fn show_cursor(&mut self) -> AResult<()> { |
|
|
|
|
self.term.show_cursor()?; |
|
|
|
|
Ok(()) |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
pub async fn draw(&mut self) -> AResult<()> { |
|
|
|
|
let wids_to_redraw = if self.redraw_all { |
|
|
|
|
self.redraw_all = false; |
|
|
|
|
let mut wids = self.queue.keys().cloned().collect::<Vec<WndId>>(); |
|
|
|
|
wids.sort(); |
|
|
|
|
wids |
|
|
|
|
self.queue.keys().collect::<Vec<&WndId>>() |
|
|
|
|
} else { |
|
|
|
|
vec![*WndId::last_wid()] |
|
|
|
|
vec![self.queue.keys().last().unwrap()] |
|
|
|
|
}; |
|
|
|
|
for wid in wids_to_redraw { |
|
|
|
|
let mut wnd_locked = match self.queue.get(&wid) { |
|
|
|
@ -202,19 +199,17 @@ impl WindowsHandler { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
if self.queue.is_empty() { |
|
|
|
|
self.show_cursor().unwrap(); |
|
|
|
|
teardown().unwrap(); |
|
|
|
|
exit(0); |
|
|
|
|
GENERAL_EVENT_CHANNEL.send(GEvent::Exit); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
pub fn send_handle_kbd(&self, k: KeyCode) { |
|
|
|
|
WND_EVENT_CHANNEL.send(WndEvent::Key(k)); |
|
|
|
|
pub async fn send_handle_kbd(&self, k: KeyCode) { |
|
|
|
|
let current_wnd = self.get_last_wnd(); |
|
|
|
|
current_wnd.send(WndEvent::Key(k)).await; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
pub async fn update(&mut self) -> AResult<()> { |
|
|
|
|
let current_wnd = self.get_last_wnd(); |
|
|
|
|
current_wnd.send_update(); |
|
|
|
|
current_wnd.window.lock().await.handle_update().await?; |
|
|
|
|
Ok(()) |
|
|
|
|
} |
|
|
|
|