use std::fmt;
use std::hash::{Hash, Hasher};
use std::ptr;
use std::rc::Rc;
use super::pairs::{self, Pairs};
use super::queueable_token::QueueableToken;
use super::tokens::{self, Tokens};
use span::{self, Span};
use RuleType;
#[derive(Clone)]
pub struct Pair<'i, R> {
queue: Rc<Vec<QueueableToken<R>>>,
input: &'i str,
start: usize,
}
pub fn new<R: RuleType>(
queue: Rc<Vec<QueueableToken<R>>>,
input: &str,
start: usize
) -> Pair<R> {
Pair {
queue,
input,
start,
}
}
impl<'i, R: RuleType> Pair<'i, R> {
#[inline]
pub fn as_rule(&self) -> R {
match self.queue[self.pair()] {
QueueableToken::End { rule, .. } => rule,
_ => unreachable!()
}
}
#[inline]
pub fn as_str(&self) -> &'i str {
let start = self.pos(self.start);
let end = self.pos(self.pair());
unsafe { self.input.slice_unchecked(start, end) }
}
#[inline]
pub fn into_span(self) -> Span<'i> {
let start = self.pos(self.start);
let end = self.pos(self.pair());
span::new(self.input, start, end)
}
#[inline]
pub fn into_inner(self) -> Pairs<'i, R> {
let pair = self.pair();
pairs::new(
self.queue,
self.input,
self.start + 1,
pair - 1
)
}
#[inline]
pub fn tokens(self) -> Tokens<'i, R> {
let end = self.pair();
tokens::new(
self.queue,
self.input,
self.start,
end + 1
)
}
fn pair(&self) -> usize {
match self.queue[self.start] {
QueueableToken::Start { pair, .. } => pair,
_ => unreachable!()
}
}
fn pos(&self, index: usize) -> usize {
match self.queue[index] {
QueueableToken::Start { pos, .. } | QueueableToken::End { pos, .. } => pos
}
}
}
impl<'i, R: RuleType> fmt::Debug for Pair<'i, R> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "Pair {{ rule: {:?}, span: {:?}, inner: {:?} }}",
self.as_rule(), self.clone().into_span(), self.clone().into_inner())
}
}
impl<'i, R: RuleType> fmt::Display for Pair<'i, R> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
let rule = self.as_rule();
let start = self.pos(self.start);
let end = self.pos(self.pair());
let mut pairs = self.clone().into_inner().peekable();
if pairs.peek().is_none() {
write!(f, "{:?}({}, {})", rule, start, end)
} else {
write!(f, "{:?}({}, {}, [{}])", rule, start, end,
pairs.map(|pair| format!("{}", pair))
.collect::<Vec<_>>()
.join(", "))
}
}
}
impl<'i, R: PartialEq> PartialEq for Pair<'i, R> {
fn eq(&self, other: &Pair<'i, R>) -> bool {
Rc::ptr_eq(&self.queue, &other.queue) && ptr::eq(self.input, other.input) &&
self.start == other.start
}
}
impl<'i, R: Eq> Eq for Pair<'i, R> {}
impl<'i, R: Hash> Hash for Pair<'i, R> {
fn hash<H: Hasher>(&self, state: &mut H) {
(&*self.queue as *const Vec<QueueableToken<R>>).hash(state);
(self.input as *const str).hash(state);
self.start.hash(state);
}
}