1
1
mirror of https://github.com/wez/wezterm.git synced 2024-12-29 16:42:13 +03:00
wezterm/src/font/ftwrap.rs
Wez Furlong 6aae40ac21 add hex print of the freetype error code
It makes it easier to look up the error codes from the fterrdef.h
header file.
2019-02-26 08:38:51 -08:00

179 lines
4.8 KiB
Rust

//! Higher level freetype bindings
use failure::Error;
pub use freetype::freetype::*;
use freetype::succeeded;
use std::ffi::CString;
use std::ptr;
/// Translate an error and value into a result
fn ft_result<T>(err: FT_Error, t: T) -> Result<T, Error> {
if succeeded(err) {
Ok(t)
} else {
Err(format_err!("FreeType error {:?} 0x{:x}", err, err))
}
}
pub struct Face {
pub face: FT_Face,
}
impl Drop for Face {
fn drop(&mut self) {
unsafe {
FT_Done_Face(self.face);
}
}
}
impl Face {
pub fn set_char_size(
&mut self,
char_width: FT_F26Dot6,
char_height: FT_F26Dot6,
horz_resolution: FT_UInt,
vert_resolution: FT_UInt,
) -> Result<(), Error> {
ft_result(
unsafe {
FT_Set_Char_Size(
self.face,
char_width,
char_height,
horz_resolution,
vert_resolution,
)
},
(),
)
}
#[allow(unused)]
pub fn set_pixel_sizes(&mut self, char_width: u32, char_height: u32) -> Result<(), Error> {
ft_result(
unsafe { FT_Set_Pixel_Sizes(self.face, char_width, char_height) },
(),
)
}
pub fn select_size(&mut self, idx: usize) -> Result<(), Error> {
ft_result(unsafe { FT_Select_Size(self.face, idx as i32) }, ())
}
pub fn load_codepoint(
&mut self,
codepoint: char,
) -> Result<(FT_UInt, FT_Glyph_Metrics_), Error> {
unsafe {
let glyph_pos = FT_Get_Char_Index(self.face, FT_ULong::from(u32::from(codepoint)));
let res = FT_Load_Glyph(self.face, glyph_pos, FT_LOAD_COLOR as i32);
ensure!(
succeeded(res),
"load_codepoint {}: FreeType error {:?} 0x{:x}",
codepoint,
res,
res
);
let glyph = &(*(*self.face).glyph);
Ok((glyph_pos, glyph.metrics))
}
}
pub fn load_and_render_glyph(
&mut self,
glyph_index: FT_UInt,
load_flags: FT_Int32,
render_mode: FT_Render_Mode,
) -> Result<&FT_GlyphSlotRec_, Error> {
unsafe {
let res = FT_Load_Glyph(self.face, glyph_index, load_flags);
if succeeded(res) {
let render = FT_Render_Glyph((*self.face).glyph, render_mode);
if !succeeded(render) {
bail!("FT_Render_Glyph failed: {:?}", render);
}
}
ft_result(res, &*(*self.face).glyph)
}
}
pub fn cell_metrics(&mut self) -> (f64, f64) {
unsafe {
let metrics = &(*(*self.face).size).metrics;
let height = (metrics.y_scale as f64 * f64::from((*self.face).height))
/ (f64::from(0x1_0000) * 64.0);
let mut width = 0.0;
for i in 32..128 {
let glyph_pos = FT_Get_Char_Index(self.face, i);
let res = FT_Load_Glyph(self.face, glyph_pos, FT_LOAD_COLOR as i32);
if succeeded(res) {
let glyph = &(*(*self.face).glyph);
if glyph.metrics.horiAdvance as f64 > width {
width = glyph.metrics.horiAdvance as f64;
}
}
}
(width / 64.0, height)
}
}
}
pub struct Library {
lib: FT_Library,
}
impl Drop for Library {
fn drop(&mut self) {
unsafe {
FT_Done_FreeType(self.lib);
}
}
}
impl Library {
pub fn new() -> Result<Library, Error> {
let mut lib = ptr::null_mut();
let res = unsafe { FT_Init_FreeType(&mut lib as *mut _) };
let lib = ft_result(res, lib)?;
Ok(Library { lib })
}
#[allow(dead_code)]
pub fn new_face<P>(&self, path: P, face_index: FT_Long) -> Result<Face, Error>
where
P: Into<Vec<u8>>,
{
let mut face = ptr::null_mut();
let path = CString::new(path.into())?;
let res = unsafe { FT_New_Face(self.lib, path.as_ptr(), face_index, &mut face as *mut _) };
Ok(Face {
face: ft_result(res, face)?,
})
}
pub fn new_face_from_slice(&self, data: &[u8], face_index: FT_Long) -> Result<Face, Error> {
let mut face = ptr::null_mut();
let res = unsafe {
FT_New_Memory_Face(
self.lib,
data.as_ptr(),
data.len() as _,
face_index,
&mut face as *mut _,
)
};
Ok(Face {
face: ft_result(res, face)?,
})
}
pub fn set_lcd_filter(&mut self, filter: FT_LcdFilter) -> Result<(), Error> {
unsafe { ft_result(FT_Library_SetLcdFilter(self.lib, filter), ()) }
}
}