hecto/src/row.rs

324 lines
9.4 KiB
Rust
Raw Normal View History

2021-03-15 13:53:30 -04:00
use crate::highlighting;
2021-03-16 10:38:26 -04:00
use crate::HighlightingOptions;
use crate::SearchDirection;
2021-03-08 10:43:40 -05:00
use std::cmp;
2021-03-12 17:46:31 -05:00
use termion::color;
2021-03-08 14:21:24 -05:00
use unicode_segmentation::UnicodeSegmentation;
2021-03-08 10:43:40 -05:00
2021-03-10 13:48:21 -05:00
#[derive(Default)]
2021-03-08 10:21:06 -05:00
pub struct Row {
string: String,
2021-03-15 13:53:30 -04:00
highlighting: Vec<highlighting::Type>,
2021-03-08 14:21:24 -05:00
len: usize,
2021-03-08 10:43:40 -05:00
}
impl From<&str> for Row {
fn from(slice: &str) -> Self {
Self {
2021-03-08 10:43:40 -05:00
string: String::from(slice),
2021-03-15 13:53:30 -04:00
highlighting: Vec::new(),
len: slice.graphemes(true).count(),
}
2021-03-08 10:43:40 -05:00
}
}
impl Row {
pub fn render(&self, start: usize, end: usize) -> String {
let end = cmp::min(end, self.string.len());
let start = cmp::min(start, end);
2021-03-08 14:21:24 -05:00
let mut result = String::new();
2021-03-15 14:37:41 -04:00
let mut current_highlighting = &highlighting::Type::None;
2021-03-08 14:21:24 -05:00
#[allow(clippy::integer_arithmetic)]
2021-03-15 14:37:41 -04:00
for (index, grapheme) in self.string[..]
2021-03-08 14:21:24 -05:00
.graphemes(true)
2021-03-15 14:37:41 -04:00
.enumerate()
2021-03-08 14:21:24 -05:00
.skip(start)
.take(end - start)
{
2021-03-12 17:46:31 -05:00
if let Some(c) = grapheme.chars().next() {
2021-03-15 14:37:41 -04:00
let highlighting_type = self
.highlighting
.get(index)
.unwrap_or(&highlighting::Type::None);
if highlighting_type != current_highlighting {
current_highlighting = highlighting_type;
let start_highlight =
format!("{}", termion::color::Fg(highlighting_type.to_color()));
result.push_str(&start_highlight[..]);
}
2021-03-12 17:46:31 -05:00
if c == '\t' {
result.push_str(" ");
} else {
result.push(c);
}
2021-03-08 14:21:24 -05:00
}
}
2021-03-15 14:37:41 -04:00
let end_highlight = format!("{}", termion::color::Fg(color::Reset));
result.push_str(&end_highlight[..]);
2021-03-08 14:21:24 -05:00
result
}
pub fn len(&self) -> usize {
self.len
}
pub fn is_empty(&self) -> bool {
self.len == 0
}
2021-03-10 13:48:21 -05:00
pub fn insert(&mut self, at: usize, c: char) {
if at >= self.len() {
self.string.push(c);
self.len += 1;
return;
}
let mut result = String::new();
let mut length = 0;
for (index, grapheme) in self.string[..].graphemes(true).enumerate() {
length += 1;
2021-03-10 13:48:21 -05:00
if index == at {
length += 1;
result.push(c);
}
2021-03-10 13:48:21 -05:00
result.push_str(grapheme);
2021-03-10 13:48:21 -05:00
}
self.len = length;
self.string = result;
2021-03-10 13:48:21 -05:00
}
#[allow(clippy::integer_arithmetic)]
2021-03-10 13:48:21 -05:00
pub fn delete(&mut self, at: usize) {
if at >= self.len() {
return;
}
2021-03-10 13:48:21 -05:00
let mut result = String::new();
let mut length = 0;
2021-03-10 13:48:21 -05:00
for (index, grapheme) in self.string[..].graphemes(true).enumerate() {
if index != at {
length += 1;
result.push_str(grapheme);
}
2021-03-10 13:48:21 -05:00
}
self.len = length;
self.string = result;
2021-03-10 13:48:21 -05:00
}
2021-03-10 14:37:58 -05:00
pub fn append(&mut self, new: &Self) {
self.string = format!("{}{}", self.string, new.string);
self.len += new.len;
2021-03-10 14:37:58 -05:00
}
pub fn split(&mut self, at: usize) -> Self {
let mut row = String::new();
let mut length = 0;
let mut splitted_row = String::new();
let mut splittend_length = 0;
for (index, grapheme) in self.string[..].graphemes(true).enumerate() {
if index < at {
length += 1;
row.push_str(grapheme);
} else {
splittend_length += 1;
splitted_row.push_str(grapheme);
}
}
2021-03-10 14:37:58 -05:00
self.string = row;
self.len = length;
2021-03-10 14:37:58 -05:00
Self {
string: splitted_row,
len: splittend_length,
2021-03-15 13:53:30 -04:00
highlighting: Vec::new(),
}
2021-03-10 14:37:58 -05:00
}
2021-03-10 14:51:11 -05:00
pub fn as_bytes(&self) -> &[u8] {
self.string.as_bytes()
}
2021-03-12 11:50:28 -05:00
pub fn find(&self, query: &str, at: usize, direction: SearchDirection) -> Option<usize> {
2021-03-15 14:53:08 -04:00
if at > self.len || query.is_empty() {
return None;
}
let (start, end) = match direction {
SearchDirection::Forward => (at, self.len),
SearchDirection::Backward => (0, at),
};
let substring: String = self.string[..]
.graphemes(true)
.skip(start)
.take(end - start)
.collect();
let matching_byte_index = if direction == SearchDirection::Forward {
substring.find(query)
} else {
substring.rfind(query)
};
2021-03-12 12:17:32 -05:00
2021-03-12 11:50:28 -05:00
if let Some(matching_byte_index) = matching_byte_index {
for (grapheme_index, (byte_index, _)) in
2021-03-12 12:17:32 -05:00
substring[..].grapheme_indices(true).enumerate()
2021-03-12 11:50:28 -05:00
{
if matching_byte_index == byte_index {
return Some(start + grapheme_index);
2021-03-12 11:50:28 -05:00
}
}
}
None
}
2021-03-15 13:53:30 -04:00
2021-03-16 10:38:26 -04:00
pub fn highlight(&mut self, opts: HighlightingOptions, word: Option<&str>) {
2021-03-15 13:53:30 -04:00
let mut highlighting = Vec::new();
2021-03-15 14:53:08 -04:00
let chars: Vec<char> = self.string.chars().collect();
let mut matches = Vec::new();
let mut search_index = 0;
if let Some(word) = word {
while let Some(search_match) = self.find(word, search_index, SearchDirection::Forward) {
matches.push(search_match);
if let Some(next_index) = search_match.checked_add(word[..].graphemes(true).count())
{
search_index = next_index
} else {
break;
}
}
}
2021-03-16 09:28:07 -04:00
let mut prev_is_separator = true;
2021-03-16 10:45:19 -04:00
let mut in_string = false;
2021-03-15 14:53:08 -04:00
let mut index = 0;
while let Some(c) = chars.get(index) {
if let Some(word) = word {
if matches.contains(&index) {
for _ in word[..].graphemes(true) {
index += 1;
highlighting.push(highlighting::Type::Match);
}
continue;
}
}
2021-03-15 13:53:30 -04:00
2021-03-16 09:28:07 -04:00
let previous_highlight = if index > 0 {
highlighting
.get(index - 1)
.unwrap_or(&highlighting::Type::None)
} else {
&highlighting::Type::None
};
2021-03-16 10:57:15 -04:00
if opts.characters() && !in_string && *c == '\'' {
prev_is_separator = true;
if let Some(next_char) = chars.get(index.saturating_add(1)) {
let closing_index = if *next_char == '\\' {
index.saturating_add(3)
} else {
index.saturating_add(2)
};
if let Some(closing_char) = chars.get(closing_index) {
if *closing_char == '\'' {
for _ in 0..=closing_index.saturating_sub(index) {
highlighting.push(highlighting::Type::Character);
index += 1;
}
continue;
}
}
}
highlighting.push(highlighting::Type::None);
index += 1;
continue;
}
2021-03-16 10:45:19 -04:00
if opts.strings() {
if in_string {
highlighting.push(highlighting::Type::String);
// Don't let an escaped character stop string highlighting
if *c == '\\' && index < self.len().saturating_sub(1) {
highlighting.push(highlighting::Type::String);
index += 2;
continue;
}
2021-03-16 10:45:19 -04:00
if *c == '"' {
in_string = false;
prev_is_separator = true;
} else {
prev_is_separator = false;
}
index += 1;
continue;
} else if prev_is_separator && *c == '"' {
highlighting.push(highlighting::Type::String);
in_string = true;
prev_is_separator = true;
index += 1;
continue;
}
}
2021-03-16 11:02:27 -04:00
if opts.comments() && *c == '/' {
if let Some(next_char) = chars.get(index.saturating_add(1)) {
if *next_char == '/' {
for _ in index..chars.len() {
highlighting.push(highlighting::Type::Comment);
}
break;
}
};
}
2021-03-16 10:38:26 -04:00
if opts.numbers() {
if (c.is_ascii_digit()
&& (prev_is_separator || *previous_highlight == highlighting::Type::Number))
|| (*c == '.' && *previous_highlight == highlighting::Type::Number)
2021-03-16 10:38:26 -04:00
{
highlighting.push(highlighting::Type::Number)
} else {
highlighting.push(highlighting::Type::None)
}
2021-03-15 13:53:30 -04:00
} else {
highlighting.push(highlighting::Type::None)
}
2021-03-15 14:53:08 -04:00
2021-03-16 09:28:07 -04:00
prev_is_separator = c.is_ascii_punctuation() || c.is_ascii_whitespace();
2021-03-15 14:53:08 -04:00
index += 1;
2021-03-15 13:53:30 -04:00
}
self.highlighting = highlighting;
}
}