use std::iter::Enumerate;
use std::vec::IntoIter;
use super::Node;
use crate::dom::bindings::root::DomRoot;
pub struct EvaluationCtx {
pub starting_node: DomRoot<Node>,
pub context_node: DomRoot<Node>,
pub predicate_ctx: Option<PredicateCtx>,
pub predicate_nodes: Option<Vec<DomRoot<Node>>>,
}
#[derive(Clone, Copy)]
pub struct PredicateCtx {
pub index: usize,
pub size: usize,
}
impl EvaluationCtx {
pub fn new(context_node: &Node) -> EvaluationCtx {
EvaluationCtx {
starting_node: DomRoot::from_ref(context_node),
context_node: DomRoot::from_ref(context_node),
predicate_ctx: None,
predicate_nodes: None,
}
}
pub fn subcontext_for_node(&self, node: &Node) -> EvaluationCtx {
EvaluationCtx {
starting_node: self.starting_node.clone(),
context_node: DomRoot::from_ref(node),
predicate_ctx: self.predicate_ctx,
predicate_nodes: self.predicate_nodes.clone(),
}
}
pub fn update_predicate_nodes(&self, nodes: Vec<&Node>) -> EvaluationCtx {
EvaluationCtx {
starting_node: self.starting_node.clone(),
context_node: self.context_node.clone(),
predicate_ctx: None,
predicate_nodes: Some(nodes.into_iter().map(DomRoot::from_ref).collect()),
}
}
pub fn subcontext_iter_for_nodes(&self) -> EvalNodesetIter {
let size = self.predicate_nodes.as_ref().map_or(0, |v| v.len());
EvalNodesetIter {
ctx: self,
nodes_iter: self
.predicate_nodes
.as_ref()
.map_or_else(|| Vec::new().into_iter(), |v| v.clone().into_iter())
.enumerate(),
size,
}
}
}
pub struct EvalNodesetIter<'a> {
ctx: &'a EvaluationCtx,
nodes_iter: Enumerate<IntoIter<DomRoot<Node>>>,
size: usize,
}
impl<'a> Iterator for EvalNodesetIter<'a> {
type Item = EvaluationCtx;
fn next(&mut self) -> Option<EvaluationCtx> {
self.nodes_iter.next().map(|(idx, node)| EvaluationCtx {
starting_node: self.ctx.starting_node.clone(),
context_node: node.clone(),
predicate_nodes: self.ctx.predicate_nodes.clone(),
predicate_ctx: Some(PredicateCtx {
index: idx + 1,
size: self.size,
}),
})
}
}