alacritty/src/main.rs

281 lines
7.6 KiB
Rust
Raw Normal View History

//! Alacritty - The GPU Enhanced Terminal
#![feature(question_mark)]
#![feature(range_contains)]
#![feature(inclusive_range_syntax)]
#![feature(io)]
#![feature(unicode)]
extern crate fontconfig;
extern crate freetype;
extern crate libc;
extern crate glutin;
extern crate cgmath;
extern crate notify;
#[macro_use]
extern crate bitflags;
#[macro_use]
mod macros;
mod list_fonts;
mod text;
2016-02-26 04:59:21 +00:00
mod renderer;
pub mod grid;
mod meter;
mod tty;
pub mod ansi;
mod term;
2016-06-08 04:11:23 +00:00
mod util;
use std::io::{Read, Write, BufWriter};
use std::sync::Arc;
2016-06-09 15:16:44 +00:00
use std::sync::mpsc;
use grid::Grid;
use meter::Meter;
use renderer::{QuadRenderer, GlyphCache};
2016-06-09 15:16:44 +00:00
use term::Term;
use text::FontDesc;
use tty::process_should_exit;
2016-06-09 15:16:44 +00:00
use util::thread;
/// Things that the render/update thread needs to respond to
#[derive(Debug)]
enum Event {
PtyChar(char),
Glutin(glutin::Event),
}
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
enum ShouldExit {
Yes,
No
}
fn handle_event<W>(event: Event,
writer: &mut W,
terminal: &mut Term,
pty_parser: &mut ansi::Parser) -> ShouldExit
where W: Write
{
match event {
// Handle char from pty
Event::PtyChar(c) => pty_parser.advance(terminal, c),
// Handle keyboard/mouse input and other window events
Event::Glutin(gevent) => match gevent {
glutin::Event::Closed => return ShouldExit::Yes,
glutin::Event::ReceivedCharacter(c) => {
let encoded = c.encode_utf8();
writer.write(encoded.as_slice()).unwrap();
},
glutin::Event::KeyboardInput(state, _code, key) => {
match state {
glutin::ElementState::Pressed => {
match key {
Some(glutin::VirtualKeyCode::Up) => {
writer.write("\x1b[A".as_bytes()).unwrap();
},
Some(glutin::VirtualKeyCode::Down) => {
writer.write("\x1b[B".as_bytes()).unwrap();
},
Some(glutin::VirtualKeyCode::Left) => {
writer.write("\x1b[D".as_bytes()).unwrap();
},
Some(glutin::VirtualKeyCode::Right) => {
writer.write("\x1b[C".as_bytes()).unwrap();
},
_ => (),
}
},
_ => (),
}
},
_ => ()
}
}
ShouldExit::No
}
#[derive(Debug, Eq, PartialEq, Copy, Clone, Default)]
pub struct Rgb {
r: u8,
g: u8,
b: u8,
}
mod gl {
include!(concat!(env!("OUT_DIR"), "/gl_bindings.rs"));
}
#[derive(Debug)]
pub struct TermProps {
width: f32,
height: f32,
cell_width: f32,
cell_height: f32,
sep_x: f32,
sep_y: f32,
}
2016-02-21 16:15:41 +00:00
fn main() {
let window = glutin::WindowBuilder::new()
.with_title("alacritty".into())
.build()
.unwrap();
let (width, height) = window.get_inner_size_pixels().unwrap();
unsafe {
2016-02-26 05:02:18 +00:00
window.make_current().unwrap();
}
unsafe {
gl::load_with(|symbol| window.get_proc_address(symbol) as *const _);
gl::Viewport(0, 0, width as i32, height as i32);
}
2016-02-28 04:26:31 +00:00
let (dpi_x, dpi_y) = window.get_dpi().unwrap();
let dpr = window.hidpi_factor();
let font_size = 11.;
let sep_x = 2;
let sep_y = 5;
let desc = FontDesc::new("DejaVu Sans Mono", "Book");
2016-02-28 04:26:31 +00:00
let mut rasterizer = text::Rasterizer::new(dpi_x, dpi_y, dpr);
2016-02-26 04:59:21 +00:00
let (cell_width, cell_height) = rasterizer.box_size_for_font(&desc, font_size);
let num_cols = grid::num_cells_axis(cell_width, sep_x, width);
let num_rows = grid::num_cells_axis(cell_height, sep_y, height);
let tty = tty::new(num_rows as u8, num_cols as u8);
tty.resize(num_rows as usize, num_cols as usize, width as usize, height as usize);
let reader = tty.reader();
let writer = tty.writer();
println!("num_cols, num_rows = {}, {}", num_cols, num_rows);
let grid = Grid::new(num_rows as usize, num_cols as usize);
let props = TermProps {
cell_width: cell_width as f32,
sep_x: sep_x as f32,
cell_height: cell_height as f32,
sep_y: sep_y as f32,
height: height as f32,
width: width as f32,
};
let mut renderer = QuadRenderer::new(width, height);
let mut glyph_cache = GlyphCache::new(rasterizer, desc, font_size);
renderer.with_api(&props, |mut api| {
glyph_cache.init(&mut api);
});
unsafe {
gl::Enable(gl::BLEND);
gl::BlendFunc(gl::SRC1_COLOR, gl::ONE_MINUS_SRC1_COLOR);
gl::Enable(gl::MULTISAMPLE);
}
let (tx, rx) = mpsc::channel();
let reader_tx = tx.clone();
let reader_thread = thread::spawn_named("TTY Reader", move || {
for c in reader.chars() {
let c = c.unwrap();
reader_tx.send(Event::PtyChar(c)).unwrap();
}
});
let mut terminal = Term::new(tty, grid);
let mut meter = Meter::new();
let mut pty_parser = ansi::Parser::new();
let window = Arc::new(window);
let window_ref = window.clone();
let input_thread = thread::spawn_named("Input Thread", move || {
for event in window_ref.wait_events() {
tx.send(Event::Glutin(event)).unwrap();
if process_should_exit() {
break;
}
}
});
'main_loop: loop {
{
let mut writer = BufWriter::new(&writer);
// Block waiting for next event
match rx.recv() {
Ok(e) => {
let res = handle_event(e, &mut writer, &mut terminal, &mut pty_parser);
if res == ShouldExit::Yes {
break;
}
},
Err(mpsc::RecvError) => break,
}
// Handle Any events that have been queued
loop {
match rx.try_recv() {
Ok(e) => {
let res = handle_event(e, &mut writer, &mut terminal, &mut pty_parser);
if res == ShouldExit::Yes {
break;
}
},
Err(mpsc::TryRecvError::Disconnected) => break 'main_loop,
Err(mpsc::TryRecvError::Empty) => break,
}
// TODO make sure this doesn't block renders
}
}
unsafe {
gl::ClearColor(0.0, 0.0, 0.00, 1.0);
gl::Clear(gl::COLOR_BUFFER_BIT);
}
{
let _sampler = meter.sampler();
renderer.with_api(&props, |mut api| {
// Draw the grid
api.render_grid(terminal.grid(), &mut glyph_cache);
// Also draw the cursor
if !terminal.mode().contains(term::mode::TEXT_CURSOR) {
api.render_cursor(terminal.cursor(), &mut glyph_cache);
}
})
}
// Draw render timer
let timing = format!("{:.3} usec", meter.average());
let color = Rgb { r: 0xd5, g: 0x4e, b: 0x53 };
renderer.with_api(&props, |mut api| {
api.render_string(&timing[..], &mut glyph_cache, &color);
});
window.swap_buffers().unwrap();
if process_should_exit() {
break;
}
}
reader_thread.join().ok();
input_thread.join().ok();
println!("Goodbye");
}