use crate::*;
impl Renderer {
pub fn render(&mut self, vnode: VirtualNode) {
let new_unwrapped: VirtualNode = self.unwrap_component(&vnode);
let old_tree: Option<VirtualNode> = self.try_get_current_tree().clone();
if let Some(old_vnode) = old_tree {
self.patch_root(&old_vnode, &new_unwrapped);
} else {
while let Some(child) = self.get_root().first_child() {
if let Some(element) = child.dyn_ref::<Element>() {
self.cleanup_dom_subtree(element);
}
self.get_root().remove_child(&child).unwrap();
}
let dom_node: Node = self.create_dom_node(&new_unwrapped);
self.get_root().append_child(&dom_node).unwrap();
}
self.set_current_tree(Some(new_unwrapped));
}
pub fn render_full_replace(&mut self, vnode: VirtualNode) {
let new_unwrapped: VirtualNode = self.unwrap_component(&vnode);
while let Some(child) = self.get_root().first_child() {
if let Some(element) = child.dyn_ref::<Element>() {
self.cleanup_dom_subtree(element);
}
self.get_root().remove_child(&child).unwrap();
}
let dom_node: Node = self.create_dom_node(&new_unwrapped);
self.get_root().append_child(&dom_node).unwrap();
self.set_current_tree(Some(new_unwrapped));
}
fn patch_root(&mut self, old_node: &VirtualNode, new_node: &VirtualNode) {
let dom_child: Option<Node> = self.get_root().first_child();
let is_element: bool = if let Some(ref dom_child) = dom_child {
dom_child.dyn_ref::<Element>().is_some()
} else {
false
};
if is_element {
let element: Element = dom_child.unwrap().dyn_into::<Element>().unwrap();
self.patch_node(old_node, new_node, &element);
} else if let Some(dom_child) = dom_child {
if let Some(element) = dom_child.dyn_ref::<Element>() {
self.cleanup_dom_subtree(element);
}
let new_dom_node: Node = self.create_dom_node(new_node);
self.get_root()
.replace_child(&new_dom_node, &dom_child)
.unwrap();
} else {
let new_dom_node: Node = self.create_dom_node(new_node);
self.get_root().append_child(&new_dom_node).unwrap();
}
}
fn patch_node(
&mut self,
old_node: &VirtualNode,
new_node: &VirtualNode,
dom_element: &Element,
) {
match (old_node, new_node) {
(VirtualNode::Text(old_text), VirtualNode::Text(new_text)) => {
if old_text != new_text {
dom_element.set_text_content(Some(new_text.get_content()));
}
}
(
VirtualNode::Element {
tag: old_tag,
attributes: old_attrs,
children: old_children,
key: _old_key,
},
VirtualNode::Element {
tag: new_tag,
attributes: new_attrs,
children: new_children,
key: _new_key,
},
) => {
if old_tag != new_tag {
let new_dom_node: Node = self.create_dom_node(new_node);
if let Some(parent) = dom_element.parent_node() {
self.cleanup_dom_subtree(dom_element);
parent.replace_child(&new_dom_node, dom_element).unwrap();
}
return;
}
self.patch_attributes(dom_element, old_attrs, new_attrs);
self.patch_children(dom_element, old_children, new_children);
}
(VirtualNode::Fragment(old_children), VirtualNode::Fragment(new_children)) => {
self.patch_children(dom_element, old_children, new_children);
}
(VirtualNode::Dynamic(old_dynamic), VirtualNode::Dynamic(new_dynamic)) => {
let old_rendered: VirtualNode = old_dynamic.render();
let new_rendered: VirtualNode = new_dynamic.render();
let new_unwrapped: VirtualNode = self.unwrap_component(&new_rendered);
if !Self::visual_eq(&old_rendered, &new_unwrapped) {
while let Some(child) = dom_element.first_child() {
if let Some(child_element) = child.dyn_ref::<Element>() {
self.cleanup_dom_subtree(child_element);
}
let _ = dom_element.remove_child(&child);
}
if let Some(dynamic_id_str) = dom_element.get_attribute(DATA_EUV_DYNAMIC_ID)
&& let Ok(dynamic_id) = dynamic_id_str.parse::<usize>()
{
cleanup_dynamic_node(dynamic_id);
}
let dynamic_id: usize = Self::assign_dynamic_id(dom_element);
let initial_dom: Node =
self.setup_dynamic_node(new_dynamic, dynamic_id, dom_element, true);
let _ = dom_element.append_child(&initial_dom);
}
}
(VirtualNode::Dynamic(_), _) => {
let new_dom_node: Node = self.create_dom_node(new_node);
if let Some(parent) = dom_element.parent_node() {
self.cleanup_dom_subtree(dom_element);
parent.replace_child(&new_dom_node, dom_element).unwrap();
}
}
(_, VirtualNode::Dynamic(_)) => {
let new_dom_node: Node = self.create_dom_node(new_node);
if let Some(parent) = dom_element.parent_node() {
self.cleanup_dom_subtree(dom_element);
parent.replace_child(&new_dom_node, dom_element).unwrap();
}
}
_ => {
let new_dom_node: Node = self.create_dom_node(new_node);
if let Some(parent) = dom_element.parent_node() {
self.cleanup_dom_subtree(dom_element);
parent.replace_child(&new_dom_node, dom_element).unwrap();
}
}
}
}
fn patch_attributes(
&mut self,
element: &Element,
old_attrs: &[AttributeEntry],
new_attrs: &[AttributeEntry],
) {
let old_map: HashMap<&str, &AttributeValue> = old_attrs
.iter()
.map(|attr: &AttributeEntry| (attr.get_name().as_str(), attr.get_value()))
.collect();
let new_map: HashMap<&str, ()> = new_attrs
.iter()
.map(|attr: &AttributeEntry| (attr.get_name().as_str(), ()))
.collect();
for old_attr in old_attrs {
if !new_map.contains_key(old_attr.get_name().as_str()) {
if let AttributeValue::Event(_) = old_attr.get_value()
&& let Some(euv_id_str) = element.get_attribute(DATA_EUV_ID)
&& let Ok(euv_id) = euv_id_str.parse::<usize>()
{
cleanup_event_handler(euv_id, old_attr.get_name());
}
remove_dom_attribute_or_property(element, old_attr.get_name());
}
}
for new_attr in new_attrs {
let old_value: Option<&AttributeValue> =
old_map.get(new_attr.get_name().as_str()).copied();
let should_set: bool = match old_value {
Some(old_val) => old_val != new_attr.get_value(),
None => true,
};
if should_set {
match new_attr.get_value() {
AttributeValue::Text(value) => {
if value.is_empty() {
remove_dom_attribute_or_property(element, new_attr.get_name());
} else {
set_dom_attribute_or_property(element, new_attr.get_name(), value);
}
}
AttributeValue::Signal(signal) => {
let value: String = signal.get();
if value.is_empty() && !is_boolean_property(new_attr.get_name()) {
remove_dom_attribute_or_property(element, new_attr.get_name());
} else {
set_dom_attribute_or_property(element, new_attr.get_name(), &value);
}
}
AttributeValue::Event(handler) => {
self.attach_event_listener(element, handler);
}
AttributeValue::Dynamic(_) => {}
AttributeValue::Css(css) => {
css.inject_style();
set_dom_attribute_or_property(element, new_attr.get_name(), css.get_name());
}
}
}
}
}
fn get_child_node(parent: &Element, index: u32) -> Option<Node> {
parent.child_nodes().get(index)
}
fn patch_children(
&mut self,
parent: &Element,
old_children: &[VirtualNode],
new_children: &[VirtualNode],
) {
let old_has_keys: bool =
!old_children.is_empty() && old_children.iter().all(Self::node_has_key);
let new_has_keys: bool =
!new_children.is_empty() && new_children.iter().all(Self::node_has_key);
if old_has_keys && new_has_keys {
self.patch_children_keyed(parent, old_children, new_children);
} else {
self.patch_children_positional(parent, old_children, new_children);
}
}
fn node_has_key(node: &VirtualNode) -> bool {
match node {
VirtualNode::Element { key, .. } => key.is_some(),
_ => false,
}
}
fn get_node_key(node: &VirtualNode) -> Option<&str> {
match node {
VirtualNode::Element { key, .. } => key.as_deref(),
_ => None,
}
}
fn patch_children_keyed(
&mut self,
parent: &Element,
old_children: &[VirtualNode],
new_children: &[VirtualNode],
) {
let mut old_key_map: HashMap<&str, usize> = HashMap::with_capacity(old_children.len());
for (index, old_child) in old_children.iter().enumerate() {
if let Some(key) = Self::get_node_key(old_child) {
old_key_map.insert(key, index);
}
}
let mut reused_indices: HashSet<usize> = HashSet::with_capacity(new_children.len());
let child_nodes: NodeList = parent.child_nodes();
let dom_child_count: u32 = child_nodes.length();
for (new_index, new_child) in new_children.iter().enumerate() {
let new_key: &str = Self::get_node_key(new_child).unwrap_or_default();
if let Some(&old_index) = old_key_map.get(new_key) {
reused_indices.insert(old_index);
let old_child: &VirtualNode = &old_children[old_index];
let mapped_dom_index: u32 = old_index as u32;
if mapped_dom_index < dom_child_count
&& let Some(dom_node) = child_nodes.get(mapped_dom_index)
&& let Some(element) = dom_node.dyn_ref::<Element>()
{
self.patch_node(old_child, new_child, element);
}
let current_dom_index: u32 = new_index as u32;
if mapped_dom_index != current_dom_index
&& current_dom_index < dom_child_count
&& let Some(dom_node) = child_nodes.get(mapped_dom_index)
{
if let Some(reference_node) = child_nodes.get(current_dom_index) {
let _ = parent.insert_before(&dom_node, Some(&reference_node));
} else {
let _ = parent.append_child(&dom_node);
}
}
} else {
let new_dom_node: Node = self.create_dom_node(new_child);
if (new_index as u32) < dom_child_count
&& let Some(reference_node) = child_nodes.get(new_index as u32)
{
let _ = parent.insert_before(&new_dom_node, Some(&reference_node));
} else {
let _ = parent.append_child(&new_dom_node);
}
}
}
let mut indices_to_remove: Vec<usize> = Vec::new();
for (index, old_child) in old_children.iter().enumerate() {
if !reused_indices.contains(&index) {
indices_to_remove.push(index);
}
let _ = old_child;
}
indices_to_remove.sort_unstable_by(|left_index: &usize, right_index: &usize| {
right_index.cmp(left_index)
});
for old_index in indices_to_remove {
let mapped_dom_index: u32 = old_index as u32;
if mapped_dom_index < parent.child_nodes().length()
&& let Some(dom_node) = parent.child_nodes().get(mapped_dom_index)
{
if let Some(element) = dom_node.dyn_ref::<Element>() {
self.cleanup_dom_subtree(element);
}
let _ = parent.remove_child(&dom_node);
}
}
}
fn patch_children_positional(
&mut self,
parent: &Element,
old_children: &[VirtualNode],
new_children: &[VirtualNode],
) {
let old_len: usize = old_children.len();
let new_len: usize = new_children.len();
let common_len: usize = old_len.min(new_len);
for index in 0..common_len {
let old_child: &VirtualNode = &old_children[index];
let new_child: &VirtualNode = &new_children[index];
if let Some(dom_child) = Self::get_child_node(parent, index as u32) {
if let Some(element) = dom_child.dyn_ref::<Element>() {
self.patch_node(old_child, new_child, element);
} else if let (VirtualNode::Text(old_text), VirtualNode::Text(new_text)) =
(old_child, new_child)
{
if old_text != new_text {
dom_child.set_text_content(Some(new_text.get_content()));
}
} else {
let new_dom_node: Node = self.create_dom_node(new_child);
if let Some(parent_node) = dom_child.parent_node() {
if let Some(child_element) = dom_child.dyn_ref::<Element>() {
self.cleanup_dom_subtree(child_element);
}
let _ = parent_node.replace_child(&new_dom_node, &dom_child);
}
}
}
}
if new_len > old_len {
for new_child in new_children.iter().skip(common_len) {
let new_dom_node: Node = self.create_dom_node(new_child);
parent.append_child(&new_dom_node).unwrap();
}
} else if old_len > new_len {
for _ in common_len..old_len {
if let Some(last_child) = parent.last_child()
&& let Some(element) = last_child.dyn_ref::<Element>()
{
self.cleanup_dom_subtree(element);
}
if let Some(last_child) = parent.last_child() {
parent.remove_child(&last_child).unwrap();
}
}
}
}
fn create_dom_node(&mut self, node: &VirtualNode) -> Node {
let document: Document = window().unwrap().document().unwrap();
self.create_dom_node_with_document(node, &document)
}
fn create_dom_node_with_document(&mut self, node: &VirtualNode, document: &Document) -> Node {
match node {
VirtualNode::Element {
tag,
attributes,
children,
..
} => {
let element: Element = match tag {
Tag::Element(name) => document.create_element(name).unwrap(),
Tag::Component(_) => {
let unwrapped: VirtualNode = self.unwrap_component(node);
return self.create_dom_node_with_document(&unwrapped, document);
}
};
for attr in attributes {
match attr.get_value() {
AttributeValue::Text(value) => {
if !value.is_empty() || is_boolean_property(attr.get_name()) {
set_dom_attribute_or_property(&element, attr.get_name(), value);
}
}
AttributeValue::Signal(signal) => {
let initial_value: String = signal.get();
if !initial_value.is_empty() || is_boolean_property(attr.get_name()) {
set_dom_attribute_or_property(
&element,
attr.get_name(),
&initial_value,
);
}
let signal_addr: usize = signal.get_inner_addr();
let existing_addrs: String = element
.get_attribute(DATA_EUV_SIGNAL_ADDRS)
.unwrap_or_default();
let updated_addrs: String = if existing_addrs.is_empty() {
signal_addr.to_string()
} else {
format!("{existing_addrs}{SIGNAL_ADDRS_SEPARATOR}{signal_addr}")
};
let _ = element.set_attribute(DATA_EUV_SIGNAL_ADDRS, &updated_addrs);
let attr_name: String = attr.get_name().clone();
let element_clone: Element = element.clone();
let signal_for_sub: Signal<String> = *signal;
let sub_signal: Signal<String> = signal_for_sub;
signal_for_sub.replace_subscribe(move || {
if !is_node_connected(&element_clone) {
sub_signal.clear_listeners();
return;
}
let new_value: String = sub_signal.get();
if new_value.is_empty() && !is_boolean_property(&attr_name) {
remove_dom_attribute_or_property(&element_clone, &attr_name);
} else {
set_dom_attribute_or_property(
&element_clone,
&attr_name,
&new_value,
);
}
});
}
AttributeValue::Event(handler) => {
self.attach_event_listener(&element, handler);
}
AttributeValue::Dynamic(_) => {}
AttributeValue::Css(css) => {
css.inject_style();
set_dom_attribute_or_property(
&element,
attr.get_name(),
css.get_name(),
);
}
}
}
for child in children {
let child_node: Node = self.create_dom_node_with_document(child, document);
element.append_child(&child_node).unwrap();
if let VirtualNode::Text(text_node) = child
&& let Some(signal) = text_node.try_get_signal()
{
let signal_addr: usize = signal.get_inner_addr();
let existing_addrs: String = element
.get_attribute(DATA_EUV_SIGNAL_ADDRS)
.unwrap_or_default();
let updated_addrs: String = if existing_addrs.is_empty() {
signal_addr.to_string()
} else {
format!("{existing_addrs}{SIGNAL_ADDRS_SEPARATOR}{signal_addr}")
};
let _ = element.set_attribute(DATA_EUV_SIGNAL_ADDRS, &updated_addrs);
}
}
element.into()
}
VirtualNode::Text(text_node) => {
let text: Text = document.create_text_node(text_node.get_content());
if let Some(signal) = text_node.try_get_signal() {
let text_clone: Text = text.clone();
let signal_clone: Signal<String> = *signal;
let sub_signal: Signal<String> = signal_clone;
signal_clone.replace_subscribe({
move || {
if !is_node_connected(&text_clone) {
sub_signal.clear_listeners();
return;
}
let new_value: String = sub_signal.get();
text_clone.set_text_content(Some(&new_value));
}
});
}
text.into()
}
VirtualNode::Fragment(children) => {
let fragment: Element = document.create_element(FRAGMENT_TAG).unwrap();
let _ = fragment.set_attribute(ATTR_STYLE, FRAGMENT_STYLE);
for child in children {
let child_node: Node = self.create_dom_node_with_document(child, document);
fragment.append_child(&child_node).unwrap();
if let VirtualNode::Text(text_node) = child
&& let Some(signal) = text_node.try_get_signal()
{
let signal_addr: usize = signal.get_inner_addr();
let existing_addrs: String = fragment
.get_attribute(DATA_EUV_SIGNAL_ADDRS)
.unwrap_or_default();
let updated_addrs: String = if existing_addrs.is_empty() {
signal_addr.to_string()
} else {
format!("{existing_addrs}{SIGNAL_ADDRS_SEPARATOR}{signal_addr}")
};
let _ = fragment.set_attribute(DATA_EUV_SIGNAL_ADDRS, &updated_addrs);
}
}
fragment.into()
}
VirtualNode::Dynamic(dynamic_node) => {
let placeholder: Element =
document.create_element(DYNAMIC_PLACEHOLDER_TAG).unwrap();
let _ = placeholder.set_attribute(ATTR_STYLE, DISPLAY_CONTENTS_STYLE);
let dynamic_id: usize = Self::assign_dynamic_id(&placeholder);
let initial_dom: Node =
self.setup_dynamic_node(dynamic_node, dynamic_id, &placeholder, true);
placeholder.append_child(&initial_dom).unwrap();
placeholder.into()
}
VirtualNode::Empty => document.create_text_node(EMPTY_STRING).into(),
}
}
fn setup_dynamic_node(
&mut self,
dynamic_node: &DynamicNode,
dynamic_id: usize,
placeholder: &Element,
skip_equal: bool,
) -> Node {
let mut hook_context: HookContext = dynamic_node.get_hook_context_value();
hook_context.reset_hook_index();
let initial_vnode: VirtualNode =
with_hook_context(hook_context.clone(), || dynamic_node.render());
let initial_unwrapped: VirtualNode = self.unwrap_component(&initial_vnode);
let initial_dom: Node = self.create_dom_node(&initial_unwrapped);
let render_fn: Rc<RefCell<RenderFnInner>> = dynamic_node.get_render_fn().clone();
let placeholder_clone: Element = placeholder.clone();
let mut renderer_for_sub: Self = Self::new(placeholder_clone.clone());
renderer_for_sub.set_current_tree(Some(initial_unwrapped));
let renderer_rc: Rc<RefCell<Renderer>> = Rc::new(RefCell::new(renderer_for_sub));
let initial_arm: usize = hook_context.get_inner().borrow().get_arm_changed();
let last_arm: Rc<RefCell<usize>> = Rc::new(RefCell::new(initial_arm));
let callback: Box<dyn FnMut()> = Box::new(move || {
if placeholder_clone.parent_node().is_none() {
return;
}
hook_context.reset_hook_index();
let prev_arm: usize = *last_arm.borrow();
let new_vnode: VirtualNode = with_hook_context(hook_context.clone(), || {
let mut inner: RefMut<RenderFnInner> = render_fn.borrow_mut();
(inner.get_mut_render_fn())()
});
let current_arm: usize = hook_context.get_inner().borrow().get_arm_changed();
let arm_switched: bool = prev_arm != current_arm;
*last_arm.borrow_mut() = current_arm;
if skip_equal && !arm_switched {
let renderer_ref: Ref<Renderer> = renderer_rc.borrow();
if let Some(old_vnode) = renderer_ref.try_get_current_tree() {
let new_unwrapped: VirtualNode = Self::unwrap_component_static(&new_vnode);
if Self::visual_eq(old_vnode, &new_unwrapped) {
return;
}
}
}
let mut renderer_mut: RefMut<Renderer> = renderer_rc.borrow_mut();
if arm_switched {
renderer_mut.render_full_replace(new_vnode);
} else {
renderer_mut.render(new_vnode);
}
});
register_dynamic_listener(dynamic_id, callback);
initial_dom
}
fn unwrap_component(&self, node: &VirtualNode) -> VirtualNode {
match node {
VirtualNode::Element {
tag: Tag::Component(_),
children,
..
} => {
if children.len() == 1 {
self.unwrap_component(&children[0])
} else {
VirtualNode::Fragment(children.clone())
}
}
VirtualNode::Element {
tag,
attributes,
children,
key,
} => {
if !children.iter().any(Self::subtree_has_component) {
return node.clone();
}
let unwrapped_children: Vec<VirtualNode> = children
.iter()
.map(|child: &VirtualNode| self.unwrap_component(child))
.collect();
VirtualNode::Element {
tag: tag.clone(),
attributes: attributes.clone(),
children: unwrapped_children,
key: key.clone(),
}
}
VirtualNode::Fragment(children) => {
if !children.iter().any(Self::subtree_has_component) {
return node.clone();
}
let unwrapped_children: Vec<VirtualNode> = children
.iter()
.map(|child: &VirtualNode| self.unwrap_component(child))
.collect();
VirtualNode::Fragment(unwrapped_children)
}
other => other.clone(),
}
}
fn unwrap_component_static(node: &VirtualNode) -> VirtualNode {
match node {
VirtualNode::Element {
tag: Tag::Component(_),
children,
..
} => {
if children.len() == 1 {
Self::unwrap_component_static(&children[0])
} else {
VirtualNode::Fragment(children.clone())
}
}
VirtualNode::Element {
tag,
attributes,
children,
key,
} => {
if !children.iter().any(Self::subtree_has_component) {
return node.clone();
}
let unwrapped_children: Vec<VirtualNode> =
children.iter().map(Self::unwrap_component_static).collect();
VirtualNode::Element {
tag: tag.clone(),
attributes: attributes.clone(),
children: unwrapped_children,
key: key.clone(),
}
}
VirtualNode::Fragment(children) => {
if !children.iter().any(Self::subtree_has_component) {
return node.clone();
}
let unwrapped_children: Vec<VirtualNode> =
children.iter().map(Self::unwrap_component_static).collect();
VirtualNode::Fragment(unwrapped_children)
}
other => other.clone(),
}
}
fn subtree_has_component(node: &VirtualNode) -> bool {
match node {
VirtualNode::Element {
tag: Tag::Component(_),
..
} => true,
VirtualNode::Element { children, .. } => {
children.iter().any(Self::subtree_has_component)
}
VirtualNode::Fragment(children) => children.iter().any(Self::subtree_has_component),
_ => false,
}
}
fn visual_eq(old_node: &VirtualNode, new_node: &VirtualNode) -> bool {
match (old_node, new_node) {
(VirtualNode::Text(old_text), VirtualNode::Text(new_text)) => old_text == new_text,
(
VirtualNode::Element {
tag: old_tag,
attributes: old_attrs,
children: old_children,
..
},
VirtualNode::Element {
tag: new_tag,
attributes: new_attrs,
children: new_children,
..
},
) => {
old_tag == new_tag
&& old_attrs.len() == new_attrs.len()
&& old_attrs
.iter()
.zip(new_attrs.iter())
.all(|(old_attr, new_attr)| old_attr == new_attr)
&& old_children.len() == new_children.len()
&& old_children
.iter()
.zip(new_children.iter())
.all(|(old_child, new_child)| Self::visual_eq(old_child, new_child))
}
(VirtualNode::Fragment(old_children), VirtualNode::Fragment(new_children)) => {
old_children.len() == new_children.len()
&& old_children
.iter()
.zip(new_children.iter())
.all(|(old_child, new_child)| Self::visual_eq(old_child, new_child))
}
(VirtualNode::Dynamic(_), VirtualNode::Dynamic(_)) => true,
(VirtualNode::Empty, VirtualNode::Empty) => true,
_ => false,
}
}
fn assign_dynamic_id(placeholder: &Element) -> usize {
let dynamic_id: usize = NEXT_EUV_DYNAMIC_ID.fetch_add(1, Ordering::Relaxed);
let _ = placeholder.set_attribute(DATA_EUV_DYNAMIC_ID, &dynamic_id.to_string());
dynamic_id
}
fn cleanup_dom_subtree(&self, element: &Element) {
if let Some(euv_id_str) = element.get_attribute(DATA_EUV_ID)
&& let Ok(euv_id) = euv_id_str.parse::<usize>()
{
cleanup_element_handlers(euv_id);
}
if let Some(dynamic_id_str) = element.get_attribute(DATA_EUV_DYNAMIC_ID)
&& let Ok(dynamic_id) = dynamic_id_str.parse::<usize>()
{
cleanup_dynamic_node(dynamic_id);
}
if let Some(signal_addrs_str) = element.get_attribute(DATA_EUV_SIGNAL_ADDRS) {
for addr_str in signal_addrs_str.split(CHAR_SIGNAL_ADDRS_SEPARATOR) {
if let Ok(addr) = addr_str.parse::<usize>() {
clear_signal_listeners_by_addr(addr);
}
}
}
let child_nodes: NodeList = element.child_nodes();
let length: u32 = child_nodes.length();
for child_index in 0..length {
if let Some(child) = child_nodes.get(child_index)
&& let Some(child_element) = child.dyn_ref::<Element>()
{
self.cleanup_dom_subtree(child_element);
}
}
}
fn attach_event_listener(&self, element: &Element, handler: &NativeEventHandler) {
let euv_id: usize = match element.get_attribute(DATA_EUV_ID) {
Some(id_str) => id_str
.parse::<usize>()
.unwrap_or_else(|_error: ParseIntError| {
let new_id: usize = NEXT_EUV_ID.fetch_add(1, Ordering::Relaxed);
let _ = element.set_attribute(DATA_EUV_ID, &new_id.to_string());
new_id
}),
None => {
let new_id: usize = NEXT_EUV_ID.fetch_add(1, Ordering::Relaxed);
let _ = element.set_attribute(DATA_EUV_ID, &new_id.to_string());
new_id
}
};
let event_name: String = handler.get_event_name().clone();
if !DELEGATABLE_EVENT_NAMES.contains(&event_name.as_str()) {
ensure_delegated_listener(event_name.clone());
}
let key: (usize, String) = (euv_id, event_name);
let registry_ref: &mut HashMap<(usize, String), HandlerEntry> =
ensure_handler_registry_mut();
if let Some(existing_entry) = registry_ref.get(&key) {
let mut slot: RefMut<HandlerSlot> = existing_entry.borrow_mut();
slot.set_handler(Some(handler.clone()));
} else {
let handler_slot: HandlerEntry =
Rc::new(RefCell::new(HandlerSlot::new(Some(handler.clone()))));
registry_ref.insert(key, handler_slot);
}
}
}