Skip to content

Commit

Permalink
factor in font ligatures while rendering hints
Browse files Browse the repository at this point in the history
  • Loading branch information
purajit committed Sep 14, 2024
1 parent ae91d5f commit 8eb0d75
Show file tree
Hide file tree
Showing 6 changed files with 143 additions and 9 deletions.
95 changes: 93 additions & 2 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ clap = "2.34.0"
base64 = "0.13.1"
unicode-width = "0.1.10"
lazy_static = "1.4.0"
rustybuzz = "0.18.0"

[[bin]]
name = "thumbs"
Expand Down
Binary file added MononokiNerdFontPropo-Regular.ttf
Binary file not shown.
8 changes: 8 additions & 0 deletions src/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -136,6 +136,12 @@ fn app_args<'a>() -> clap::ArgMatches<'a> {
.short("t")
.takes_value(true),
)
.arg(
Arg::with_name("font_file")
.help("Font file to use to calculate ligatures; will skip ligature counting if not provided")
.long("font-file")
.default_value(""),
)
.get_matches()
}

Expand Down Expand Up @@ -163,6 +169,7 @@ fn main() {
let select_background_color = colors::get_color(args.value_of("select_background_color").unwrap());
let multi_foreground_color = colors::get_color(args.value_of("multi_foreground_color").unwrap());
let multi_background_color = colors::get_color(args.value_of("multi_background_color").unwrap());
let font_file = args.value_of("font_file").unwrap();

let stdin = io::stdin();
let mut handle = stdin.lock();
Expand Down Expand Up @@ -190,6 +197,7 @@ fn main() {
background_color,
hint_foreground_color,
hint_background_color,
font_file,
);

viewbox.present()
Expand Down
1 change: 1 addition & 0 deletions src/swapper.rs
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,7 @@ impl<'a> Swapper<'a> {
"select-bg-color",
"multi-fg-color",
"multi-bg-color",
"font-file",
];

if string_params.iter().any(|&x| x == name) {
Expand Down
47 changes: 40 additions & 7 deletions src/view.rs
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
use super::*;
use std::char;
use std::fs;
use std::io::{stdout, Read, Write};
use termion::async_stdin;
use termion::event::Key;
Expand All @@ -8,6 +9,7 @@ use termion::raw::IntoRawMode;
use termion::screen::AlternateScreen;
use termion::{color, cursor};

use rustybuzz::{Face, UnicodeBuffer};
use unicode_width::UnicodeWidthStr;

pub struct View<'a> {
Expand All @@ -25,6 +27,7 @@ pub struct View<'a> {
background_color: Box<dyn color::Color>,
hint_background_color: Box<dyn color::Color>,
hint_foreground_color: Box<dyn color::Color>,
font_file: &'a str,
chosen: Vec<(String, bool)>,
}

Expand All @@ -49,6 +52,7 @@ impl<'a> View<'a> {
background_color: Box<dyn color::Color>,
hint_foreground_color: Box<dyn color::Color>,
hint_background_color: Box<dyn color::Color>,
font_file: &'a str,
) -> View<'a> {
let matches = state.matches(reverse, unique);
let skip = if reverse { matches.len() - 1 } else { 0 };
Expand All @@ -68,6 +72,7 @@ impl<'a> View<'a> {
background_color,
hint_foreground_color,
hint_background_color,
font_file,
chosen: vec![],
}
}
Expand All @@ -92,6 +97,27 @@ impl<'a> View<'a> {
}
}

fn get_ligature_offset(&self, font_path: &str, text: &str) -> u32 {
let font_data = fs::read(font_path).expect("Font file not found");
let face = Face::from_slice(&font_data, 0).expect("Failed to load font face");

let mut buffer = UnicodeBuffer::new();
buffer.push_str(text);
let glyph_buffer = rustybuzz::shape(&face, &[], buffer);

let mut ligature_offset = 0;
let mut prev_cluster_index = 0;
for glyph_info in glyph_buffer.glyph_infos() {
let cluster_size = glyph_info.cluster - prev_cluster_index;
if cluster_size > 1 {
ligature_offset += cluster_size - 1;
}
prev_cluster_index = glyph_info.cluster;
}

ligature_offset
}

fn render(&self, stdout: &mut dyn Write, typed_hint: &str) -> () {
write!(stdout, "{}", cursor::Hide).unwrap();

Expand Down Expand Up @@ -126,14 +152,15 @@ impl<'a> View<'a> {
// Find long utf sequences and extract it from mat.x
let line = &self.state.lines[mat.y as usize];
let prefix = &line[0..mat.x as usize];
let ligature_offset = if self.font_file.is_empty() {0} else {self.get_ligature_offset(self.font_file, prefix)};
let extra = prefix.width_cjk() - prefix.chars().count();
let offset = (mat.x as u16) - (extra as u16);
let offset = (mat.x as u16) - (extra as u16) - (ligature_offset as u16);
let text = self.make_hint_text(mat.text);

print!(
"{goto}{background}{foregroud}{text}{resetf}{resetb}",
"{goto}{background}{foreground}{text}{resetf}{resetb}",
goto = cursor::Goto(offset + 1, mat.y as u16 + 1),
foregroud = color::Fg(&**selected_color),
foreground = color::Fg(&**selected_color),
background = color::Bg(&**selected_background_color),
resetf = color::Fg(color::Reset),
resetb = color::Bg(color::Reset),
Expand All @@ -152,9 +179,9 @@ impl<'a> View<'a> {
let final_position = std::cmp::max(offset as i16 + extra_position as i16, 0);

print!(
"{goto}{background}{foregroud}{text}{resetf}{resetb}",
"{goto}{background}{foreground}{text}{resetf}{resetb}",
goto = cursor::Goto(final_position as u16 + 1, mat.y as u16 + 1),
foregroud = color::Fg(&*self.hint_foreground_color),
foreground = color::Fg(&*self.hint_foreground_color),
background = color::Bg(&*self.hint_background_color),
resetf = color::Fg(color::Reset),
resetb = color::Bg(color::Reset),
Expand All @@ -163,9 +190,9 @@ impl<'a> View<'a> {

if hint.starts_with(typed_hint) {
print!(
"{goto}{background}{foregroud}{text}{resetf}{resetb}",
"{goto}{background}{foreground}{text}{resetf}{resetb}",
goto = cursor::Goto(final_position as u16 + 1, mat.y as u16 + 1),
foregroud = color::Fg(&*self.multi_foreground_color),
foreground = color::Fg(&*self.multi_foreground_color),
background = color::Bg(&*self.multi_background_color),
resetf = color::Fg(color::Reset),
resetb = color::Bg(color::Reset),
Expand Down Expand Up @@ -346,4 +373,10 @@ mod tests {
let result = view.make_hint_text("a");
assert_eq!(result, "[a]".to_string());
}

#[test]
fn ligatures() {
let result = View::get_ligature_offset("MononokiNerdFontPropo-Regular.ttf", &"↳ 20240913-ligatures-test".to_string());
assert_eq!(result, 2)
}
}

0 comments on commit 8eb0d75

Please sign in to comment.