summaryrefslogtreecommitdiffstats
path: root/src/util/event.rs
blob: b3955a6958203f51997b64381fae3afc87343be8 (plain)
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
use std::io;
use std::sync::mpsc;
use std::thread;

use termion::event::Key;
use termion::input::TermRead;

#[derive(Debug)]
pub enum Event {
    Input(Key),
    IOWorkerProgress(u64),
    IOWorkerResult(std::io::Result<u64>),
}

#[derive(Debug, Clone, Copy)]
pub struct Config {}

impl Default for Config {
    fn default() -> Config {
        Config {}
    }
}

/// A small event handler that wrap termion input and tick events. Each event
/// type is handled in its own thread and returned to a common `Receiver`
pub struct Events {
    pub event_tx: mpsc::Sender<Event>,
    event_rx: mpsc::Receiver<Event>,
    pub input_tx: mpsc::SyncSender<()>,
    // fileio_handle: thread::JoinHandle<()>,
}

impl Events {
    pub fn new() -> Self {
        Events::with_config()
    }

    pub fn with_config() -> Self {
        let (input_tx, input_rx) = mpsc::sync_channel(1);
        let (event_tx, event_rx) = mpsc::channel();

        {
            let event_tx = event_tx.clone();
            thread::spawn(move || {
                let stdin = io::stdin();
                let mut keys = stdin.keys();
                match keys.next() {
                    Some(key) => match key {
                        Ok(key) => {
                            if let Err(e) = event_tx.send(Event::Input(key)) {
                                eprintln!("Input thread send err: {:#?}", e);
                                return;
                            }
                        }
                        _ => return,
                    },
                    _ => return,
                }

                while let Ok(_) = input_rx.recv() {
                    if let Some(key) = keys.next() {
                        if let Ok(key) = key {
                            if let Err(e) = event_tx.send(Event::Input(key)) {
                                eprintln!("Input thread send err: {:#?}", e);
                                return;
                            }
                        }
                    }
                }
            })
        };

        Events {
            event_tx,
            event_rx,
            input_tx,
        }
    }

    pub fn next(&self) -> Result<Event, mpsc::RecvError> {
        let event = self.event_rx.recv()?;
        Ok(event)
    }

    pub fn flush(&self) {
        let _ = self.input_tx.send(());
    }
}