aboutsummaryrefslogtreecommitdiff
path: root/helix-term/src/line.rs
blob: 7cbfab617bbeef96d028cdfec82aa0727b770017 (plain) (blame)
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
//! Demonstrates how to read events asynchronously with async-std.
//!
//! cargo run --features="event-stream" --example event-stream-async-std

use std::{
    io::{stdout, Write},
    time::Duration,
};

use futures::{future::FutureExt, select, StreamExt};
use smol::Timer;
// use futures_timer::Delay;

use tui::{backend::CrosstermBackend, Terminal};

use crossterm::{
    cursor::position,
    event::{DisableMouseCapture, EnableMouseCapture, Event, EventStream, KeyCode},
    execute,
    terminal::{disable_raw_mode, enable_raw_mode},
    Result,
};

const HELP: &str = r#"EventStream based on futures::Stream with async-std
 - Keyboard, mouse and terminal resize events enabled
 - Prints "." every second if there's no event
 - Hit "c" to print current cursor position
 - Use Esc to quit
"#;

async fn print_events() {
    let mut reader = EventStream::new();

    loop {
        let mut delay = Timer::after(Duration::from_millis(1_000)).fuse();
        let mut event = reader.next().fuse();

        select! {
            _ = delay => { println!(".\r"); },
            maybe_event = event => {
                match maybe_event {
                    Some(Ok(event)) => {
                        println!("Event::{:?}\r", event);

                        if event == Event::Key(KeyCode::Char('c').into()) {
                            println!("Cursor position: {:?}\r", position());
                        }

                        if event == Event::Key(KeyCode::Esc.into()) {
                            break;
                        }
                    }
                    Some(Err(e)) => println!("Error: {:?}\r", e),
                    None => break,
                }
            }
        };
    }
}

fn main() -> Result<()> {
    println!("{}", HELP);

    enable_raw_mode()?;

    let mut stdout = stdout();
    execute!(stdout, EnableMouseCapture)?;

    let backend = CrosstermBackend::new(stdout);
    let mut terminal = Terminal::new(backend)?;

    use std::thread;

    // Same number of threads as there are CPU cores.
    let num_threads = num_cpus::get().max(1);

    // A channel that sends the shutdown signal.
    let (s, r) = piper::chan::<()>(0);
    let mut threads = Vec::new();

    // Create an executor thread pool.
    for _ in 0..num_threads {
        // Spawn an executor thread that waits for the shutdown signal.
        let r = r.clone();
        threads.push(thread::spawn(move || smol::run(r.recv())));
    }

    // No need to `run()`, now we can just block on the main future.
    smol::block_on(print_events());

    // Send a shutdown signal.
    drop(s);

    // Wait for threads to finish.
    for t in threads {
        t.join().unwrap();
    }

    execute!(stdout, DisableMouseCapture)?;

    disable_raw_mode()
}