//! Process window events use std::sync::{Arc, mpsc}; use glutin; use input; use sync::FairMutex; use term::Term; use util::encode_char; /// The event processor pub struct Processor { notifier: N, input_processor: input::Processor, terminal: Arc>, resize_tx: mpsc::Sender<(u32, u32)>, } impl Processor { /// 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( notifier: N, terminal: Arc>, resize_tx: mpsc::Sender<(u32, u32)> ) -> Processor { Processor { notifier: notifier, 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}' => (), // Extra thing on macOS delete? '\u{f728}' => (), // OSX arrow keys send invalid characters; ignore. '\u{f700}' | '\u{f701}' | '\u{f702}' | '\u{f703}' => (), // Same with home/end. Am I missing something? Would be // nice if glutin provided the received char in // KeyboardInput event so a choice could be made there // instead of having to special case everything. '\u{f72b}' | '\u{f729}' | '\u{f72c}' | '\u{f72d}' => (), // These letters are handled in the bindings system 'v' => (), _ => { println!("printing char {:?}", c); let buf = encode_char(c); self.notifier.notify(buf); } } }, glutin::Event::Resized(w, h) => { self.resize_tx.send((w, h)).expect("send new size"); // Acquire term lock let mut terminal = self.terminal.lock(); terminal.dirty = true; }, glutin::Event::KeyboardInput(state, _code, key, mods) => { // Acquire term lock let terminal = self.terminal.lock(); let processor = &mut self.input_processor; let notifier = &mut self.notifier; processor.process_key(state, key, mods, notifier, *terminal.mode()); }, glutin::Event::MouseInput(state, button) => { let terminal = self.terminal.lock(); let processor = &mut self.input_processor; let notifier = &mut self.notifier; processor.mouse_input(state, button, notifier, *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); } } }