use std::fmt;
use std::hash::{Hash, Hasher};
use std::rc::Rc;
use super::pairs::{self, Pairs};
use super::queueable_token::QueueableToken;
use super::token_iterator::{self, TokenIterator};
use super::super::inputs::{Input, span, Span};
use super::super::RuleType;
#[derive(Clone)]
pub struct Pair<R, I: Input> {
queue: Rc<Vec<QueueableToken<R>>>,
input: Rc<I>,
start: usize
}
pub fn new<R: RuleType, I: Input>(
queue: Rc<Vec<QueueableToken<R>>>,
input: Rc<I>,
start: usize
) -> Pair<R, I> {
Pair {
queue,
input,
start
}
}
impl<R: RuleType, I: Input> Pair<R, I> {
#[inline]
pub fn as_rule(&self) -> R {
match self.queue[self.pair()] {
QueueableToken::End { rule, .. } => rule,
_ => unreachable!()
}
}
#[inline]
pub fn as_str(&self) -> &str {
let start = self.pos(self.start);
let end = self.pos(self.pair());
unsafe { self.input.slice(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<R, I> {
let pair = self.pair();
pairs::new(
self.queue,
self.input,
self.start + 1,
pair - 1
)
}
#[inline]
pub fn tokens(self) -> TokenIterator<R, I> {
let end = self.pair();
token_iterator::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<R: RuleType, I: Input> fmt::Debug for Pair<R, I> {
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<R: RuleType, I: Input> fmt::Display for Pair<R, I> {
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<R: PartialEq, I: Input> PartialEq for Pair<R, I> {
fn eq(&self, other: &Pair<R, I>) -> bool {
Rc::ptr_eq(&self.queue, &other.queue) && Rc::ptr_eq(&self.input, &other.input) &&
self.start == other.start
}
}
impl<R: Eq, I: Input> Eq for Pair<R, I> {}
impl<R: Hash, I: Input> Hash for Pair<R, I> {
fn hash<H: Hasher>(&self, state: &mut H) {
(&*self.queue as *const Vec<QueueableToken<R>>).hash(state);
(&*self.input as *const I).hash(state);
self.start.hash(state);
}
}