diff options
author | Christian Duerr <contact@christianduerr.com> | 2018-02-18 18:04:48 +0100 |
---|---|---|
committer | Christian Duerr <contact@christianduerr.com> | 2018-02-18 18:04:48 +0100 |
commit | 8d3febccd1f50ac26b6dddc9ee7bde08710841a9 (patch) | |
tree | 9ec8f6936b787c950865d88fed4bd5c912790158 | |
parent | a0270d9774184755cbf6e42457c38343c3d28fff (diff) | |
download | alacritty-8d3febccd1f50ac26b6dddc9ee7bde08710841a9.tar.gz alacritty-8d3febccd1f50ac26b6dddc9ee7bde08710841a9.zip |
Add bounds check to underline
To make sure the underline never goes beyond the cell bounds, a check
has been added.
Whenever an underline would normally be below the bottom line of the
cell, it is now placed `underline_thickness` above the bottom line. So
the position metric is ignored and the underline is rendered with the
correct thickness but right at the bottom of the cell.
-rw-r--r-- | src/display.rs | 49 |
1 files changed, 34 insertions, 15 deletions
diff --git a/src/display.rs b/src/display.rs index 24b34694..a022216f 100644 --- a/src/display.rs +++ b/src/display.rs @@ -496,10 +496,7 @@ fn calculate_cell_line_state( // If line is already started, check for end Some(start) => { // No change in line - if cell.line == start.line - && cell.flags.contains(flag) - && cell.fg == start.fg - { + if cell.line == start.line && cell.flags.contains(flag) && cell.fg == start.fg { return None; } @@ -512,10 +509,14 @@ fn calculate_cell_line_state( None }; - return Some( - cell_line_rect(&start, &last_cell.unwrap(), metrics, size_info, flag) - ); - }, + return Some(cell_line_rect( + &start, + &last_cell.unwrap(), + metrics, + size_info, + flag, + )); + } // Check for new start of line None => if cell.flags.contains(flag) { *start_cell_line = Some(cell); @@ -538,17 +539,30 @@ fn cell_line_rect( let (y, height) = match flag { cell::Flags::UNDERLINE => { - // Get the baseline positon and offset it down by (-) underline position - // then move it up by half the underline thickness - let y = ((start.line.0 as f32 + 1.) * metrics.line_height as f32 + metrics.descent + // Calculate the bottom point of the underline + let underline_bottom = metrics.line_height as f32 + metrics.descent - metrics.underline_position - - metrics.underline_thickness / 2.) - .round() as u32; + + metrics.underline_thickness / 2.; + + // Check if underline is out of bounds + let y = if underline_bottom > metrics.line_height as f32 { + // Put underline at the bottom of the cell rect + ((start.line.0 as f32 + 1.) * metrics.line_height as f32 + - metrics.underline_thickness) + .round() as u32 + } else { + // Get the baseline position and offset it down by (-) underline position + // then move it up by half the underline thickness + ((start.line.0 as f32 + 1.) * metrics.line_height as f32 + metrics.descent + - metrics.underline_position + - metrics.underline_thickness / 2.) + .round() as u32 + }; let height = metrics.underline_thickness as u32; (y, height) } cell::Flags::STRIKEOUT => { - // Get the baseline positon and offset it up by strikeout position + // Get the baseline position and offset it up by strikeout position // then move it up by half the strikeout thickness let y = ((start.line.0 as f32 + 1.) * metrics.line_height as f32 + metrics.descent - metrics.strikeout_position @@ -560,7 +574,12 @@ fn cell_line_rect( _ => panic!("Invalid flag for cell line drawing specified"), }; - let rect = Rect::new(x + size.padding_x as u32, y + size.padding_y as u32, width, height); + let rect = Rect::new( + x + size.padding_x as u32, + y + size.padding_y as u32, + width, + height, + ); let color = start.fg; (rect, color) |