hecto/src/row.rs

146 lines
3.5 KiB
Rust
Raw Normal View History

2021-03-08 10:43:40 -05:00
use std::cmp;
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-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),
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();
#[allow(clippy::integer_arithmetic)]
2021-03-08 14:21:24 -05:00
for grapheme in self.string[..]
.graphemes(true)
.skip(start)
.take(end - start)
{
if grapheme == "\t" {
result.push_str(" ");
} else {
result.push_str(grapheme);
}
}
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-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
2021-03-12 12:17:32 -05:00
pub fn find(&self, query: &str, after: usize) -> Option<usize> {
let substring: String = self.string[..].graphemes(true).skip(after).collect();
let matching_byte_index = substring.find(query);
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 {
2021-03-12 12:17:32 -05:00
return Some(after + grapheme_index);
2021-03-12 11:50:28 -05:00
}
}
}
None
}
2021-03-08 10:21:06 -05:00
}