pub trait ToSegments {
fn to_segments(&self, separator: &str) -> Vec<String>;
fn to_parts(&self, separator: &str) -> Vec<String>;
fn to_head(&self, separator: &str) -> String;
fn to_first(&self, separator: &str) -> String;
fn to_remainder_end(&self, separator: &str) -> String;
fn to_last(&self, separator: &str) -> String;
fn to_remainder_start(&self, separator: &str) -> String;
fn to_end(&self, separator: &str) -> String;
fn to_segment(&self, separator: &str, index: i32) -> Option<String>;
fn to_inner_segment(&self, groups: &[(&str, i32)]) -> Option<String>;
fn to_tail(&self, separator: &str) -> String;
fn to_head_tail(&self, separator: &str) -> (String, String);
fn to_start_end(&self, separator: &str) -> (String, String);
}
impl ToSegments for str {
fn to_parts(&self, separator: &str) -> Vec<String> {
let splitter = self.split(separator);
splitter.into_iter().map(|s| s.to_string()).collect::<Vec<String>>()
}
fn to_segments(&self, separator: &str) -> Vec<String> {
let splitter = self.split(separator);
splitter.into_iter().map(|s| s.to_string()).filter(|s| s.len() > 0).collect::<Vec<String>>()
}
fn to_head(&self, separator: &str) -> String {
if let Some((head, _tail)) = self.split_once(separator) {
head.to_string()
} else {
self.to_owned()
}
}
fn to_last(&self, separator: &str) -> String {
let separator_len = separator.len();
if self.ends_with(separator) && self.len() > separator_len {
let end_index = self.len() - separator_len;
self[0..end_index].to_string().to_end(separator)
} else {
self.to_end(separator)
}
}
fn to_end(&self, separator: &str) -> String {
let parts = self.to_parts(separator);
if parts.len() > 0 {
parts.last().unwrap_or(&self.to_string()).to_owned()
} else {
self.to_owned()
}
}
fn to_tail(&self, separator: &str) -> String {
let parts = self.to_parts(separator);
let num_parts = parts.len();
if num_parts > 0 {
parts[1..num_parts].join(separator)
} else {
self.to_owned()
}
}
fn to_first(&self, separator: &str) -> String {
let separator_len = separator.len();
if self.starts_with(separator) && self.len() > separator_len {
self[separator_len..self.len()].to_string().to_head(separator)
} else {
self.to_head(separator)
}
}
fn to_remainder_end(&self, separator: &str) -> String {
let separator_len = separator.len();
if self.starts_with(separator) && self.len() > separator_len {
self[separator_len..].to_string().to_tail(separator)
} else {
self.to_tail(separator)
}
}
fn to_remainder_start(&self, separator: &str) -> String {
let separator_len = separator.len();
if self.ends_with(separator) && self.len() > separator_len {
let end_index = self.len() - separator_len;
self[0..end_index].to_string().to_tail(separator)
} else {
self.to_tail(separator)
}
}
fn to_segment(&self, separator: &str, index: i32) -> Option<String> {
let parts = self.to_segments(separator);
let num_parts = parts.len();
let target_index = if index >= 0 { index as usize } else { (num_parts as i32 + index) as usize };
if target_index < num_parts {
if let Some(segment) = parts.get(target_index) {
Some(segment.to_owned())
} else {
None
}
} else {
None
}
}
fn to_inner_segment(&self, groups: &[(&str, i32)]) -> Option<String> {
if groups.len() > 0 {
let mut matched: Option<String> = None;
let mut current_string = self.to_string();
for group in groups {
if current_string.len() > 0 {
let (separator, index) = group;
matched = current_string.to_segment(*separator, *index);
current_string = matched.clone().unwrap_or("".to_string());
}
}
matched
} else {
None
}
}
fn to_head_tail(&self, separator: &str) -> (String, String) {
if let Some((head, tail)) = self.split_once(separator) {
(head.to_string(), tail.to_string())
} else {
("".to_owned(), self.to_owned())
}
}
fn to_start_end(&self, separator: &str) -> (String, String) {
let parts = self.to_parts(separator);
let num_parts = parts.len();
if num_parts > 1 {
let end_index = num_parts - 1;
let start = parts[0..end_index].join(separator);
let end = self.to_end(separator);
(start, end)
} else {
(self.to_owned(), "".to_string())
}
}
}