use std::cell::UnsafeCell;
use std::cmp::{Ordering, PartialOrd};
use dom_struct::dom_struct;
use js::jsapi::JSTracer;
use js::rust::HandleObject;
use malloc_size_of::{MallocSizeOf, MallocSizeOfOps};
use crate::dom::abstractrange::{bp_position, AbstractRange, BoundaryPoint};
use crate::dom::bindings::cell::DomRefCell;
use crate::dom::bindings::codegen::Bindings::AbstractRangeBinding::AbstractRangeMethods;
use crate::dom::bindings::codegen::Bindings::CharacterDataBinding::CharacterDataMethods;
use crate::dom::bindings::codegen::Bindings::NodeBinding::NodeMethods;
use crate::dom::bindings::codegen::Bindings::NodeListBinding::NodeListMethods;
use crate::dom::bindings::codegen::Bindings::RangeBinding::{RangeConstants, RangeMethods};
use crate::dom::bindings::codegen::Bindings::TextBinding::TextMethods;
use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowMethods;
use crate::dom::bindings::error::{Error, ErrorResult, Fallible};
use crate::dom::bindings::inheritance::{Castable, CharacterDataTypeId, NodeTypeId};
use crate::dom::bindings::reflector::reflect_dom_object_with_proto;
use crate::dom::bindings::root::{Dom, DomRoot};
use crate::dom::bindings::str::DOMString;
use crate::dom::bindings::trace::JSTraceable;
use crate::dom::bindings::weakref::{WeakRef, WeakRefVec};
use crate::dom::characterdata::CharacterData;
use crate::dom::document::Document;
use crate::dom::documentfragment::DocumentFragment;
use crate::dom::element::Element;
use crate::dom::htmlscriptelement::HTMLScriptElement;
use crate::dom::node::{Node, ShadowIncluding, UnbindContext};
use crate::dom::selection::Selection;
use crate::dom::text::Text;
use crate::dom::window::Window;
use crate::script_runtime::CanGc;
#[dom_struct]
pub struct Range {
abstract_range: AbstractRange,
associated_selections: DomRefCell<Vec<Dom<Selection>>>,
}
struct ContainedChildren {
first_partially_contained_child: Option<DomRoot<Node>>,
last_partially_contained_child: Option<DomRoot<Node>>,
contained_children: Vec<DomRoot<Node>>,
}
impl Range {
fn new_inherited(
start_container: &Node,
start_offset: u32,
end_container: &Node,
end_offset: u32,
) -> Range {
debug_assert!(start_offset <= start_container.len());
debug_assert!(end_offset <= end_container.len());
Range {
abstract_range: AbstractRange::new_inherited(
start_container,
start_offset,
end_container,
end_offset,
),
associated_selections: DomRefCell::new(vec![]),
}
}
pub fn new_with_doc(
document: &Document,
proto: Option<HandleObject>,
can_gc: CanGc,
) -> DomRoot<Range> {
let root = document.upcast();
Range::new_with_proto(document, proto, root, 0, root, 0, can_gc)
}
pub fn new(
document: &Document,
start_container: &Node,
start_offset: u32,
end_container: &Node,
end_offset: u32,
can_gc: CanGc,
) -> DomRoot<Range> {
Self::new_with_proto(
document,
None,
start_container,
start_offset,
end_container,
end_offset,
can_gc,
)
}
fn new_with_proto(
document: &Document,
proto: Option<HandleObject>,
start_container: &Node,
start_offset: u32,
end_container: &Node,
end_offset: u32,
can_gc: CanGc,
) -> DomRoot<Range> {
let range = reflect_dom_object_with_proto(
Box::new(Range::new_inherited(
start_container,
start_offset,
end_container,
end_offset,
)),
document.window(),
proto,
can_gc,
);
start_container.ranges().push(WeakRef::new(&range));
if start_container != end_container {
end_container.ranges().push(WeakRef::new(&range));
}
range
}
fn contains(&self, node: &Node) -> bool {
matches!(
(
bp_position(node, 0, &self.start_container(), self.start_offset()),
bp_position(node, node.len(), &self.end_container(), self.end_offset()),
),
(Some(Ordering::Greater), Some(Ordering::Less))
)
}
fn partially_contains(&self, node: &Node) -> bool {
self.start_container()
.inclusive_ancestors(ShadowIncluding::No)
.any(|n| &*n == node) !=
self.end_container()
.inclusive_ancestors(ShadowIncluding::No)
.any(|n| &*n == node)
}
fn contained_children(&self) -> Fallible<ContainedChildren> {
let start_node = self.start_container();
let end_node = self.end_container();
let common_ancestor = self.CommonAncestorContainer();
let first_partially_contained_child = if start_node.is_inclusive_ancestor_of(&end_node) {
None
} else {
common_ancestor
.children()
.find(|node| Range::partially_contains(self, node))
};
let last_partially_contained_child = if end_node.is_inclusive_ancestor_of(&start_node) {
None
} else {
common_ancestor
.rev_children()
.find(|node| Range::partially_contains(self, node))
};
let contained_children: Vec<DomRoot<Node>> = common_ancestor
.children()
.filter(|n| self.contains(n))
.collect();
if contained_children.iter().any(|n| n.is_doctype()) {
return Err(Error::HierarchyRequest);
}
Ok(ContainedChildren {
first_partially_contained_child,
last_partially_contained_child,
contained_children,
})
}
fn set_start(&self, node: &Node, offset: u32) {
if self.start().node() != node || self.start_offset() != offset {
self.report_change();
}
if self.start().node() != node {
if self.start().node() == self.end().node() {
node.ranges().push(WeakRef::new(self));
} else if self.end().node() == node {
self.start_container().ranges().remove(self);
} else {
node.ranges()
.push(self.start_container().ranges().remove(self));
}
}
self.start().set(node, offset);
}
fn set_end(&self, node: &Node, offset: u32) {
if self.end().node() != node || self.end_offset() != offset {
self.report_change();
}
if self.end().node() != node {
if self.end().node() == self.start().node() {
node.ranges().push(WeakRef::new(self));
} else if self.start().node() == node {
self.end_container().ranges().remove(self);
} else {
node.ranges()
.push(self.end_container().ranges().remove(self));
}
}
self.end().set(node, offset);
}
fn compare_point(&self, node: &Node, offset: u32) -> Fallible<Ordering> {
let start_node = self.start_container();
let start_node_root = start_node
.inclusive_ancestors(ShadowIncluding::No)
.last()
.unwrap();
let node_root = node
.inclusive_ancestors(ShadowIncluding::No)
.last()
.unwrap();
if start_node_root != node_root {
return Err(Error::WrongDocument);
}
if node.is_doctype() {
return Err(Error::InvalidNodeType);
}
if offset > node.len() {
return Err(Error::IndexSize);
}
if let Ordering::Less = bp_position(node, offset, &start_node, self.start_offset()).unwrap()
{
return Ok(Ordering::Less);
}
if let Ordering::Greater =
bp_position(node, offset, &self.end_container(), self.end_offset()).unwrap()
{
return Ok(Ordering::Greater);
}
Ok(Ordering::Equal)
}
pub fn associate_selection(&self, selection: &Selection) {
let mut selections = self.associated_selections.borrow_mut();
if !selections.iter().any(|s| &**s == selection) {
selections.push(Dom::from_ref(selection));
}
}
pub fn disassociate_selection(&self, selection: &Selection) {
self.associated_selections
.borrow_mut()
.retain(|s| &**s != selection);
}
fn report_change(&self) {
self.associated_selections
.borrow()
.iter()
.for_each(|s| s.queue_selectionchange_task());
}
fn abstract_range(&self) -> &AbstractRange {
&self.abstract_range
}
fn start(&self) -> &BoundaryPoint {
self.abstract_range().start()
}
fn end(&self) -> &BoundaryPoint {
self.abstract_range().end()
}
pub fn start_container(&self) -> DomRoot<Node> {
self.abstract_range().StartContainer()
}
pub fn start_offset(&self) -> u32 {
self.abstract_range().StartOffset()
}
pub fn end_container(&self) -> DomRoot<Node> {
self.abstract_range().EndContainer()
}
pub fn end_offset(&self) -> u32 {
self.abstract_range().EndOffset()
}
pub fn collapsed(&self) -> bool {
self.abstract_range().Collapsed()
}
}
impl RangeMethods<crate::DomTypeHolder> for Range {
fn Constructor(
window: &Window,
proto: Option<HandleObject>,
can_gc: CanGc,
) -> Fallible<DomRoot<Range>> {
let document = window.Document();
Ok(Range::new_with_doc(&document, proto, can_gc))
}
fn CommonAncestorContainer(&self) -> DomRoot<Node> {
self.end_container()
.common_ancestor(&self.start_container(), ShadowIncluding::No)
.expect("Couldn't find common ancestor container")
}
#[allow(clippy::neg_cmp_op_on_partial_ord)]
fn SetStart(&self, node: &Node, offset: u32) -> ErrorResult {
if node.is_doctype() {
Err(Error::InvalidNodeType)
} else if offset > node.len() {
Err(Error::IndexSize)
} else {
self.set_start(node, offset);
if !(self.start() <= self.end()) {
self.set_end(node, offset);
}
Ok(())
}
}
#[allow(clippy::neg_cmp_op_on_partial_ord)]
fn SetEnd(&self, node: &Node, offset: u32) -> ErrorResult {
if node.is_doctype() {
Err(Error::InvalidNodeType)
} else if offset > node.len() {
Err(Error::IndexSize)
} else {
self.set_end(node, offset);
if !(self.end() >= self.start()) {
self.set_start(node, offset);
}
Ok(())
}
}
fn SetStartBefore(&self, node: &Node) -> ErrorResult {
let parent = node.GetParentNode().ok_or(Error::InvalidNodeType)?;
self.SetStart(&parent, node.index())
}
fn SetStartAfter(&self, node: &Node) -> ErrorResult {
let parent = node.GetParentNode().ok_or(Error::InvalidNodeType)?;
self.SetStart(&parent, node.index() + 1)
}
fn SetEndBefore(&self, node: &Node) -> ErrorResult {
let parent = node.GetParentNode().ok_or(Error::InvalidNodeType)?;
self.SetEnd(&parent, node.index())
}
fn SetEndAfter(&self, node: &Node) -> ErrorResult {
let parent = node.GetParentNode().ok_or(Error::InvalidNodeType)?;
self.SetEnd(&parent, node.index() + 1)
}
fn Collapse(&self, to_start: bool) {
if to_start {
self.set_end(&self.start_container(), self.start_offset());
} else {
self.set_start(&self.end_container(), self.end_offset());
}
}
fn SelectNode(&self, node: &Node) -> ErrorResult {
let parent = node.GetParentNode().ok_or(Error::InvalidNodeType)?;
let index = node.index();
self.set_start(&parent, index);
self.set_end(&parent, index + 1);
Ok(())
}
fn SelectNodeContents(&self, node: &Node) -> ErrorResult {
if node.is_doctype() {
return Err(Error::InvalidNodeType);
}
let length = node.len();
self.set_start(node, 0);
self.set_end(node, length);
Ok(())
}
fn CompareBoundaryPoints(&self, how: u16, other: &Range) -> Fallible<i16> {
if how > RangeConstants::END_TO_START {
return Err(Error::NotSupported);
}
let this_root = self
.start_container()
.inclusive_ancestors(ShadowIncluding::No)
.last()
.unwrap();
let other_root = other
.start_container()
.inclusive_ancestors(ShadowIncluding::No)
.last()
.unwrap();
if this_root != other_root {
return Err(Error::WrongDocument);
}
let (this_point, other_point) = match how {
RangeConstants::START_TO_START => (self.start(), other.start()),
RangeConstants::START_TO_END => (self.end(), other.start()),
RangeConstants::END_TO_END => (self.end(), other.end()),
RangeConstants::END_TO_START => (self.start(), other.end()),
_ => unreachable!(),
};
match this_point.partial_cmp(other_point).unwrap() {
Ordering::Less => Ok(-1),
Ordering::Equal => Ok(0),
Ordering::Greater => Ok(1),
}
}
fn CloneRange(&self, can_gc: CanGc) -> DomRoot<Range> {
let start_node = self.start_container();
let owner_doc = start_node.owner_doc();
Range::new(
&owner_doc,
&start_node,
self.start_offset(),
&self.end_container(),
self.end_offset(),
can_gc,
)
}
fn IsPointInRange(&self, node: &Node, offset: u32) -> Fallible<bool> {
match self.compare_point(node, offset) {
Ok(Ordering::Less) => Ok(false),
Ok(Ordering::Equal) => Ok(true),
Ok(Ordering::Greater) => Ok(false),
Err(Error::WrongDocument) => {
Ok(false)
},
Err(error) => Err(error),
}
}
fn ComparePoint(&self, node: &Node, offset: u32) -> Fallible<i16> {
self.compare_point(node, offset).map(|order| match order {
Ordering::Less => -1,
Ordering::Equal => 0,
Ordering::Greater => 1,
})
}
fn IntersectsNode(&self, node: &Node) -> bool {
let start_node = self.start_container();
let start_node_root = self
.start_container()
.inclusive_ancestors(ShadowIncluding::No)
.last()
.unwrap();
let node_root = node
.inclusive_ancestors(ShadowIncluding::No)
.last()
.unwrap();
if start_node_root != node_root {
return false;
}
let parent = match node.GetParentNode() {
Some(parent) => parent,
None => {
return true;
},
};
let offset = node.index();
Ordering::Greater ==
bp_position(&parent, offset + 1, &start_node, self.start_offset()).unwrap() &&
Ordering::Less ==
bp_position(&parent, offset, &self.end_container(), self.end_offset())
.unwrap()
}
fn CloneContents(&self, can_gc: CanGc) -> Fallible<DomRoot<DocumentFragment>> {
let start_node = self.start_container();
let start_offset = self.start_offset();
let end_node = self.end_container();
let end_offset = self.end_offset();
let fragment = DocumentFragment::new(&start_node.owner_doc(), can_gc);
if self.start() == self.end() {
return Ok(fragment);
}
if end_node == start_node {
if let Some(cdata) = start_node.downcast::<CharacterData>() {
let data = cdata
.SubstringData(start_offset, end_offset - start_offset)
.unwrap();
let clone = cdata.clone_with_data(data, &start_node.owner_doc(), can_gc);
fragment.upcast::<Node>().AppendChild(&clone)?;
return Ok(fragment);
}
}
let ContainedChildren {
first_partially_contained_child,
last_partially_contained_child,
contained_children,
} = self.contained_children()?;
if let Some(child) = first_partially_contained_child {
if let Some(cdata) = child.downcast::<CharacterData>() {
assert!(child == start_node);
let data = cdata
.SubstringData(start_offset, start_node.len() - start_offset)
.unwrap();
let clone = cdata.clone_with_data(data, &start_node.owner_doc(), can_gc);
fragment.upcast::<Node>().AppendChild(&clone)?;
} else {
let clone = child.CloneNode(false, can_gc)?;
fragment.upcast::<Node>().AppendChild(&clone)?;
let subrange = Range::new(
&clone.owner_doc(),
&start_node,
start_offset,
&child,
child.len(),
can_gc,
);
let subfragment = subrange.CloneContents(can_gc)?;
clone.AppendChild(subfragment.upcast())?;
}
}
for child in contained_children {
let clone = child.CloneNode(true, can_gc)?;
fragment.upcast::<Node>().AppendChild(&clone)?;
}
if let Some(child) = last_partially_contained_child {
if let Some(cdata) = child.downcast::<CharacterData>() {
assert!(child == end_node);
let data = cdata.SubstringData(0, end_offset).unwrap();
let clone = cdata.clone_with_data(data, &start_node.owner_doc(), can_gc);
fragment.upcast::<Node>().AppendChild(&clone)?;
} else {
let clone = child.CloneNode(false, can_gc)?;
fragment.upcast::<Node>().AppendChild(&clone)?;
let subrange =
Range::new(&clone.owner_doc(), &child, 0, &end_node, end_offset, can_gc);
let subfragment = subrange.CloneContents(can_gc)?;
clone.AppendChild(subfragment.upcast())?;
}
}
Ok(fragment)
}
fn ExtractContents(&self, can_gc: CanGc) -> Fallible<DomRoot<DocumentFragment>> {
let start_node = self.start_container();
let start_offset = self.start_offset();
let end_node = self.end_container();
let end_offset = self.end_offset();
let fragment = DocumentFragment::new(&start_node.owner_doc(), can_gc);
if self.collapsed() {
return Ok(fragment);
}
if end_node == start_node {
if let Some(end_data) = end_node.downcast::<CharacterData>() {
let clone = end_node.CloneNode(true, can_gc)?;
let text = end_data.SubstringData(start_offset, end_offset - start_offset);
clone
.downcast::<CharacterData>()
.unwrap()
.SetData(text.unwrap());
fragment.upcast::<Node>().AppendChild(&clone)?;
end_data.ReplaceData(start_offset, end_offset - start_offset, DOMString::new())?;
return Ok(fragment);
}
}
let ContainedChildren {
first_partially_contained_child,
last_partially_contained_child,
contained_children,
} = self.contained_children()?;
let (new_node, new_offset) = if start_node.is_inclusive_ancestor_of(&end_node) {
(DomRoot::from_ref(&*start_node), start_offset)
} else {
let reference_node = start_node
.ancestors()
.take_while(|n| !n.is_inclusive_ancestor_of(&end_node))
.last()
.unwrap_or(DomRoot::from_ref(&start_node));
(
reference_node.GetParentNode().unwrap(),
reference_node.index() + 1,
)
};
if let Some(child) = first_partially_contained_child {
if let Some(start_data) = child.downcast::<CharacterData>() {
assert!(child == start_node);
let clone = start_node.CloneNode(true, can_gc)?;
let text = start_data.SubstringData(start_offset, start_node.len() - start_offset);
clone
.downcast::<CharacterData>()
.unwrap()
.SetData(text.unwrap());
fragment.upcast::<Node>().AppendChild(&clone)?;
start_data.ReplaceData(
start_offset,
start_node.len() - start_offset,
DOMString::new(),
)?;
} else {
let clone = child.CloneNode(false, can_gc)?;
fragment.upcast::<Node>().AppendChild(&clone)?;
let subrange = Range::new(
&clone.owner_doc(),
&start_node,
start_offset,
&child,
child.len(),
can_gc,
);
let subfragment = subrange.ExtractContents(can_gc)?;
clone.AppendChild(subfragment.upcast())?;
}
}
for child in contained_children {
fragment.upcast::<Node>().AppendChild(&child)?;
}
if let Some(child) = last_partially_contained_child {
if let Some(end_data) = child.downcast::<CharacterData>() {
assert!(child == end_node);
let clone = end_node.CloneNode(true, can_gc)?;
let text = end_data.SubstringData(0, end_offset);
clone
.downcast::<CharacterData>()
.unwrap()
.SetData(text.unwrap());
fragment.upcast::<Node>().AppendChild(&clone)?;
end_data.ReplaceData(0, end_offset, DOMString::new())?;
} else {
let clone = child.CloneNode(false, can_gc)?;
fragment.upcast::<Node>().AppendChild(&clone)?;
let subrange =
Range::new(&clone.owner_doc(), &child, 0, &end_node, end_offset, can_gc);
let subfragment = subrange.ExtractContents(can_gc)?;
clone.AppendChild(subfragment.upcast())?;
}
}
self.SetStart(&new_node, new_offset)?;
self.SetEnd(&new_node, new_offset)?;
Ok(fragment)
}
fn Detach(&self) {
}
fn InsertNode(&self, node: &Node, can_gc: CanGc) -> ErrorResult {
let start_node = self.start_container();
let start_offset = self.start_offset();
if &*start_node == node {
return Err(Error::HierarchyRequest);
}
match start_node.type_id() {
NodeTypeId::CharacterData(CharacterDataTypeId::Text(_)) => (),
NodeTypeId::CharacterData(_) => return Err(Error::HierarchyRequest),
_ => (),
}
let (reference_node, parent) = match start_node.type_id() {
NodeTypeId::CharacterData(CharacterDataTypeId::Text(_)) => {
let parent = match start_node.GetParentNode() {
Some(parent) => parent,
None => return Err(Error::HierarchyRequest),
};
(Some(DomRoot::from_ref(&*start_node)), parent)
},
_ => {
let child = start_node.ChildNodes().Item(start_offset);
(child, DomRoot::from_ref(&*start_node))
},
};
Node::ensure_pre_insertion_validity(node, &parent, reference_node.as_deref())?;
let split_text;
let reference_node = match start_node.downcast::<Text>() {
Some(text) => {
split_text = text.SplitText(start_offset, can_gc)?;
let new_reference = DomRoot::upcast::<Node>(split_text);
assert!(new_reference.GetParentNode().as_deref() == Some(&parent));
Some(new_reference)
},
_ => reference_node,
};
let reference_node = if Some(node) == reference_node.as_deref() {
node.GetNextSibling()
} else {
reference_node
};
node.remove_self();
let new_offset = reference_node
.as_ref()
.map_or(parent.len(), |node| node.index());
let new_offset = new_offset +
if let NodeTypeId::DocumentFragment(_) = node.type_id() {
node.len()
} else {
1
};
Node::pre_insert(node, &parent, reference_node.as_deref())?;
if self.collapsed() {
self.set_end(&parent, new_offset);
}
Ok(())
}
fn DeleteContents(&self) -> ErrorResult {
if self.collapsed() {
return Ok(());
}
let start_node = self.start_container();
let end_node = self.end_container();
let start_offset = self.start_offset();
let end_offset = self.end_offset();
if start_node == end_node {
if let Some(text) = start_node.downcast::<CharacterData>() {
if end_offset > start_offset {
self.report_change();
}
return text.ReplaceData(start_offset, end_offset - start_offset, DOMString::new());
}
}
rooted_vec!(let mut contained_children);
let ancestor = self.CommonAncestorContainer();
let mut iter = start_node.following_nodes(&ancestor);
let mut next = iter.next();
while let Some(child) = next {
if self.contains(&child) {
contained_children.push(Dom::from_ref(&*child));
next = iter.next_skipping_children();
} else {
next = iter.next();
}
}
let (new_node, new_offset) = if start_node.is_inclusive_ancestor_of(&end_node) {
(DomRoot::from_ref(&*start_node), start_offset)
} else {
fn compute_reference(start_node: &Node, end_node: &Node) -> (DomRoot<Node>, u32) {
let mut reference_node = DomRoot::from_ref(start_node);
while let Some(parent) = reference_node.GetParentNode() {
if parent.is_inclusive_ancestor_of(end_node) {
return (parent, reference_node.index() + 1);
}
reference_node = parent;
}
unreachable!()
}
compute_reference(&start_node, &end_node)
};
if let Some(text) = start_node.downcast::<CharacterData>() {
text.ReplaceData(
start_offset,
start_node.len() - start_offset,
DOMString::new(),
)
.unwrap();
}
for child in &*contained_children {
child.remove_self();
}
if let Some(text) = end_node.downcast::<CharacterData>() {
text.ReplaceData(0, end_offset, DOMString::new()).unwrap();
}
self.SetStart(&new_node, new_offset).unwrap();
self.SetEnd(&new_node, new_offset).unwrap();
Ok(())
}
fn SurroundContents(&self, new_parent: &Node, can_gc: CanGc) -> ErrorResult {
let start = self.start_container();
let end = self.end_container();
if start
.inclusive_ancestors(ShadowIncluding::No)
.any(|n| !n.is_inclusive_ancestor_of(&end) && !n.is::<Text>()) ||
end.inclusive_ancestors(ShadowIncluding::No)
.any(|n| !n.is_inclusive_ancestor_of(&start) && !n.is::<Text>())
{
return Err(Error::InvalidState);
}
match new_parent.type_id() {
NodeTypeId::Document(_) |
NodeTypeId::DocumentType |
NodeTypeId::DocumentFragment(_) => {
return Err(Error::InvalidNodeType);
},
_ => (),
}
let fragment = self.ExtractContents(can_gc)?;
Node::replace_all(None, new_parent);
self.InsertNode(new_parent, can_gc)?;
new_parent.AppendChild(fragment.upcast())?;
self.SelectNode(new_parent)
}
fn Stringifier(&self) -> DOMString {
let start_node = self.start_container();
let end_node = self.end_container();
let mut s = DOMString::new();
if let Some(text_node) = start_node.downcast::<Text>() {
let char_data = text_node.upcast::<CharacterData>();
if start_node == end_node {
return char_data
.SubstringData(self.start_offset(), self.end_offset() - self.start_offset())
.unwrap();
}
s.push_str(
&char_data
.SubstringData(
self.start_offset(),
char_data.Length() - self.start_offset(),
)
.unwrap(),
);
}
let ancestor = self.CommonAncestorContainer();
let iter = start_node
.following_nodes(&ancestor)
.filter_map(DomRoot::downcast::<Text>);
for child in iter {
if self.contains(child.upcast()) {
s.push_str(&child.upcast::<CharacterData>().Data());
}
}
if let Some(text_node) = end_node.downcast::<Text>() {
let char_data = text_node.upcast::<CharacterData>();
s.push_str(&char_data.SubstringData(0, self.end_offset()).unwrap());
}
s
}
fn CreateContextualFragment(
&self,
fragment: DOMString,
can_gc: CanGc,
) -> Fallible<DomRoot<DocumentFragment>> {
let node = self.start_container();
let owner_doc = node.owner_doc();
let element = match node.type_id() {
NodeTypeId::Document(_) | NodeTypeId::DocumentFragment(_) => None,
NodeTypeId::Element(_) => Some(DomRoot::downcast::<Element>(node).unwrap()),
NodeTypeId::CharacterData(CharacterDataTypeId::Comment) |
NodeTypeId::CharacterData(CharacterDataTypeId::Text(_)) => node.GetParentElement(),
NodeTypeId::CharacterData(CharacterDataTypeId::ProcessingInstruction) |
NodeTypeId::DocumentType => unreachable!(),
NodeTypeId::Attr => unreachable!(),
};
let element = Element::fragment_parsing_context(&owner_doc, element.as_deref(), can_gc);
let fragment_node = element.parse_fragment(fragment, can_gc)?;
for node in fragment_node
.upcast::<Node>()
.traverse_preorder(ShadowIncluding::No)
{
if let Some(script) = node.downcast::<HTMLScriptElement>() {
script.set_already_started(false);
script.set_parser_inserted(false);
}
}
Ok(fragment_node)
}
}
pub struct WeakRangeVec {
cell: UnsafeCell<WeakRefVec<Range>>,
}
impl Default for WeakRangeVec {
fn default() -> Self {
WeakRangeVec {
cell: UnsafeCell::new(WeakRefVec::new()),
}
}
}
#[allow(unsafe_code)]
impl WeakRangeVec {
pub fn new() -> Self {
Self::default()
}
pub fn is_empty(&self) -> bool {
unsafe { (*self.cell.get()).is_empty() }
}
pub fn increase_above(&self, node: &Node, offset: u32, delta: u32) {
self.map_offset_above(node, offset, |offset| offset + delta);
}
pub fn decrease_above(&self, node: &Node, offset: u32, delta: u32) {
self.map_offset_above(node, offset, |offset| offset - delta);
}
pub fn drain_to_parent(&self, context: &UnbindContext, child: &Node) {
if self.is_empty() {
return;
}
let offset = context.index();
let parent = context.parent;
unsafe {
let ranges = &mut *self.cell.get();
ranges.update(|entry| {
let range = entry.root().unwrap();
if range.start().node() == parent || range.end().node() == parent {
entry.remove();
}
if range.start().node() == child {
range.report_change();
range.start().set(context.parent, offset);
}
if range.end().node() == child {
range.report_change();
range.end().set(context.parent, offset);
}
});
(*context.parent.ranges().cell.get()).extend(ranges.drain(..));
}
}
pub fn drain_to_preceding_text_sibling(&self, node: &Node, sibling: &Node, length: u32) {
if self.is_empty() {
return;
}
unsafe {
let ranges = &mut *self.cell.get();
ranges.update(|entry| {
let range = entry.root().unwrap();
if range.start().node() == sibling || range.end().node() == sibling {
entry.remove();
}
if range.start().node() == node {
range.report_change();
range.start().set(sibling, range.start_offset() + length);
}
if range.end().node() == node {
range.report_change();
range.end().set(sibling, range.end_offset() + length);
}
});
(*sibling.ranges().cell.get()).extend(ranges.drain(..));
}
}
pub fn move_to_text_child_at(&self, node: &Node, offset: u32, child: &Node, new_offset: u32) {
unsafe {
let child_ranges = &mut *child.ranges().cell.get();
(*self.cell.get()).update(|entry| {
let range = entry.root().unwrap();
let node_is_start = range.start().node() == node;
let node_is_end = range.end().node() == node;
let move_start = node_is_start && range.start_offset() == offset;
let move_end = node_is_end && range.end_offset() == offset;
let remove_from_node =
move_start && (move_end || !node_is_end) || move_end && !node_is_start;
let already_in_child = range.start().node() == child || range.end().node() == child;
let push_to_child = !already_in_child && (move_start || move_end);
if remove_from_node {
let ref_ = entry.remove();
if push_to_child {
child_ranges.push(ref_);
}
} else if push_to_child {
child_ranges.push(WeakRef::new(&range));
}
if move_start {
range.report_change();
range.start().set(child, new_offset);
}
if move_end {
range.report_change();
range.end().set(child, new_offset);
}
});
}
}
pub fn replace_code_units(
&self,
node: &Node,
offset: u32,
removed_code_units: u32,
added_code_units: u32,
) {
self.map_offset_above(node, offset, |range_offset| {
if range_offset <= offset + removed_code_units {
offset
} else {
range_offset + added_code_units - removed_code_units
}
});
}
pub fn move_to_following_text_sibling_above(&self, node: &Node, offset: u32, sibling: &Node) {
unsafe {
let sibling_ranges = &mut *sibling.ranges().cell.get();
(*self.cell.get()).update(|entry| {
let range = entry.root().unwrap();
let start_offset = range.start_offset();
let end_offset = range.end_offset();
let node_is_start = range.start().node() == node;
let node_is_end = range.end().node() == node;
let move_start = node_is_start && start_offset > offset;
let move_end = node_is_end && end_offset > offset;
let remove_from_node =
move_start && (move_end || !node_is_end) || move_end && !node_is_start;
let already_in_sibling =
range.start().node() == sibling || range.end().node() == sibling;
let push_to_sibling = !already_in_sibling && (move_start || move_end);
if remove_from_node {
let ref_ = entry.remove();
if push_to_sibling {
sibling_ranges.push(ref_);
}
} else if push_to_sibling {
sibling_ranges.push(WeakRef::new(&range));
}
if move_start {
range.report_change();
range.start().set(sibling, start_offset - offset);
}
if move_end {
range.report_change();
range.end().set(sibling, end_offset - offset);
}
});
}
}
pub fn increment_at(&self, node: &Node, offset: u32) {
unsafe {
(*self.cell.get()).update(|entry| {
let range = entry.root().unwrap();
if range.start().node() == node && offset == range.start_offset() {
range.report_change();
range.start().set_offset(offset + 1);
}
if range.end().node() == node && offset == range.end_offset() {
range.report_change();
range.end().set_offset(offset + 1);
}
});
}
}
fn map_offset_above<F: FnMut(u32) -> u32>(&self, node: &Node, offset: u32, mut f: F) {
unsafe {
(*self.cell.get()).update(|entry| {
let range = entry.root().unwrap();
let start_offset = range.start_offset();
if range.start().node() == node && start_offset > offset {
range.report_change();
range.start().set_offset(f(start_offset));
}
let end_offset = range.end_offset();
if range.end().node() == node && end_offset > offset {
range.report_change();
range.end().set_offset(f(end_offset));
}
});
}
}
pub fn push(&self, ref_: WeakRef<Range>) {
unsafe {
(*self.cell.get()).push(ref_);
}
}
fn remove(&self, range: &Range) -> WeakRef<Range> {
unsafe {
let ranges = &mut *self.cell.get();
let position = ranges.iter().position(|ref_| ref_ == range).unwrap();
ranges.swap_remove(position)
}
}
}
#[allow(unsafe_code)]
impl MallocSizeOf for WeakRangeVec {
fn size_of(&self, ops: &mut MallocSizeOfOps) -> usize {
unsafe { (*self.cell.get()).size_of(ops) }
}
}
#[allow(unsafe_code)]
unsafe impl JSTraceable for WeakRangeVec {
unsafe fn trace(&self, _: *mut JSTracer) {
(*self.cell.get()).retain_alive()
}
}