aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorJoe Wilm <jwilm@users.noreply.github.com>2017-05-28 17:36:55 -0700
committerGitHub <noreply@github.com>2017-05-28 17:36:55 -0700
commit13eac6b673a089afe6f15adb73d6812916c73f73 (patch)
tree6c45c2e9dd13813a5265e7329b0acc0f9ce0d4c3
parentb9ec141c927b333bfea5d8a285deedd164181dd8 (diff)
downloadalacritty-13eac6b673a089afe6f15adb73d6812916c73f73.tar.gz
alacritty-13eac6b673a089afe6f15adb73d6812916c73f73.zip
Fixes font raster for mono, gray bitmaps (#590)
As it turns out, FreeType does not always provide glyph data in LCD mode as we requested. We now correctly handle several common modes returned from FreeType including Lcd, Mono, and Gray. Note that we don't check number of grays at this time since it's 1. Almost always 256, according to FreeType docs 2. Not available in the Rust FreeType bindings being used Resolves #515 Resolves #185 Resolves #482
-rw-r--r--font/src/ft/mod.rs83
1 files changed, 71 insertions, 12 deletions
diff --git a/font/src/ft/mod.rs b/font/src/ft/mod.rs
index e1086372..c0fb5e0b 100644
--- a/font/src/ft/mod.rs
+++ b/font/src/ft/mod.rs
@@ -14,6 +14,7 @@
//
//! Rasterization powered by FreeType and FontConfig
use std::collections::HashMap;
+use std::cmp::min;
use freetype::{self, Library, Face};
@@ -205,27 +206,85 @@ impl FreeTypeRasterizer {
);
}
- let bitmap = glyph.bitmap();
- let buf = bitmap.buffer();
- let pitch = bitmap.pitch() as usize;
-
- let mut packed = Vec::with_capacity((bitmap.rows() * bitmap.width()) as usize);
- for i in 0..bitmap.rows() {
- let start = (i as usize) * pitch;
- let stop = start + bitmap.width() as usize;
- packed.extend_from_slice(&buf[start..stop]);
- }
+ let (pixel_width, buf) = Self::normalize_buffer(&glyph.bitmap())?;
Ok(RasterizedGlyph {
c: c,
top: glyph.bitmap_top(),
left: glyph.bitmap_left(),
- width: glyph.bitmap().width() / 3,
+ width: pixel_width,
height: glyph.bitmap().rows(),
- buf: packed,
+ buf: buf,
})
}
+
+ /// Given a FreeType `Bitmap`, returns packed buffer with 1 byte per LCD channel.
+ ///
+ /// The i32 value in the return type is the number of pixels per row.
+ fn normalize_buffer(bitmap: &freetype::bitmap::Bitmap) -> freetype::FtResult<(i32, Vec<u8>)> {
+ use freetype::bitmap::PixelMode;
+
+ let buf = bitmap.buffer();
+ let mut packed = Vec::with_capacity((bitmap.rows() * bitmap.width()) as usize);
+ let pitch = bitmap.pitch().abs() as usize;
+ match bitmap.pixel_mode()? {
+ PixelMode::Lcd => {
+ for i in 0..bitmap.rows() {
+ let start = (i as usize) * pitch;
+ let stop = start + bitmap.width() as usize;
+ packed.extend_from_slice(&buf[start..stop]);
+ }
+ Ok((bitmap.width() / 3, packed))
+ },
+ // Mono data is stored in a packed format using 1 bit per pixel.
+ PixelMode::Mono => {
+ fn unpack_byte(res: &mut Vec<u8>, byte: u8, mut count: u8) {
+ // Mono stores MSBit at top of byte
+ let mut bit = 7;
+ while count != 0 {
+ let value = ((byte >> bit) & 1) * 255;
+ // Push value 3x since result buffer should be 1 byte
+ // per channel
+ res.push(value);
+ res.push(value);
+ res.push(value);
+ count -= 1;
+ bit -= 1;
+ }
+ };
+
+ for i in 0..(bitmap.rows() as usize) {
+ let mut columns = bitmap.width();
+ let mut byte = 0;
+ let offset = i * bitmap.pitch().abs() as usize;
+ while columns != 0 {
+ let bits = min(8, columns);
+ unpack_byte(&mut packed, buf[offset + byte], bits as u8);
+
+ columns -= bits;
+ byte += 1;
+ }
+ }
+ Ok((bitmap.width(), packed))
+ },
+ // Gray data is stored as a value between 0 and 255 using 1 byte per pixel.
+ PixelMode::Gray => {
+ for i in 0..bitmap.rows() {
+ let start = (i as usize) * pitch;
+ let stop = start + bitmap.width() as usize;
+ for byte in &buf[start..stop] {
+ packed.push(*byte);
+ packed.push(*byte);
+ packed.push(*byte);
+ }
+ }
+ Ok((bitmap.width(), packed))
+ },
+ mode @ _ => panic!("unhandled pixel mode: {:?}", mode)
+ }
+ }
+
fn load_face_with_glyph(&mut self, glyph: char) -> Result<FontKey, Error> {
let mut charset = fc::CharSet::new();
charset.add(glyph);