aboutsummaryrefslogtreecommitdiff
path: root/alacritty_terminal/src/renderer/rects.rs
diff options
context:
space:
mode:
Diffstat (limited to 'alacritty_terminal/src/renderer/rects.rs')
-rw-r--r--alacritty_terminal/src/renderer/rects.rs156
1 files changed, 156 insertions, 0 deletions
diff --git a/alacritty_terminal/src/renderer/rects.rs b/alacritty_terminal/src/renderer/rects.rs
new file mode 100644
index 00000000..b4f8a012
--- /dev/null
+++ b/alacritty_terminal/src/renderer/rects.rs
@@ -0,0 +1,156 @@
+// Copyright 2016 Joe Wilm, The Alacritty Project Contributors
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+use font::Metrics;
+
+use crate::index::Point;
+use crate::term::cell::Flags;
+use crate::term::color::Rgb;
+use crate::term::{RenderableCell, SizeInfo};
+
+#[derive(Debug, Copy, Clone)]
+pub struct Rect<T> {
+ pub x: T,
+ pub y: T,
+ pub width: T,
+ pub height: T,
+}
+
+impl<T> Rect<T> {
+ pub fn new(x: T, y: T, width: T, height: T) -> Self {
+ Rect { x, y, width, height }
+ }
+}
+
+#[derive(Debug)]
+struct Line {
+ flag: Flags,
+ range: Option<(RenderableCell, Point)>,
+}
+
+impl Line {
+ fn new(flag: Flags) -> Self {
+ Self { flag, range: None }
+ }
+}
+
+/// Rects for underline, strikeout and more.
+pub struct Rects<'a> {
+ inner: Vec<(Rect<f32>, Rgb)>,
+ active_lines: Vec<Line>,
+ metrics: &'a Metrics,
+ size: &'a SizeInfo,
+}
+
+impl<'a> Rects<'a> {
+ pub fn new(metrics: &'a Metrics, size: &'a SizeInfo) -> Self {
+ let active_lines = vec![Line::new(Flags::UNDERLINE), Line::new(Flags::STRIKEOUT)];
+ Self { inner: Vec::new(), active_lines, metrics, size }
+ }
+
+ /// Convert the stored rects to rectangles for the renderer.
+ pub fn rects(&self) -> &Vec<(Rect<f32>, Rgb)> {
+ &self.inner
+ }
+
+ /// Update the stored lines with the next cell info.
+ pub fn update_lines(&mut self, size_info: &SizeInfo, cell: &RenderableCell) {
+ for line in self.active_lines.iter_mut() {
+ match line.range {
+ // Check for end if line is present
+ Some((ref mut start, ref mut end)) => {
+ // No change in line
+ if cell.line == start.line
+ && cell.flags.contains(line.flag)
+ && cell.fg == start.fg
+ && cell.column == end.col + 1
+ {
+ if size_info.cols() == cell.column && size_info.lines() == cell.line {
+ // Add the last rect if we've reached the end of the terminal
+ self.inner.push(create_rect(
+ &start,
+ cell.into(),
+ line.flag,
+ &self.metrics,
+ &self.size,
+ ));
+ } else {
+ // Update the length of the line
+ *end = cell.into();
+ }
+
+ continue;
+ }
+
+ self.inner.push(create_rect(start, *end, line.flag, &self.metrics, &self.size));
+
+ // Start a new line if the flag is present
+ if cell.flags.contains(line.flag) {
+ *start = cell.clone();
+ *end = cell.into();
+ } else {
+ line.range = None;
+ }
+ },
+ // Check for new start of line
+ None => {
+ if cell.flags.contains(line.flag) {
+ line.range = Some((cell.clone(), cell.into()));
+ }
+ },
+ };
+ }
+ }
+
+ // Add a rectangle
+ pub fn push(&mut self, rect: Rect<f32>, color: Rgb) {
+ self.inner.push((rect, color));
+ }
+}
+
+/// Create a rectangle that starts on the left of `start` and ends on the right
+/// of `end`, based on the given flag and size metrics.
+fn create_rect(
+ start: &RenderableCell,
+ end: Point,
+ flag: Flags,
+ metrics: &Metrics,
+ size: &SizeInfo,
+) -> (Rect<f32>, Rgb) {
+ let start_x = start.column.0 as f32 * size.cell_width;
+ let end_x = (end.col.0 + 1) as f32 * size.cell_width;
+ let width = end_x - start_x;
+
+ let (position, mut height) = match flag {
+ Flags::UNDERLINE => (metrics.underline_position, metrics.underline_thickness),
+ Flags::STRIKEOUT => (metrics.strikeout_position, metrics.strikeout_thickness),
+ _ => unimplemented!("Invalid flag for cell line drawing specified"),
+ };
+
+ // Make sure lines are always visible
+ height = height.max(1.);
+
+ let cell_bottom = (start.line.0 as f32 + 1.) * size.cell_height;
+ let baseline = cell_bottom + metrics.descent;
+
+ let mut y = baseline - position - height / 2.;
+ let max_y = cell_bottom - height;
+ if y > max_y {
+ y = max_y;
+ }
+
+ let rect =
+ Rect::new(start_x + size.padding_x, y.round() + size.padding_y, width, height.round());
+
+ (rect, start.fg)
+}