#[derive(Debug, Clone)]
pub struct Lines<'s> {
lno: usize,
rest: &'s str,
}
pub trait StrExt: AsRef<str> {
fn strip_end_counted(&self, count: usize) -> &str {
let s = self.as_ref();
&s[0..s.len().checked_sub(count).expect("stripping too much")]
}
}
impl StrExt for str {}
#[derive(Debug, Clone, amplify::Getters)]
pub struct Peeked {
#[getter(as_copy)]
line_len: usize,
}
impl<'s> Lines<'s> {
pub fn new(s: &'s str) -> Self {
Lines { lno: 1, rest: s }
}
pub fn peek_lno(&self) -> usize {
self.lno
}
pub fn peek(&self) -> Option<Peeked> {
if self.rest.is_empty() {
None
} else if let Some(newline) = self.rest.find('\n') {
Some(Peeked { line_len: newline })
} else {
Some(Peeked {
line_len: self.rest.len(),
})
}
}
pub fn remaining(&self) -> &'s str {
self.rest
}
#[allow(clippy::needless_pass_by_value)] pub fn consume_peeked(&mut self, peeked: Peeked) -> &'s str {
let line = self.peeked_line(&peeked);
self.rest = &self.rest[peeked.line_len..];
if !self.rest.is_empty() {
debug_assert!(self.rest.starts_with('\n'));
self.rest = &self.rest[1..];
}
self.lno += 1;
line
}
pub fn peeked_line(&self, peeked: &Peeked) -> &'s str {
&self.rest[0..peeked.line_len()]
}
}
impl<'s> Iterator for Lines<'s> {
type Item = &'s str;
fn next(&mut self) -> Option<&'s str> {
let peeked = self.peek()?;
let line = self.consume_peeked(peeked);
Some(line)
}
}