/* 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 http://mozilla.org/MPL/2.0/. */ use dom::bindings::codegen::Bindings::CharacterDataBinding::CharacterDataMethods; use dom::bindings::codegen::Bindings::NodeBinding::NodeConstants; use dom::bindings::codegen::Bindings::NodeBinding::NodeMethods; use dom::bindings::codegen::Bindings::NodeListBinding::NodeListMethods; use dom::bindings::codegen::Bindings::RangeBinding::RangeMethods; use dom::bindings::codegen::Bindings::RangeBinding::{self, RangeConstants}; use dom::bindings::codegen::Bindings::TextBinding::TextMethods; use dom::bindings::codegen::Bindings::WindowBinding::WindowMethods; use dom::bindings::error::{Error, ErrorResult, Fallible}; use dom::bindings::global::GlobalRef; use dom::bindings::inheritance::Castable; use dom::bindings::inheritance::{CharacterDataTypeId, NodeTypeId}; use dom::bindings::js::{JS, MutHeap, Root, RootedReference}; use dom::bindings::reflector::{Reflector, reflect_dom_object}; use dom::bindings::trace::RootedVec; use dom::characterdata::CharacterData; use dom::document::Document; use dom::documentfragment::DocumentFragment; use dom::node::Node; use dom::text::Text; use std::cell::Cell; use std::cmp::{Ord, Ordering, PartialEq, PartialOrd}; use util::str::DOMString; #[dom_struct] pub struct Range { reflector_: Reflector, start: BoundaryPoint, end: BoundaryPoint, } impl Range { fn new_inherited(start_container: &Node, start_offset: u32, end_container: &Node, end_offset: u32) -> Range { Range { reflector_: Reflector::new(), start: BoundaryPoint::new(start_container, start_offset), end: BoundaryPoint::new(end_container, end_offset), } } pub fn new_with_doc(document: &Document) -> Root { let root = document.upcast(); Range::new(document, root, 0, root, 0) } pub fn new(document: &Document, start_container: &Node, start_offset: u32, end_container: &Node, end_offset: u32) -> Root { reflect_dom_object(box Range::new_inherited(start_container, start_offset, end_container, end_offset), GlobalRef::Window(document.window()), RangeBinding::Wrap) } // https://dom.spec.whatwg.org/#dom-range pub fn Constructor(global: GlobalRef) -> Fallible> { let document = global.as_window().Document(); Ok(Range::new_with_doc(document.r())) } // https://dom.spec.whatwg.org/#contained fn contains(&self, node: &Node) -> bool { match (bp_position(node, 0, &self.StartContainer(), self.StartOffset()), bp_position(node, node.len(), &self.EndContainer(), self.EndOffset())) { (Some(Ordering::Greater), Some(Ordering::Less)) => true, _ => false } } // https://dom.spec.whatwg.org/#partially-contained fn partially_contains(&self, node: &Node) -> bool { self.StartContainer().inclusive_ancestors().any(|n| &*n == node) != self.EndContainer().inclusive_ancestors().any(|n| &*n == node) } // https://dom.spec.whatwg.org/#concept-range-clone fn contained_children(&self) -> Fallible<(Option>, Option>, Vec>)> { let start_node = self.StartContainer(); let end_node = self.EndContainer(); // Steps 5-6. let common_ancestor = self.CommonAncestorContainer(); let first_contained_child = if start_node.is_inclusive_ancestor_of(end_node.r()) { // 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.r()) { // 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| Range::contains(self, 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)) } // https://dom.spec.whatwg.org/#concept-range-bp-set pub fn set_start(&self, node: &Node, offset: u32) { self.start.set(node, offset); if !(self.start <= self.end) { self.end.set(node, offset); } } // https://dom.spec.whatwg.org/#concept-range-bp-set pub fn set_end(&self, node: &Node, offset: u32) { self.end.set(node, offset); if !(self.end >= self.start) { self.start.set(node, offset); } } // https://dom.spec.whatwg.org/#dom-range-comparepointnode-offset fn compare_point(&self, node: &Node, offset: u32) -> Fallible { let start_node = self.StartContainer(); let start_node_root = start_node.inclusive_ancestors().last().unwrap(); let node_root = node.inclusive_ancestors().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.StartOffset()).unwrap() { // Step 4. return Ok(Ordering::Less); } if let Ordering::Greater = bp_position(node, offset, &self.EndContainer(), self.EndOffset()).unwrap() { // Step 5. return Ok(Ordering::Greater); } // Step 6. Ok(Ordering::Equal) } } impl RangeMethods for Range { // https://dom.spec.whatwg.org/#dom-range-startcontainer fn StartContainer(&self) -> Root { self.start.node.get() } // https://dom.spec.whatwg.org/#dom-range-startoffset fn StartOffset(&self) -> u32 { self.start.offset.get() } // https://dom.spec.whatwg.org/#dom-range-endcontainer fn EndContainer(&self) -> Root { self.end.node.get() } // https://dom.spec.whatwg.org/#dom-range-endoffset fn EndOffset(&self) -> u32 { self.end.offset.get() } // https://dom.spec.whatwg.org/#dom-range-collapsed fn Collapsed(&self) -> bool { self.start == self.end } // https://dom.spec.whatwg.org/#dom-range-commonancestorcontainer fn CommonAncestorContainer(&self) -> Root { let end_container = self.EndContainer(); // Step 1. for container in self.StartContainer().inclusive_ancestors() { // Step 2. if container.is_inclusive_ancestor_of(&end_container) { // Step 3. return container; } } unreachable!(); } // https://dom.spec.whatwg.org/#dom-range-setstartnode-offset 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-4. self.set_start(node, offset); Ok(()) } } // https://dom.spec.whatwg.org/#dom-range-setendnode-offset 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-4. self.set_end(node, offset); Ok(()) } } // https://dom.spec.whatwg.org/#dom-range-setstartbeforenode fn SetStartBefore(&self, node: &Node) -> ErrorResult { let parent = try!(node.GetParentNode().ok_or(Error::InvalidNodeType)); self.SetStart(parent.r(), node.index()) } // https://dom.spec.whatwg.org/#dom-range-setstartafternode fn SetStartAfter(&self, node: &Node) -> ErrorResult { let parent = try!(node.GetParentNode().ok_or(Error::InvalidNodeType)); self.SetStart(parent.r(), node.index() + 1) } // https://dom.spec.whatwg.org/#dom-range-setendbeforenode fn SetEndBefore(&self, node: &Node) -> ErrorResult { let parent = try!(node.GetParentNode().ok_or(Error::InvalidNodeType)); self.SetEnd(parent.r(), node.index()) } // https://dom.spec.whatwg.org/#dom-range-setendafternode fn SetEndAfter(&self, node: &Node) -> ErrorResult { let parent = try!(node.GetParentNode().ok_or(Error::InvalidNodeType)); self.SetEnd(parent.r(), node.index() + 1) } // https://dom.spec.whatwg.org/#dom-range-collapsetostart fn Collapse(&self, to_start: bool) { if to_start { self.end.set(&self.StartContainer(), self.StartOffset()); } else { self.start.set(&self.EndContainer(), self.EndOffset()); } } // https://dom.spec.whatwg.org/#dom-range-selectnodenode fn SelectNode(&self, node: &Node) -> ErrorResult { // Steps 1, 2. let parent = try!(node.GetParentNode().ok_or(Error::InvalidNodeType)); // Step 3. let index = node.index(); // Step 4. self.start.set(&parent, index); // Step 5. self.end.set(&parent, index + 1); Ok(()) } // https://dom.spec.whatwg.org/#dom-range-selectnodecontentsnode 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.start.set(node, 0); // Step 4. self.end.set(node, length); Ok(()) } // https://dom.spec.whatwg.org/#dom-range-compareboundarypointshow-sourcerange fn CompareBoundaryPoints(&self, how: u16, other: &Range) -> Fallible { if how > RangeConstants::END_TO_START { // Step 1. return Err(Error::NotSupported); } let this_root = self.StartContainer().inclusive_ancestors().last().unwrap(); let other_root = other.StartContainer().inclusive_ancestors().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), } } // https://dom.spec.whatwg.org/#dom-range-clonerange fn CloneRange(&self) -> Root { let start_node = self.StartContainer(); let owner_doc = start_node.owner_doc(); Range::new(&owner_doc, &start_node, self.StartOffset(), &self.EndContainer(), self.EndOffset()) } // https://dom.spec.whatwg.org/#dom-range-ispointinrangenode-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), } } // https://dom.spec.whatwg.org/#dom-range-comparepointnode-offset 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, } }) } // https://dom.spec.whatwg.org/#dom-range-intersectsnode fn IntersectsNode(&self, node: &Node) -> bool { let start_node = self.StartContainer(); let start_node_root = self.StartContainer().inclusive_ancestors().last().unwrap(); let node_root = node.inclusive_ancestors().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.r(), offset + 1, &start_node, self.StartOffset()).unwrap() && Ordering::Less == bp_position(parent.r(), offset, &self.EndContainer(), self.EndOffset()).unwrap() } // https://dom.spec.whatwg.org/#dom-range-clonecontents // https://dom.spec.whatwg.org/#concept-range-clone fn CloneContents(&self) -> Fallible> { // Step 3. let start_node = self.StartContainer(); let start_offset = self.StartOffset(); let end_node = self.EndContainer(); let end_offset = self.EndOffset(); // Step 1. let fragment = DocumentFragment::new(start_node.owner_doc().r()); // Step 2. if self.start == self.end { return Ok(fragment); } if end_node == start_node { if let Some(text) = start_node.downcast::() { // Step 4.1. let clone = start_node.CloneNode(true); // Step 4.2. let text = text.SubstringData(start_offset, end_offset - start_offset); clone.downcast::().unwrap().SetData(text.unwrap()); // Step 4.3. try!(fragment.upcast::().AppendChild(&clone)); // Step 4.4 return Ok(fragment); } } // Steps 5-12. let (first_contained_child, last_contained_child, contained_children) = try!(self.contained_children()); if let Some(child) = first_contained_child { // Step 13. if let Some(text) = child.downcast::() { assert!(child == start_node); // Step 13.1. let clone = start_node.CloneNode(true); // CharacterData has no children. // Step 13.2 let text = text.SubstringData(start_offset, start_node.len() - start_offset); clone.downcast::().unwrap().SetData(text.unwrap()); // Step 13.3. try!(fragment.upcast::().AppendChild(&clone)); } else { // Step 14.1. let clone = child.CloneNode(false); // Step 14.2. try!(fragment.upcast::().AppendChild(&clone)); // Step 14.3. let subrange = Range::new(clone.owner_doc().r(), start_node.r(), start_offset, child.r(), child.len()); // Step 14.4. let subfragment = try!(subrange.CloneContents()); // Step 14.5. try!(clone.AppendChild(subfragment.upcast())); } } // Step 15. for child in contained_children { // Step 15.1. let clone = child.CloneNode(true); // Step 15.2. try!(fragment.upcast::().AppendChild(&clone)); } if let Some(child) = last_contained_child { // Step 16. if let Some(text) = child.downcast::() { assert!(child == end_node); // Step 16.1. let clone = end_node.CloneNode(true); // CharacterData has no children. // Step 16.2. let text = text.SubstringData(0, end_offset); clone.downcast::().unwrap().SetData(text.unwrap()); // Step 16.3. try!(fragment.upcast::().AppendChild(&clone)); } else { // Step 17.1. let clone = child.CloneNode(false); // Step 17.2. try!(fragment.upcast::().AppendChild(&clone)); // Step 17.3. let subrange = Range::new(clone.owner_doc().r(), child.r(), 0, end_node.r(), end_offset); // Step 17.4. let subfragment = try!(subrange.CloneContents()); // Step 17.5. try!(clone.AppendChild(subfragment.upcast())); } } // Step 18. Ok(fragment) } // https://dom.spec.whatwg.org/#dom-range-extractcontents // https://dom.spec.whatwg.org/#concept-range-extract fn ExtractContents(&self) -> Fallible> { // Step 3. let start_node = self.StartContainer(); let start_offset = self.StartOffset(); let end_node = self.EndContainer(); let end_offset = self.EndOffset(); // Step 1. let fragment = DocumentFragment::new(start_node.owner_doc().r()); // 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(true); // Step 4.2. let text = end_data.SubstringData(start_offset, end_offset - start_offset); clone.downcast::().unwrap().SetData(text.unwrap()); // Step 4.3. try!(fragment.upcast::().AppendChild(&clone)); // Step 4.4. try!(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) = try!(self.contained_children()); let (new_node, new_offset) = if start_node.is_inclusive_ancestor_of(end_node.r()) { // Step 13. (Root::from_ref(start_node.r()), start_offset) } else { // Step 14.1-2. let reference_node = start_node.ancestors() .find(|n| n.is_inclusive_ancestor_of(end_node.r())) .unwrap(); // 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(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. try!(fragment.upcast::().AppendChild(&clone)); // Step 15.4. try!(start_data.ReplaceData(start_offset, start_node.len() - start_offset, DOMString::new())); } else { // Step 16.1. let clone = child.CloneNode(false); // Step 16.2. try!(fragment.upcast::().AppendChild(&clone)); // Step 16.3. let subrange = Range::new(clone.owner_doc().r(), start_node.r(), start_offset, child.r(), child.len()); // Step 16.4. let subfragment = try!(subrange.ExtractContents()); // Step 16.5. try!(clone.AppendChild(subfragment.upcast())); } } // Step 17. for child in contained_children { try!(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(true); // Step 18.2. let text = end_data.SubstringData(0, end_offset); clone.downcast::().unwrap().SetData(text.unwrap()); // Step 18.3. try!(fragment.upcast::().AppendChild(&clone)); // Step 18.4. try!(end_data.ReplaceData(0, end_offset, DOMString::new())); } else { // Step 19.1. let clone = child.CloneNode(false); // Step 19.2. try!(fragment.upcast::().AppendChild(&clone)); // Step 19.3. let subrange = Range::new(clone.owner_doc().r(), child.r(), 0, end_node.r(), end_offset); // Step 19.4. let subfragment = try!(subrange.ExtractContents()); // Step 19.5. try!(clone.AppendChild(subfragment.upcast())); } } // Step 20. try!(self.SetStart(new_node.r(), new_offset)); try!(self.SetEnd(new_node.r(), new_offset)); // Step 21. Ok(fragment) } // https://dom.spec.whatwg.org/#dom-range-detach fn Detach(&self) { // This method intentionally left blank. } // https://dom.spec.whatwg.org/#dom-range-insertnode // https://dom.spec.whatwg.org/#concept-range-insert fn InsertNode(&self, node: &Node) -> ErrorResult { let start_node = self.StartContainer(); let start_offset = self.StartOffset(); // Step 1. 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) = if 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(Root::from_ref(start_node.r())), parent) } else { // Steps 4-5. let child = start_node.ChildNodes().Item(start_offset); (child, Root::from_ref(start_node.r())) }; // Step 6. try!(Node::ensure_pre_insertion_validity(node, parent.r(), reference_node.r())); // Step 7. let split_text; let reference_node = match start_node.downcast::() { Some(text) => { split_text = try!(text.SplitText(start_offset)); let new_reference = Root::upcast::(split_text); assert!(new_reference.GetParentNode().r() == Some(parent.r())); Some(new_reference) }, _ => reference_node }; // Step 8. let reference_node = if Some(node) == reference_node.r() { node.GetNextSibling() } else { reference_node }; // Step 9. node.remove_self(); // Step 10. let new_offset = reference_node.r().map_or(parent.len(), |node| node.index()); // Step 11 let new_offset = new_offset + if node.type_id() == NodeTypeId::DocumentFragment { node.len() } else { 1 }; // Step 12. try!(Node::pre_insert(node, parent.r(), reference_node.r())); // Step 13. if self.Collapsed() { self.set_end(parent.r(), new_offset); } Ok(()) } // https://dom.spec.whatwg.org/#dom-range-deletecontents fn DeleteContents(&self) -> ErrorResult { // Step 1. if self.Collapsed() { return Ok(()); } // Step 2. let start_node = self.StartContainer(); let end_node = self.EndContainer(); let start_offset = self.StartOffset(); let end_offset = self.EndOffset(); // Step 3. if start_node == end_node { if let Some(text) = start_node.downcast::() { return text.ReplaceData(start_offset, end_offset - start_offset, DOMString::from("")); } } // Step 4. let mut contained_children: RootedVec> = RootedVec::new(); let ancestor = self.CommonAncestorContainer(); for child in start_node.following_nodes(ancestor.r()) { if self.contains(child.r()) && !contained_children.contains(&JS::from_ref(child.GetParentNode().unwrap().r())) { contained_children.push(JS::from_ref(child.r())); } } let (new_node, new_offset) = if start_node.is_inclusive_ancestor_of(end_node.r()) { // Step 5. (Root::from_ref(start_node.r()), start_offset) } else { // Step 6. fn compute_reference(start_node: &Node, end_node: &Node) -> (Root, u32) { let mut reference_node = Root::from_ref(start_node); while let Some(parent) = reference_node.GetParentNode() { if parent.is_inclusive_ancestor_of(end_node) { return (parent, reference_node.index()) } reference_node = parent; } panic!() } compute_reference(start_node.r(), end_node.r()) }; // Step 7. if let Some(text) = start_node.downcast::() { try!(text.ReplaceData(start_offset, start_node.len() - start_offset, DOMString::from(""))); } // Step 8. for child in contained_children.r() { child.remove_self(); } // Step 9. if let Some(text) = end_node.downcast::() { try!(text.ReplaceData(0, end_offset, DOMString::from(""))); } // Step 10. try!(self.SetStart(new_node.r(), new_offset)); self.SetEnd(new_node.r(), new_offset) } // https://dom.spec.whatwg.org/#dom-range-surroundcontents fn SurroundContents(&self, new_parent: &Node) -> ErrorResult { // Step 1. let start = self.StartContainer(); let end = self.EndContainer(); if start.inclusive_ancestors().any(|n| !n.is_inclusive_ancestor_of(end.r()) && !n.is::()) || end.inclusive_ancestors().any(|n| !n.is_inclusive_ancestor_of(start.r()) && !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 = try!(self.ExtractContents()); // Step 4. Node::replace_all(None, new_parent); // Step 5. try!(self.InsertNode(new_parent)); // Step 6. try!(new_parent.AppendChild(fragment.upcast())); // Step 7. self.SelectNode(new_parent) } } #[derive(JSTraceable)] #[must_root] #[privatize] #[derive(HeapSizeOf)] pub struct BoundaryPoint { node: MutHeap>, offset: Cell, } impl BoundaryPoint { fn new(node: &Node, offset: u32) -> BoundaryPoint { debug_assert!(!node.is_doctype()); debug_assert!(offset <= node.len()); BoundaryPoint { node: MutHeap::new(node), offset: Cell::new(offset), } } fn set(&self, node: &Node, offset: u32) { debug_assert!(!node.is_doctype()); debug_assert!(offset <= node.len()); self.node.set(node); self.offset.set(offset); } } #[allow(unrooted_must_root)] impl PartialOrd for BoundaryPoint { fn partial_cmp(&self, other: &Self) -> Option { bp_position(&self.node.get(), self.offset.get(), &other.node.get(), other.offset.get()) } } #[allow(unrooted_must_root)] impl PartialEq for BoundaryPoint { fn eq(&self, other: &Self) -> bool { self.node.get() == other.node.get() && self.offset.get() == other.offset.get() } } // https://dom.spec.whatwg.org/#concept-range-bp-position fn bp_position(a_node: &Node, a_offset: u32, b_node: &Node, b_offset: u32) -> Option { if a_node as *const Node == b_node as *const Node { // Step 1. return Some(a_offset.cmp(&b_offset)); } let position = b_node.CompareDocumentPosition(a_node); if position & NodeConstants::DOCUMENT_POSITION_DISCONNECTED != 0 { // No order is defined for nodes not in the same tree. None } else if position & NodeConstants::DOCUMENT_POSITION_FOLLOWING != 0 { // Step 2. match bp_position(b_node, b_offset, a_node, a_offset).unwrap() { Ordering::Less => Some(Ordering::Greater), Ordering::Greater => Some(Ordering::Less), Ordering::Equal => unreachable!(), } } else if position & NodeConstants::DOCUMENT_POSITION_CONTAINS != 0 { // Step 3-1, 3-2. let mut b_ancestors = b_node.inclusive_ancestors(); let child = b_ancestors.find(|child| { child.GetParentNode().unwrap().r() == a_node }).unwrap(); // Step 3-3. if child.index() < a_offset { Some(Ordering::Greater) } else { // Step 4. Some(Ordering::Less) } } else { // Step 4. Some(Ordering::Less) } }