use super::{VDiff, VNode, VText};
use cfg_if::cfg_if;
use std::ops::{Deref, DerefMut};
cfg_if! {
if #[cfg(feature = "std_web")] {
use stdweb::web::{Element, Node};
} else if #[cfg(feature = "web_sys")] {
use web_sys::{Element, Node};
}
}
#[derive(Clone, Debug, PartialEq, Default)]
pub struct VList {
pub children: Vec<VNode>,
elide_placeholder: bool,
}
impl Deref for VList {
type Target = Vec<VNode>;
fn deref(&self) -> &Self::Target {
&self.children
}
}
impl DerefMut for VList {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.children
}
}
impl VList {
pub fn new() -> Self {
Self::default()
}
pub fn new_with_children(children: Vec<VNode>) -> Self {
VList {
children,
elide_placeholder: false,
}
}
pub(crate) fn new_without_placeholder() -> Self {
VList {
children: Vec::new(),
elide_placeholder: true,
}
}
pub fn add_child(&mut self, child: VNode) {
self.children.push(child);
}
}
impl VDiff for VList {
fn detach(&mut self, parent: &Element) -> Option<Node> {
let mut next_sibling = None;
for mut child in self.children.drain(..) {
next_sibling = child.detach(parent);
}
next_sibling
}
fn apply(
&mut self,
parent: &Element,
previous_sibling: Option<&Node>,
ancestor: Option<VNode>,
) -> Option<Node> {
let mut previous_sibling = previous_sibling.cloned();
let mut rights = {
match ancestor {
Some(VNode::VList(vlist)) => {
vlist.children
}
Some(vnode) => {
vec![vnode]
}
None => Vec::new(),
}
};
if self.children.is_empty() && !self.elide_placeholder {
let placeholder = VText::new("".into());
self.children.push(placeholder.into());
}
let mut lefts = self.children.iter_mut();
let mut rights = rights.drain(..);
loop {
match (lefts.next(), rights.next()) {
(Some(left), Some(right)) => {
previous_sibling = left.apply(parent, previous_sibling.as_ref(), Some(right));
}
(Some(left), None) => {
previous_sibling = left.apply(parent, previous_sibling.as_ref(), None);
}
(None, Some(ref mut right)) => {
right.detach(parent);
}
(None, None) => break,
}
}
previous_sibling
}
}
#[cfg(test)]
mod tests {
use crate::{html, Component, ComponentLink, Html, ShouldRender};
#[cfg(feature = "wasm_test")]
use wasm_bindgen_test::{wasm_bindgen_test as test, wasm_bindgen_test_configure};
#[cfg(feature = "wasm_test")]
wasm_bindgen_test_configure!(run_in_browser);
struct Comp;
impl Component for Comp {
type Message = ();
type Properties = ();
fn create(_: Self::Properties, _: ComponentLink<Self>) -> Self {
Comp
}
fn update(&mut self, _: Self::Message) -> ShouldRender {
unimplemented!();
}
fn view(&self) -> Html {
unimplemented!();
}
}
#[test]
fn check_fragments() {
let fragment = html! {
<>
</>
};
html! {
<div>
{ fragment }
</div>
};
}
}