use std::fmt;
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;
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> {
pub fn as_rule(&self) -> R {
match self.queue[self.pair()] {
QueueableToken::End { rule, .. } => rule,
_ => unreachable!()
}
}
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)
}
pub fn into_inner(self) -> Pairs<R, I> {
let pair = self.pair();
pairs::new(
self.queue,
self.input,
self.start + 1,
pair - 1
)
}
pub fn into_iter(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, .. } => 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: Clone, I: Input> Clone for Pair<R, I> {
fn clone(&self) -> Pair<R, I> {
Pair {
queue: self.queue.clone(),
input: self.input.clone(),
start: self.start
}
}
}