use dom_struct::dom_struct;
use html5ever::{LocalName, Namespace};
use crate::dom::bindings::codegen::Bindings::MutationRecordBinding::MutationRecord_Binding::MutationRecordMethods;
use crate::dom::bindings::reflector::{reflect_dom_object, Reflector};
use crate::dom::bindings::root::{Dom, DomRoot, MutNullableDom};
use crate::dom::bindings::str::DOMString;
use crate::dom::node::{window_from_node, Node};
use crate::dom::nodelist::NodeList;
#[dom_struct]
pub struct MutationRecord {
reflector_: Reflector,
record_type: DOMString,
target: Dom<Node>,
attribute_name: Option<DOMString>,
attribute_namespace: Option<DOMString>,
old_value: Option<DOMString>,
added_nodes: MutNullableDom<NodeList>,
removed_nodes: MutNullableDom<NodeList>,
next_sibling: Option<Dom<Node>>,
prev_sibling: Option<Dom<Node>>,
}
impl MutationRecord {
#[allow(crown::unrooted_must_root)]
pub fn attribute_mutated(
target: &Node,
attribute_name: &LocalName,
attribute_namespace: Option<&Namespace>,
old_value: Option<DOMString>,
) -> DomRoot<MutationRecord> {
let record = Box::new(MutationRecord::new_inherited(
"attributes",
target,
Some(DOMString::from(&**attribute_name)),
attribute_namespace.map(|n| DOMString::from(&**n)),
old_value,
None,
None,
None,
None,
));
reflect_dom_object(record, &*window_from_node(target))
}
pub fn character_data_mutated(
target: &Node,
old_value: Option<DOMString>,
) -> DomRoot<MutationRecord> {
reflect_dom_object(
Box::new(MutationRecord::new_inherited(
"characterData",
target,
None,
None,
old_value,
None,
None,
None,
None,
)),
&*window_from_node(target),
)
}
pub fn child_list_mutated(
target: &Node,
added_nodes: Option<&[&Node]>,
removed_nodes: Option<&[&Node]>,
next_sibling: Option<&Node>,
prev_sibling: Option<&Node>,
) -> DomRoot<MutationRecord> {
let window = window_from_node(target);
let added_nodes = added_nodes.map(|list| NodeList::new_simple_list_slice(&window, list));
let removed_nodes =
removed_nodes.map(|list| NodeList::new_simple_list_slice(&window, list));
reflect_dom_object(
Box::new(MutationRecord::new_inherited(
"childList",
target,
None,
None,
None,
added_nodes.as_deref(),
removed_nodes.as_deref(),
next_sibling,
prev_sibling,
)),
&*window,
)
}
#[allow(clippy::too_many_arguments)]
fn new_inherited(
record_type: &str,
target: &Node,
attribute_name: Option<DOMString>,
attribute_namespace: Option<DOMString>,
old_value: Option<DOMString>,
added_nodes: Option<&NodeList>,
removed_nodes: Option<&NodeList>,
next_sibling: Option<&Node>,
prev_sibling: Option<&Node>,
) -> MutationRecord {
MutationRecord {
reflector_: Reflector::new(),
record_type: DOMString::from(record_type),
target: Dom::from_ref(target),
attribute_name,
attribute_namespace,
old_value,
added_nodes: MutNullableDom::new(added_nodes),
removed_nodes: MutNullableDom::new(removed_nodes),
next_sibling: next_sibling.map(Dom::from_ref),
prev_sibling: prev_sibling.map(Dom::from_ref),
}
}
}
impl MutationRecordMethods for MutationRecord {
fn Type(&self) -> DOMString {
self.record_type.clone()
}
fn Target(&self) -> DomRoot<Node> {
DomRoot::from_ref(&*self.target)
}
fn GetAttributeName(&self) -> Option<DOMString> {
self.attribute_name.clone()
}
fn GetAttributeNamespace(&self) -> Option<DOMString> {
self.attribute_namespace.clone()
}
fn GetOldValue(&self) -> Option<DOMString> {
self.old_value.clone()
}
fn AddedNodes(&self) -> DomRoot<NodeList> {
self.added_nodes.or_init(|| {
let window = window_from_node(&*self.target);
NodeList::empty(&window)
})
}
fn RemovedNodes(&self) -> DomRoot<NodeList> {
self.removed_nodes.or_init(|| {
let window = window_from_node(&*self.target);
NodeList::empty(&window)
})
}
fn GetPreviousSibling(&self) -> Option<DomRoot<Node>> {
self.prev_sibling
.as_ref()
.map(|node| DomRoot::from_ref(&**node))
}
fn GetNextSibling(&self) -> Option<DomRoot<Node>> {
self.next_sibling
.as_ref()
.map(|node| DomRoot::from_ref(&**node))
}
}