use crate::error::Error;
use crate::private::Marker;
use crate::private::Sealed;
use crate::token::Token;
use crate::ParseBuffer;
use crate::ParseStream;
use std::cell::RefCell;
use std::collections::HashSet;
use either::Either;
pub struct Lookahead<'a> {
stream: ParseBuffer<'a>,
comparisons: RefCell<HashSet<String>>,
}
impl<'a> Lookahead<'a> {
pub(crate) fn new(stream: ParseBuffer<'a>) -> Lookahead<'a> {
Lookahead {
stream,
comparisons: RefCell::new(HashSet::new()),
}
}
pub fn peek<T: Peek>(&self, token: T) -> bool {
if self.stream.peek(token) {
true
} else {
self.comparisons.borrow_mut().insert(T::Token::display());
false
}
}
pub(crate) fn peek_turbofish<T: Token>(&self) -> bool {
if self.stream.peek_turbofish::<T>() {
true
} else {
self.comparisons.borrow_mut().insert(T::display());
false
}
}
pub fn error(self) -> Error {
self.stream.unexpected_token(self.comparisons.into_inner())
}
}
pub trait Peek: Sealed {
#[doc(hidden)]
type Token: Token;
}
impl<F: FnOnce(Marker) -> T, T: Token> Sealed for F {}
impl<F: FnOnce(Marker) -> T, T: Token> Peek for F {
type Token = T;
}
pub trait TokenLike {
type TokenTypes: TokenTypes;
fn peek(input: ParseStream<'_>) -> bool {
Self::TokenTypes::peek(input)
}
fn lookahead(lookahead: &Lookahead<'_>) -> bool {
Self::TokenTypes::lookahead(lookahead)
}
}
pub trait TokenTypes: Sealed {
fn peek(input: ParseStream<'_>) -> bool;
fn lookahead(lookahead: &Lookahead<'_>) -> bool;
}
impl<T: Token> TokenTypes for T {
fn peek(input: ParseStream<'_>) -> bool {
input.peek_turbofish::<T>()
}
fn lookahead(lookahead: &Lookahead<'_>) -> bool {
lookahead.peek_turbofish::<T>()
}
}
impl<L: Token, R: TokenTypes> Sealed for Either<L, R> {}
impl<L: Token, R: TokenTypes> TokenTypes for Either<L, R> {
fn peek(input: ParseStream<'_>) -> bool {
L::peek(input) || R::peek(input)
}
fn lookahead(lookahead: &Lookahead<'_>) -> bool {
L::lookahead(lookahead) || R::lookahead(lookahead)
}
}