aboutsummaryrefslogtreecommitdiff
path: root/src/event.rs
blob: bb5eebe0ca5d6dc2537afe0d0d5ce29862513d6b (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
//! Process window events
use std::sync::{Arc, mpsc};
use std;

use glutin;

use input;
use sync::PriorityMutex;
use term::Term;
use tty::process_should_exit;

/// The event processor
pub struct Processor<'a, W: 'a> {
    writer: &'a mut W,
    input_processor: input::Processor,
    terminal: Arc<PriorityMutex<Term>>,
    resize_tx: mpsc::Sender<(u32, u32)>,
}

impl<'a, W> Processor<'a, W>
    where W: std::io::Write
{
    /// Create a new event processor
    ///
    /// Takes a writer which is expected to be hooked up to the write end of a
    /// pty.
    pub fn new(writer: &mut W,
               terminal: Arc<PriorityMutex<Term>>,
               resize_tx: mpsc::Sender<(u32, u32)>)
               -> Processor<W>
    {
        Processor {
            writer: writer,
            terminal: terminal,
            input_processor: input::Processor::new(),
            resize_tx: resize_tx,
        }
    }

    fn handle_event(&mut self, event: glutin::Event) {
        match event {
            glutin::Event::Closed => panic!("window closed"), // TODO ...
            glutin::Event::ReceivedCharacter(c) => {
                match c {
                    // Ignore BACKSPACE and DEL. These are handled specially.
                    '\u{8}' | '\u{7f}' => (),
                    // OSX arrow keys send invalid characters; ignore.
                    '\u{f700}' | '\u{f701}' | '\u{f702}' | '\u{f703}' => (),
                    _ => {
                        let encoded = c.encode_utf8();
                        self.writer.write(encoded.as_slice()).unwrap();
                    }
                }
            },
            glutin::Event::Resized(w, h) => {
                self.resize_tx.send((w, h)).expect("send new size");
                // Acquire term lock
                let mut terminal = self.terminal.lock_high();
                terminal.dirty = true;
            },
            glutin::Event::KeyboardInput(state, _code, key, mods) => {
                // Acquire term lock
                let terminal = self.terminal.lock_high();

                self.input_processor.process(state, key, mods,
                                             &mut input::WriteNotifier(self.writer),
                                             *terminal.mode());
            },
            _ => (),
        }
    }

    /// Process at least one event and handle any additional queued events.
    pub fn process_events(&mut self, window: &glutin::Window) {
        for event in window.wait_events() {
            self.handle_event(event);
            break;
        }

        for event in window.poll_events() {
            self.handle_event(event);
        }
    }
}