mirror of
https://github.com/servo/servo.git
synced 2025-06-06 16:45:39 +00:00
1235 lines
43 KiB
Rust
1235 lines
43 KiB
Rust
/* 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::{JSTraceable, RootedVec};
|
|
use dom::bindings::weakref::{WeakRef, WeakRefVec};
|
|
use dom::characterdata::CharacterData;
|
|
use dom::document::Document;
|
|
use dom::documentfragment::DocumentFragment;
|
|
use dom::node::{Node, UnbindContext};
|
|
use dom::text::Text;
|
|
use heapsize::HeapSizeOf;
|
|
use js::jsapi::JSTracer;
|
|
use std::cell::{Cell, UnsafeCell};
|
|
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<Range> {
|
|
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<Range> {
|
|
let range = reflect_dom_object(box Range::new_inherited(start_container, start_offset,
|
|
end_container, end_offset),
|
|
GlobalRef::Window(document.window()),
|
|
RangeBinding::Wrap);
|
|
start_container.ranges().push(WeakRef::new(&range));
|
|
if start_container != end_container {
|
|
end_container.ranges().push(WeakRef::new(&range));
|
|
}
|
|
range
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range
|
|
pub fn Constructor(global: GlobalRef) -> Fallible<Root<Range>> {
|
|
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<Root<Node>>,
|
|
Option<Root<Node>>,
|
|
Vec<Root<Node>>)> {
|
|
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<Root<Node>> =
|
|
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))
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#concept-range-bp-set
|
|
fn set_start(&self, node: &Node, offset: u32) {
|
|
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.StartContainer().ranges().remove(self);
|
|
} else {
|
|
node.ranges().push(self.StartContainer().ranges().remove(self));
|
|
}
|
|
}
|
|
self.start.set(node, offset);
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#concept-range-bp-set
|
|
fn set_end(&self, node: &Node, offset: u32) {
|
|
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.EndContainer().ranges().remove(self);
|
|
} else {
|
|
node.ranges().push(self.EndContainer().ranges().remove(self));
|
|
}
|
|
}
|
|
self.end.set(node, offset);
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-comparepointnode-offset
|
|
fn compare_point(&self, node: &Node, offset: u32) -> Fallible<Ordering> {
|
|
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<Node> {
|
|
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<Node> {
|
|
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<Node> {
|
|
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-setstart
|
|
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(())
|
|
}
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-setend
|
|
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(())
|
|
}
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-setstartbefore
|
|
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-setstartafter
|
|
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-setendbefore
|
|
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-setendafter
|
|
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-collapse
|
|
fn Collapse(&self, to_start: bool) {
|
|
if to_start {
|
|
self.set_end(&self.StartContainer(), self.StartOffset());
|
|
} else {
|
|
self.set_start(&self.EndContainer(), self.EndOffset());
|
|
}
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-selectnode
|
|
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.set_start(&parent, index);
|
|
// Step 5.
|
|
self.set_end(&parent, index + 1);
|
|
Ok(())
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-selectnodecontents
|
|
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(())
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-compareboundarypoints
|
|
fn CompareBoundaryPoints(&self, how: u16, other: &Range)
|
|
-> Fallible<i16> {
|
|
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<Range> {
|
|
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-ispointinrange
|
|
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) => {
|
|
// Step 2.
|
|
Ok(false)
|
|
}
|
|
Err(error) => Err(error),
|
|
}
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-comparepoint
|
|
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,
|
|
}
|
|
})
|
|
}
|
|
|
|
// 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<Root<DocumentFragment>> {
|
|
// 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(cdata) = start_node.downcast::<CharacterData>() {
|
|
// 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.
|
|
try!(fragment.upcast::<Node>().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(cdata) = child.downcast::<CharacterData>() {
|
|
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.
|
|
try!(fragment.upcast::<Node>().AppendChild(&clone));
|
|
} else {
|
|
// Step 14.1.
|
|
let clone = child.CloneNode(false);
|
|
// Step 14.2.
|
|
try!(fragment.upcast::<Node>().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::<Node>().AppendChild(&clone));
|
|
}
|
|
|
|
if let Some(child) = last_contained_child {
|
|
// Step 16.
|
|
if let Some(cdata) = child.downcast::<CharacterData>() {
|
|
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.
|
|
try!(fragment.upcast::<Node>().AppendChild(&clone));
|
|
} else {
|
|
// Step 17.1.
|
|
let clone = child.CloneNode(false);
|
|
// Step 17.2.
|
|
try!(fragment.upcast::<Node>().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<Root<DocumentFragment>> {
|
|
// 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::<CharacterData>() {
|
|
// 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::<CharacterData>().unwrap().SetData(text.unwrap());
|
|
// Step 4.3.
|
|
try!(fragment.upcast::<Node>().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()
|
|
.take_while(|n| !n.is_inclusive_ancestor_of(&end_node))
|
|
.last()
|
|
.unwrap_or(Root::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::<CharacterData>() {
|
|
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::<CharacterData>().unwrap().SetData(text.unwrap());
|
|
// Step 15.3.
|
|
try!(fragment.upcast::<Node>().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::<Node>().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::<Node>().AppendChild(&child));
|
|
}
|
|
|
|
if let Some(child) = last_contained_child {
|
|
if let Some(end_data) = child.downcast::<CharacterData>() {
|
|
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::<CharacterData>().unwrap().SetData(text.unwrap());
|
|
// Step 18.3.
|
|
try!(fragment.upcast::<Node>().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::<Node>().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.
|
|
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) =
|
|
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::<Text>() {
|
|
Some(text) => {
|
|
split_text = try!(text.SplitText(start_offset));
|
|
let new_reference = Root::upcast::<Node>(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::<CharacterData>() {
|
|
return text.ReplaceData(start_offset,
|
|
end_offset - start_offset,
|
|
DOMString::new());
|
|
}
|
|
}
|
|
|
|
// Step 4.
|
|
let mut contained_children: RootedVec<JS<Node>> = RootedVec::new();
|
|
let ancestor = self.CommonAncestorContainer();
|
|
|
|
let mut iter = start_node.following_nodes(ancestor.r());
|
|
|
|
let mut next = iter.next();
|
|
while let Some(child) = next {
|
|
if self.contains(child.r()) {
|
|
contained_children.push(JS::from_ref(child.r()));
|
|
next = iter.next_skipping_children();
|
|
} else {
|
|
next = iter.next();
|
|
}
|
|
}
|
|
|
|
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<Node>, 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() + 1)
|
|
}
|
|
reference_node = parent;
|
|
}
|
|
unreachable!()
|
|
}
|
|
|
|
compute_reference(start_node.r(), end_node.r())
|
|
};
|
|
|
|
// Step 7.
|
|
if let Some(text) = start_node.downcast::<CharacterData>() {
|
|
text.ReplaceData(start_offset,
|
|
start_node.len() - start_offset,
|
|
DOMString::new()).unwrap();
|
|
}
|
|
|
|
// Step 8.
|
|
for child in contained_children.r() {
|
|
child.remove_self();
|
|
}
|
|
|
|
// Step 9.
|
|
if let Some(text) = end_node.downcast::<CharacterData>() {
|
|
text.ReplaceData(0, end_offset, DOMString::new()).unwrap();
|
|
}
|
|
|
|
// Step 10.
|
|
self.SetStart(new_node.r(), new_offset).unwrap();
|
|
self.SetEnd(new_node.r(), new_offset).unwrap();
|
|
Ok(())
|
|
}
|
|
|
|
// 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::<Text>()) ||
|
|
end.inclusive_ancestors().any(|n| !n.is_inclusive_ancestor_of(start.r()) && !n.is::<Text>()) {
|
|
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)
|
|
}
|
|
|
|
// https://dom.spec.whatwg.org/#dom-range-stringifier
|
|
fn Stringifier(&self) -> DOMString {
|
|
let start_node = self.StartContainer();
|
|
let end_node = self.EndContainer();
|
|
|
|
// Step 1.
|
|
let mut s = DOMString::new();
|
|
|
|
if let Some(text_node) = start_node.downcast::<Text>() {
|
|
let char_data = text_node.upcast::<CharacterData>();
|
|
|
|
// Step 2.
|
|
if start_node == end_node {
|
|
return char_data.SubstringData(self.StartOffset(),
|
|
self.EndOffset() - self.StartOffset()).unwrap();
|
|
}
|
|
|
|
// Step 3.
|
|
s.push_str(&*char_data.SubstringData(self.StartOffset(),
|
|
char_data.Length() - self.StartOffset()).unwrap());
|
|
}
|
|
|
|
// Step 4.
|
|
let ancestor = self.CommonAncestorContainer();
|
|
let mut iter = start_node.following_nodes(ancestor.r())
|
|
.filter_map(Root::downcast::<Text>);
|
|
|
|
while let Some(child) = iter.next() {
|
|
if self.contains(child.upcast()) {
|
|
s.push_str(&*child.upcast::<CharacterData>().Data());
|
|
}
|
|
}
|
|
|
|
// Step 5.
|
|
if let Some(text_node) = end_node.downcast::<Text>() {
|
|
let char_data = text_node.upcast::<CharacterData>();
|
|
s.push_str(&*char_data.SubstringData(0, self.EndOffset()).unwrap());
|
|
}
|
|
|
|
// Step 6.
|
|
s
|
|
}
|
|
}
|
|
|
|
#[derive(JSTraceable)]
|
|
#[must_root]
|
|
#[privatize]
|
|
#[derive(HeapSizeOf)]
|
|
pub struct BoundaryPoint {
|
|
node: MutHeap<JS<Node>>,
|
|
offset: Cell<u32>,
|
|
}
|
|
|
|
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),
|
|
}
|
|
}
|
|
|
|
pub fn set(&self, node: &Node, offset: u32) {
|
|
self.node.set(node);
|
|
self.set_offset(offset);
|
|
}
|
|
|
|
pub fn set_offset(&self, offset: u32) {
|
|
self.offset.set(offset);
|
|
}
|
|
}
|
|
|
|
#[allow(unrooted_must_root)]
|
|
impl PartialOrd for BoundaryPoint {
|
|
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
|
|
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<Ordering> {
|
|
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)
|
|
}
|
|
}
|
|
|
|
pub struct WeakRangeVec {
|
|
cell: UnsafeCell<WeakRefVec<Range>>,
|
|
}
|
|
|
|
#[allow(unsafe_code)]
|
|
impl WeakRangeVec {
|
|
/// Create a new vector of weak references.
|
|
pub fn new() -> Self {
|
|
WeakRangeVec { cell: UnsafeCell::new(WeakRefVec::new()) }
|
|
}
|
|
|
|
/// 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.
|
|
/// https://dom.spec.whatwg.org/#concept-node-insert
|
|
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.
|
|
/// https://dom.spec.whatwg.org/#concept-node-remove
|
|
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.
|
|
/// https://dom.spec.whatwg.org/#concept-node-remove
|
|
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 mut 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.start.set(context.parent, offset);
|
|
}
|
|
if &range.end.node == child {
|
|
range.end.set(context.parent, offset);
|
|
}
|
|
});
|
|
|
|
(*context.parent.ranges().cell.get()).extend(ranges.drain(..));
|
|
}
|
|
}
|
|
|
|
/// Used for steps 7.1-2. when normalizing a node.
|
|
/// https://dom.spec.whatwg.org/#dom-node-normalize
|
|
pub fn drain_to_preceding_text_sibling(&self, node: &Node, sibling: &Node, length: u32) {
|
|
if self.is_empty() {
|
|
return;
|
|
}
|
|
|
|
unsafe {
|
|
let mut 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.start.set(sibling, range.StartOffset() + length);
|
|
}
|
|
if &range.end.node == node {
|
|
range.end.set(sibling, range.EndOffset() + length);
|
|
}
|
|
});
|
|
|
|
(*sibling.ranges().cell.get()).extend(ranges.drain(..));
|
|
}
|
|
}
|
|
|
|
/// Used for steps 7.3-4. when normalizing a node.
|
|
/// https://dom.spec.whatwg.org/#dom-node-normalize
|
|
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.StartOffset() == offset;
|
|
let move_end = node_is_end && range.EndOffset() == offset;
|
|
|
|
let remove_from_node = move_start && move_end ||
|
|
move_start && !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.start.set(child, new_offset);
|
|
}
|
|
if move_end {
|
|
range.end.set(child, new_offset);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/// Used for steps 8-11. when replacing character data.
|
|
/// https://dom.spec.whatwg.org/#concept-cd-replace
|
|
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.
|
|
/// https://dom.spec.whatwg.org/#concept-text-split
|
|
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.StartOffset();
|
|
let end_offset = range.EndOffset();
|
|
|
|
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 ||
|
|
move_start && !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.start.set(sibling, start_offset - offset);
|
|
}
|
|
if move_end {
|
|
range.end.set(sibling, end_offset - offset);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/// Used for steps 7.4-5. when splitting a text node.
|
|
/// https://dom.spec.whatwg.org/#concept-text-split
|
|
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.StartOffset() {
|
|
range.start.set_offset(offset + 1);
|
|
}
|
|
if &range.end.node == node && offset == range.EndOffset() {
|
|
range.end.set_offset(offset + 1);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/// Used for steps 9.1-2. when splitting a text node.
|
|
/// https://dom.spec.whatwg.org/#concept-text-split
|
|
pub fn clamp_above(&self, node: &Node, offset: u32) {
|
|
self.map_offset_above(node, offset, |_| offset);
|
|
}
|
|
|
|
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.StartOffset();
|
|
if &range.start.node == node && start_offset > offset {
|
|
range.start.set_offset(f(start_offset));
|
|
}
|
|
let end_offset = range.EndOffset();
|
|
if &range.end.node == node && end_offset > offset {
|
|
range.end.set_offset(f(end_offset));
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
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 HeapSizeOf for WeakRangeVec {
|
|
fn heap_size_of_children(&self) -> usize {
|
|
unsafe { (*self.cell.get()).heap_size_of_children() }
|
|
}
|
|
}
|
|
|
|
#[allow(unsafe_code)]
|
|
impl JSTraceable for WeakRangeVec {
|
|
fn trace(&self, _: *mut JSTracer) {
|
|
unsafe { (*self.cell.get()).retain_alive() }
|
|
}
|
|
}
|