summaryrefslogtreecommitdiff
path: root/alacritty_terminal/src/term/cell.rs
diff options
context:
space:
mode:
Diffstat (limited to 'alacritty_terminal/src/term/cell.rs')
-rw-r--r--alacritty_terminal/src/term/cell.rs205
1 files changed, 205 insertions, 0 deletions
diff --git a/alacritty_terminal/src/term/cell.rs b/alacritty_terminal/src/term/cell.rs
new file mode 100644
index 00000000..4d2f4c1c
--- /dev/null
+++ b/alacritty_terminal/src/term/cell.rs
@@ -0,0 +1,205 @@
+// 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 bitflags::bitflags;
+
+use crate::ansi::{Color, NamedColor};
+use crate::grid::{self, GridCell};
+use crate::index::Column;
+
+// Maximum number of zerowidth characters which will be stored per cell.
+pub const MAX_ZEROWIDTH_CHARS: usize = 5;
+
+bitflags! {
+ #[derive(Serialize, Deserialize)]
+ pub struct Flags: u16 {
+ const INVERSE = 0b00_0000_0001;
+ const BOLD = 0b00_0000_0010;
+ const ITALIC = 0b00_0000_0100;
+ const UNDERLINE = 0b00_0000_1000;
+ const WRAPLINE = 0b00_0001_0000;
+ const WIDE_CHAR = 0b00_0010_0000;
+ const WIDE_CHAR_SPACER = 0b00_0100_0000;
+ const DIM = 0b00_1000_0000;
+ const DIM_BOLD = 0b00_1000_0010;
+ const HIDDEN = 0b01_0000_0000;
+ const STRIKEOUT = 0b10_0000_0000;
+ }
+}
+
+const fn default_extra() -> [char; MAX_ZEROWIDTH_CHARS] {
+ [' '; MAX_ZEROWIDTH_CHARS]
+}
+
+#[derive(Copy, Clone, Debug, Serialize, Deserialize, Eq, PartialEq)]
+pub struct Cell {
+ pub c: char,
+ pub fg: Color,
+ pub bg: Color,
+ pub flags: Flags,
+ #[serde(default = "default_extra")]
+ pub extra: [char; MAX_ZEROWIDTH_CHARS],
+}
+
+impl Default for Cell {
+ fn default() -> Cell {
+ Cell::new(' ', Color::Named(NamedColor::Foreground), Color::Named(NamedColor::Background))
+ }
+}
+
+impl GridCell for Cell {
+ #[inline]
+ fn is_empty(&self) -> bool {
+ (self.c == ' ' || self.c == '\t')
+ && self.extra[0] == ' '
+ && self.bg == Color::Named(NamedColor::Background)
+ && !self
+ .flags
+ .intersects(Flags::INVERSE | Flags::UNDERLINE | Flags::STRIKEOUT | Flags::WRAPLINE)
+ }
+
+ #[inline]
+ fn is_wrap(&self) -> bool {
+ self.flags.contains(Flags::WRAPLINE)
+ }
+
+ #[inline]
+ fn set_wrap(&mut self, wrap: bool) {
+ if wrap {
+ self.flags.insert(Flags::WRAPLINE);
+ } else {
+ self.flags.remove(Flags::WRAPLINE);
+ }
+ }
+}
+
+/// Get the length of occupied cells in a line
+pub trait LineLength {
+ /// Calculate the occupied line length
+ fn line_length(&self) -> Column;
+}
+
+impl LineLength for grid::Row<Cell> {
+ fn line_length(&self) -> Column {
+ let mut length = Column(0);
+
+ if self[Column(self.len() - 1)].flags.contains(Flags::WRAPLINE) {
+ return Column(self.len());
+ }
+
+ for (index, cell) in self[..].iter().rev().enumerate() {
+ if cell.c != ' ' || cell.extra[0] != ' ' {
+ length = Column(self.len() - index);
+ break;
+ }
+ }
+
+ length
+ }
+}
+
+impl Cell {
+ #[inline]
+ pub fn bold(&self) -> bool {
+ self.flags.contains(Flags::BOLD)
+ }
+
+ #[inline]
+ pub fn inverse(&self) -> bool {
+ self.flags.contains(Flags::INVERSE)
+ }
+
+ #[inline]
+ pub fn dim(&self) -> bool {
+ self.flags.contains(Flags::DIM)
+ }
+
+ pub fn new(c: char, fg: Color, bg: Color) -> Cell {
+ Cell { extra: [' '; MAX_ZEROWIDTH_CHARS], c, bg, fg, flags: Flags::empty() }
+ }
+
+ #[inline]
+ pub fn reset(&mut self, template: &Cell) {
+ // memcpy template to self
+ *self = *template;
+ }
+
+ #[inline]
+ pub fn chars(&self) -> [char; MAX_ZEROWIDTH_CHARS + 1] {
+ unsafe {
+ let mut chars = [std::mem::uninitialized(); MAX_ZEROWIDTH_CHARS + 1];
+ std::ptr::write(&mut chars[0], self.c);
+ std::ptr::copy_nonoverlapping(
+ self.extra.as_ptr(),
+ chars.as_mut_ptr().offset(1),
+ self.extra.len(),
+ );
+ chars
+ }
+ }
+
+ #[inline]
+ pub fn push_extra(&mut self, c: char) {
+ for elem in self.extra.iter_mut() {
+ if elem == &' ' {
+ *elem = c;
+ break;
+ }
+ }
+ }
+}
+
+#[cfg(test)]
+mod tests {
+ use super::{Cell, LineLength};
+
+ use crate::grid::Row;
+ use crate::index::Column;
+
+ #[test]
+ fn line_length_works() {
+ let template = Cell::default();
+ let mut row = Row::new(Column(10), &template);
+ row[Column(5)].c = 'a';
+
+ assert_eq!(row.line_length(), Column(6));
+ }
+
+ #[test]
+ fn line_length_works_with_wrapline() {
+ let template = Cell::default();
+ let mut row = Row::new(Column(10), &template);
+ row[Column(9)].flags.insert(super::Flags::WRAPLINE);
+
+ assert_eq!(row.line_length(), Column(10));
+ }
+}
+
+#[cfg(all(test, feature = "bench"))]
+mod benches {
+ extern crate test;
+ use super::Cell;
+
+ #[bench]
+ fn cell_reset(b: &mut test::Bencher) {
+ b.iter(|| {
+ let mut cell = Cell::default();
+
+ for _ in 0..100 {
+ cell.reset(test::black_box(&Cell::default()));
+ }
+
+ test::black_box(cell);
+ });
+ }
+}