/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. */ 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; #[dom_struct] pub struct Range { abstract_range: AbstractRange, // A range that belongs to a Selection needs to know about it // so selectionchange can fire when the range changes. // A range shouldn't belong to more than one Selection at a time, // but from the spec as of Feb 1 2020 I can't rule out a corner case like: // * Select a range R in document A, from node X to Y // * Insert everything from X to Y into document B // * Set B's selection's range to R // which leaves R technically, and observably, associated with A even though // it will fail the same-root-node check on many of A's selection's methods. associated_selections: DomRefCell>>, } 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) -> DomRoot { let root = document.upcast(); Range::new_with_proto(document, proto, root, 0, root, 0) } pub fn new( document: &Document, start_container: &Node, start_offset: u32, end_container: &Node, end_offset: u32, ) -> DomRoot { Self::new_with_proto( document, None, start_container, start_offset, end_container, end_offset, ) } fn new_with_proto( document: &Document, proto: Option, start_container: &Node, start_offset: u32, end_container: &Node, end_offset: u32, ) -> DomRoot { let range = reflect_dom_object_with_proto( Box::new(Range::new_inherited( start_container, start_offset, end_container, end_offset, )), document.window(), proto, ); start_container.ranges().push(WeakRef::new(&range)); if start_container != end_container { end_container.ranges().push(WeakRef::new(&range)); } range } /// #[allow(non_snake_case)] pub fn Constructor(window: &Window, proto: Option) -> Fallible> { let document = window.Document(); Ok(Range::new_with_doc(&document, proto)) } /// 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<( Option>, Option>, Vec>, )> { let start_node = self.start_container(); let end_node = self.end_container(); // Steps 5-6. let common_ancestor = self.CommonAncestorContainer(); let first_contained_child = if start_node.is_inclusive_ancestor_of(&end_node) { // Step 7. None } else { // Step 8. common_ancestor .children() .find(|node| Range::partially_contains(self, node)) }; let last_contained_child = if end_node.is_inclusive_ancestor_of(&start_node) { // Step 9. None } else { // Step 10. common_ancestor .rev_children() .find(|node| Range::partially_contains(self, node)) }; // Step 11. let contained_children: Vec> = common_ancestor .children() .filter(|n| self.contains(n)) .collect(); // Step 12. if contained_children.iter().any(|n| n.is_doctype()) { return Err(Error::HierarchyRequest); } Ok(( first_contained_child, last_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 { 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 { // Step 1. return Err(Error::WrongDocument); } if node.is_doctype() { // Step 2. return Err(Error::InvalidNodeType); } if offset > node.len() { // Step 3. return Err(Error::IndexSize); } if let Ordering::Less = bp_position(node, offset, &start_node, self.start_offset()).unwrap() { // Step 4. return Ok(Ordering::Less); } if let Ordering::Greater = bp_position(node, offset, &self.end_container(), self.end_offset()).unwrap() { // Step 5. return Ok(Ordering::Greater); } // Step 6. 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 { self.abstract_range().StartContainer() } pub fn start_offset(&self) -> u32 { self.abstract_range().StartOffset() } pub fn end_container(&self) -> DomRoot { 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 for Range { /// fn CommonAncestorContainer(&self) -> DomRoot { self.end_container() .common_ancestor(&self.start_container(), ShadowIncluding::No) .expect("Couldn't find common ancestor container") } /// fn SetStart(&self, node: &Node, offset: u32) -> ErrorResult { if node.is_doctype() { // Step 1. Err(Error::InvalidNodeType) } else if offset > node.len() { // Step 2. Err(Error::IndexSize) } else { // Step 3. self.set_start(node, offset); if !(self.start() <= self.end()) { // Step 4. self.set_end(node, offset); } Ok(()) } } /// fn SetEnd(&self, node: &Node, offset: u32) -> ErrorResult { if node.is_doctype() { // Step 1. Err(Error::InvalidNodeType) } else if offset > node.len() { // Step 2. Err(Error::IndexSize) } else { // Step 3. self.set_end(node, offset); if !(self.end() >= self.start()) { // Step 4. 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 { // Steps 1, 2. let parent = node.GetParentNode().ok_or(Error::InvalidNodeType)?; // Step 3. let index = node.index(); // Step 4. self.set_start(&parent, index); // Step 5. self.set_end(&parent, index + 1); Ok(()) } /// fn SelectNodeContents(&self, node: &Node) -> ErrorResult { if node.is_doctype() { // Step 1. return Err(Error::InvalidNodeType); } // Step 2. let length = node.len(); // Step 3. self.set_start(node, 0); // Step 4. self.set_end(node, length); Ok(()) } /// fn CompareBoundaryPoints(&self, how: u16, other: &Range) -> Fallible { if how > RangeConstants::END_TO_START { // Step 1. 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 { // Step 2. return Err(Error::WrongDocument); } // Step 3. 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!(), }; // step 4. match this_point.partial_cmp(other_point).unwrap() { Ordering::Less => Ok(-1), Ordering::Equal => Ok(0), Ordering::Greater => Ok(1), } } /// fn CloneRange(&self) -> DomRoot { 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(), ) } /// fn IsPointInRange(&self, node: &Node, offset: u32) -> Fallible { match self.compare_point(node, offset) { Ok(Ordering::Less) => Ok(false), Ok(Ordering::Equal) => Ok(true), Ok(Ordering::Greater) => Ok(false), Err(Error::WrongDocument) => { // Step 2. Ok(false) }, Err(error) => Err(error), } } /// fn ComparePoint(&self, node: &Node, offset: u32) -> Fallible { 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 { // Step 1. return false; } let parent = match node.GetParentNode() { Some(parent) => parent, None => { // Step 3. return true; }, }; // Step 4. let offset = node.index(); // Step 5. 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) -> Fallible> { // Step 3. let start_node = self.start_container(); let start_offset = self.start_offset(); let end_node = self.end_container(); let end_offset = self.end_offset(); // Step 1. let fragment = DocumentFragment::new(&start_node.owner_doc()); // Step 2. if self.start() == self.end() { return Ok(fragment); } if end_node == start_node { if let Some(cdata) = start_node.downcast::() { // Steps 4.1-2. let data = cdata .SubstringData(start_offset, end_offset - start_offset) .unwrap(); let clone = cdata.clone_with_data(data, &start_node.owner_doc()); // Step 4.3. fragment.upcast::().AppendChild(&clone)?; // Step 4.4 return Ok(fragment); } } // Steps 5-12. let (first_contained_child, last_contained_child, contained_children) = self.contained_children()?; if let Some(child) = first_contained_child { // Step 13. if let Some(cdata) = child.downcast::() { assert!(child == start_node); // Steps 13.1-2. let data = cdata .SubstringData(start_offset, start_node.len() - start_offset) .unwrap(); let clone = cdata.clone_with_data(data, &start_node.owner_doc()); // Step 13.3. fragment.upcast::().AppendChild(&clone)?; } else { // Step 14.1. let clone = child.CloneNode(/* deep */ false)?; // Step 14.2. fragment.upcast::().AppendChild(&clone)?; // Step 14.3. let subrange = Range::new( &clone.owner_doc(), &start_node, start_offset, &child, child.len(), ); // Step 14.4. let subfragment = subrange.CloneContents()?; // Step 14.5. clone.AppendChild(subfragment.upcast())?; } } // Step 15. for child in contained_children { // Step 15.1. let clone = child.CloneNode(/* deep */ true)?; // Step 15.2. fragment.upcast::().AppendChild(&clone)?; } if let Some(child) = last_contained_child { // Step 16. if let Some(cdata) = child.downcast::() { assert!(child == end_node); // Steps 16.1-2. let data = cdata.SubstringData(0, end_offset).unwrap(); let clone = cdata.clone_with_data(data, &start_node.owner_doc()); // Step 16.3. fragment.upcast::().AppendChild(&clone)?; } else { // Step 17.1. let clone = child.CloneNode(/* deep */ false)?; // Step 17.2. fragment.upcast::().AppendChild(&clone)?; // Step 17.3. let subrange = Range::new(&clone.owner_doc(), &child, 0, &end_node, end_offset); // Step 17.4. let subfragment = subrange.CloneContents()?; // Step 17.5. clone.AppendChild(subfragment.upcast())?; } } // Step 18. Ok(fragment) } /// /// fn ExtractContents(&self) -> Fallible> { // Step 3. let start_node = self.start_container(); let start_offset = self.start_offset(); let end_node = self.end_container(); let end_offset = self.end_offset(); // Step 1. let fragment = DocumentFragment::new(&start_node.owner_doc()); // Step 2. if self.collapsed() { return Ok(fragment); } if end_node == start_node { if let Some(end_data) = end_node.downcast::() { // Step 4.1. let clone = end_node.CloneNode(/* deep */ true)?; // Step 4.2. let text = end_data.SubstringData(start_offset, end_offset - start_offset); clone .downcast::() .unwrap() .SetData(text.unwrap()); // Step 4.3. fragment.upcast::().AppendChild(&clone)?; // Step 4.4. end_data.ReplaceData(start_offset, end_offset - start_offset, DOMString::new())?; // Step 4.5. return Ok(fragment); } } // Steps 5-12. let (first_contained_child, last_contained_child, contained_children) = self.contained_children()?; let (new_node, new_offset) = if start_node.is_inclusive_ancestor_of(&end_node) { // Step 13. (DomRoot::from_ref(&*start_node), start_offset) } else { // Step 14.1-2. let reference_node = start_node .ancestors() .take_while(|n| !n.is_inclusive_ancestor_of(&end_node)) .last() .unwrap_or(DomRoot::from_ref(&start_node)); // Step 14.3. ( reference_node.GetParentNode().unwrap(), reference_node.index() + 1, ) }; if let Some(child) = first_contained_child { if let Some(start_data) = child.downcast::() { assert!(child == start_node); // Step 15.1. let clone = start_node.CloneNode(/* deep */ true)?; // Step 15.2. let text = start_data.SubstringData(start_offset, start_node.len() - start_offset); clone .downcast::() .unwrap() .SetData(text.unwrap()); // Step 15.3. fragment.upcast::().AppendChild(&clone)?; // Step 15.4. start_data.ReplaceData( start_offset, start_node.len() - start_offset, DOMString::new(), )?; } else { // Step 16.1. let clone = child.CloneNode(/* deep */ false)?; // Step 16.2. fragment.upcast::().AppendChild(&clone)?; // Step 16.3. let subrange = Range::new( &clone.owner_doc(), &start_node, start_offset, &child, child.len(), ); // Step 16.4. let subfragment = subrange.ExtractContents()?; // Step 16.5. clone.AppendChild(subfragment.upcast())?; } } // Step 17. for child in contained_children { fragment.upcast::().AppendChild(&child)?; } if let Some(child) = last_contained_child { if let Some(end_data) = child.downcast::() { assert!(child == end_node); // Step 18.1. let clone = end_node.CloneNode(/* deep */ true)?; // Step 18.2. let text = end_data.SubstringData(0, end_offset); clone .downcast::() .unwrap() .SetData(text.unwrap()); // Step 18.3. fragment.upcast::().AppendChild(&clone)?; // Step 18.4. end_data.ReplaceData(0, end_offset, DOMString::new())?; } else { // Step 19.1. let clone = child.CloneNode(/* deep */ false)?; // Step 19.2. fragment.upcast::().AppendChild(&clone)?; // Step 19.3. let subrange = Range::new(&clone.owner_doc(), &child, 0, &end_node, end_offset); // Step 19.4. let subfragment = subrange.ExtractContents()?; // Step 19.5. clone.AppendChild(subfragment.upcast())?; } } // Step 20. self.SetStart(&new_node, new_offset)?; self.SetEnd(&new_node, new_offset)?; // Step 21. Ok(fragment) } /// fn Detach(&self) { // This method intentionally left blank. } /// /// fn InsertNode(&self, node: &Node) -> ErrorResult { let start_node = self.start_container(); let start_offset = self.start_offset(); // Step 1. if &*start_node == node { return Err(Error::HierarchyRequest); } match start_node.type_id() { // Handled under step 2. NodeTypeId::CharacterData(CharacterDataTypeId::Text(_)) => (), NodeTypeId::CharacterData(_) => return Err(Error::HierarchyRequest), _ => (), } // Step 2. let (reference_node, parent) = match start_node.type_id() { NodeTypeId::CharacterData(CharacterDataTypeId::Text(_)) => { // Step 3. let parent = match start_node.GetParentNode() { Some(parent) => parent, // Step 1. None => return Err(Error::HierarchyRequest), }; // Step 5. (Some(DomRoot::from_ref(&*start_node)), parent) }, _ => { // Steps 4-5. let child = start_node.ChildNodes().Item(start_offset); (child, DomRoot::from_ref(&*start_node)) }, }; // Step 6. Node::ensure_pre_insertion_validity(node, &parent, reference_node.as_deref())?; // Step 7. let split_text; let reference_node = match start_node.downcast::() { Some(text) => { split_text = text.SplitText(start_offset)?; let new_reference = DomRoot::upcast::(split_text); assert!(new_reference.GetParentNode().as_deref() == Some(&parent)); Some(new_reference) }, _ => reference_node, }; // Step 8. let reference_node = if Some(node) == reference_node.as_deref() { node.GetNextSibling() } else { reference_node }; // Step 9. node.remove_self(); // Step 10. let new_offset = reference_node .as_ref() .map_or(parent.len(), |node| node.index()); // Step 11 let new_offset = new_offset + if let NodeTypeId::DocumentFragment(_) = node.type_id() { node.len() } else { 1 }; // Step 12. Node::pre_insert(node, &parent, reference_node.as_deref())?; // Step 13. if self.collapsed() { self.set_end(&parent, new_offset); } Ok(()) } /// fn DeleteContents(&self) -> ErrorResult { // Step 1. if self.collapsed() { return Ok(()); } // Step 2. let start_node = self.start_container(); let end_node = self.end_container(); let start_offset = self.start_offset(); let end_offset = self.end_offset(); // Step 3. if start_node == end_node { if let Some(text) = start_node.downcast::() { if end_offset > start_offset { self.report_change(); } return text.ReplaceData(start_offset, end_offset - start_offset, DOMString::new()); } } // Step 4. 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) { // Step 5. (DomRoot::from_ref(&*start_node), start_offset) } else { // Step 6. fn compute_reference(start_node: &Node, end_node: &Node) -> (DomRoot, 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) }; // Step 7. if let Some(text) = start_node.downcast::() { text.ReplaceData( start_offset, start_node.len() - start_offset, DOMString::new(), ) .unwrap(); } // Step 8. for child in &*contained_children { child.remove_self(); } // Step 9. if let Some(text) = end_node.downcast::() { text.ReplaceData(0, end_offset, DOMString::new()).unwrap(); } // Step 10. self.SetStart(&new_node, new_offset).unwrap(); self.SetEnd(&new_node, new_offset).unwrap(); Ok(()) } /// fn SurroundContents(&self, new_parent: &Node) -> ErrorResult { // Step 1. 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::()) || end.inclusive_ancestors(ShadowIncluding::No) .any(|n| !n.is_inclusive_ancestor_of(&start) && !n.is::()) { return Err(Error::InvalidState); } // Step 2. match new_parent.type_id() { NodeTypeId::Document(_) | NodeTypeId::DocumentType | NodeTypeId::DocumentFragment(_) => { return Err(Error::InvalidNodeType); }, _ => (), } // Step 3. let fragment = self.ExtractContents()?; // Step 4. Node::replace_all(None, new_parent); // Step 5. self.InsertNode(new_parent)?; // Step 6. new_parent.AppendChild(fragment.upcast())?; // Step 7. self.SelectNode(new_parent) } /// fn Stringifier(&self) -> DOMString { let start_node = self.start_container(); let end_node = self.end_container(); // Step 1. let mut s = DOMString::new(); if let Some(text_node) = start_node.downcast::() { let char_data = text_node.upcast::(); // Step 2. if start_node == end_node { return char_data .SubstringData(self.start_offset(), self.end_offset() - self.start_offset()) .unwrap(); } // Step 3. s.push_str( &char_data .SubstringData( self.start_offset(), char_data.Length() - self.start_offset(), ) .unwrap(), ); } // Step 4. let ancestor = self.CommonAncestorContainer(); let iter = start_node .following_nodes(&ancestor) .filter_map(DomRoot::downcast::); for child in iter { if self.contains(child.upcast()) { s.push_str(&child.upcast::().Data()); } } // Step 5. if let Some(text_node) = end_node.downcast::() { let char_data = text_node.upcast::(); s.push_str(&char_data.SubstringData(0, self.end_offset()).unwrap()); } // Step 6. s } /// fn CreateContextualFragment(&self, fragment: DOMString) -> Fallible> { // Step 1. 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::(node).unwrap()), NodeTypeId::CharacterData(CharacterDataTypeId::Comment) | NodeTypeId::CharacterData(CharacterDataTypeId::Text(_)) => node.GetParentElement(), NodeTypeId::CharacterData(CharacterDataTypeId::ProcessingInstruction) | NodeTypeId::DocumentType => unreachable!(), NodeTypeId::Attr => unreachable!(), }; // Step 2. let element = Element::fragment_parsing_context(&owner_doc, element.as_deref()); // Step 3. let fragment_node = element.parse_fragment(fragment)?; // Step 4. for node in fragment_node .upcast::() .traverse_preorder(ShadowIncluding::No) { if let Some(script) = node.downcast::() { script.set_already_started(false); script.set_parser_inserted(false); } } // Step 5. Ok(fragment_node) } } pub struct WeakRangeVec { cell: UnsafeCell>, } impl Default for WeakRangeVec { fn default() -> Self { WeakRangeVec { cell: UnsafeCell::new(WeakRefVec::new()), } } } #[allow(unsafe_code)] impl WeakRangeVec { /// Create a new vector of weak references. pub fn new() -> Self { Self::default() } /// Whether that vector of ranges is empty. pub fn is_empty(&self) -> bool { unsafe { (*self.cell.get()).is_empty() } } /// Used for steps 2.1-2. when inserting a node. /// pub fn increase_above(&self, node: &Node, offset: u32, delta: u32) { self.map_offset_above(node, offset, |offset| offset + delta); } /// Used for steps 4-5. when removing a node. /// pub fn decrease_above(&self, node: &Node, offset: u32, delta: u32) { self.map_offset_above(node, offset, |offset| offset - delta); } /// Used for steps 2-3. when removing a node. /// 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(..)); } } /// Used for steps 7.1-2. when normalizing a node. /// 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(..)); } } /// Used for steps 7.3-4. when normalizing a node. /// 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); } }); } } /// Used for steps 8-11. when replacing character data. /// 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 } }); } /// Used for steps 7.2-3. when splitting a text node. /// 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); } }); } } /// Used for steps 7.4-5. when splitting a text node. /// 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 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) { unsafe { (*self.cell.get()).push(ref_); } } fn remove(&self, range: &Range) -> WeakRef { 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() } }