use super::*;
#[derive(Debug, Clone, Default, PartialEq, Eq, PartialOrd, Ord, Hash)]
#[allow(missing_docs)]
pub struct TagAttribute<'s> {
pub key: &'s str,
pub value: &'s str,
}
#[derive(Debug, Clone, Default)]
pub struct TagAttributeIterator<'s> {
attrs: &'s str,
}
impl<'s> TagAttributeIterator<'s> {
#[inline]
#[must_use]
pub fn new(attrs: &'s str) -> Self {
Self { attrs: attrs.trim() }
}
#[inline]
#[must_use]
pub fn find_by_key(&self, key: &str) -> Option<&'s str> {
self.clone().find(|ta| ta.key == key).map(|ta| ta.value)
}
}
impl<'s> Iterator for TagAttributeIterator<'s> {
type Item = TagAttribute<'s>;
#[inline]
#[must_use]
fn next(&mut self) -> Option<Self::Item> {
debug_assert_eq!(self.attrs, self.attrs.trim());
if self.attrs.is_empty() {
return None;
}
#[allow(clippy::never_loop)]
'clear_and_return_none: loop {
let (key, rest) = match break_on_first_char(self.attrs, '=') {
Some((key, rest)) => (key, rest),
None => break 'clear_and_return_none,
};
self.attrs = rest;
let quote_marker = match self.attrs.chars().next() {
Some(q) if q == '\'' || q == '\"' => {
self.attrs = &self.attrs[1..];
q
}
_ => break 'clear_and_return_none,
};
let (value, rest) = match break_on_first_char(self.attrs, quote_marker) {
Some((key, rest)) => (key, rest),
None => break 'clear_and_return_none,
};
self.attrs = rest.trim_start();
return Some(TagAttribute { key, value });
}
self.attrs = "";
None
}
}
impl<'s> core::iter::FusedIterator for TagAttributeIterator<'s> {}