use dom_struct::dom_struct;
use js::rust::HandleObject;
use crate::dom::bindings::codegen::Bindings::CharacterDataBinding::CharacterDataMethods;
use crate::dom::bindings::codegen::Bindings::DocumentBinding::DocumentMethods;
use crate::dom::bindings::codegen::Bindings::NodeBinding::NodeMethods;
use crate::dom::bindings::codegen::Bindings::TextBinding::TextMethods;
use crate::dom::bindings::codegen::Bindings::WindowBinding::WindowMethods;
use crate::dom::bindings::error::{Error, Fallible};
use crate::dom::bindings::inheritance::Castable;
use crate::dom::bindings::root::DomRoot;
use crate::dom::bindings::str::DOMString;
use crate::dom::characterdata::CharacterData;
use crate::dom::document::Document;
use crate::dom::node::Node;
use crate::dom::window::Window;
use crate::script_runtime::CanGc;
#[dom_struct]
pub struct Text {
characterdata: CharacterData,
}
impl Text {
pub fn new_inherited(text: DOMString, document: &Document) -> Text {
Text {
characterdata: CharacterData::new_inherited(text, document),
}
}
pub fn new(text: DOMString, document: &Document, can_gc: CanGc) -> DomRoot<Text> {
Self::new_with_proto(text, document, None, can_gc)
}
fn new_with_proto(
text: DOMString,
document: &Document,
proto: Option<HandleObject>,
can_gc: CanGc,
) -> DomRoot<Text> {
Node::reflect_node_with_proto(
Box::new(Text::new_inherited(text, document)),
document,
proto,
can_gc,
)
}
}
impl TextMethods<crate::DomTypeHolder> for Text {
fn Constructor(
window: &Window,
proto: Option<HandleObject>,
can_gc: CanGc,
text: DOMString,
) -> Fallible<DomRoot<Text>> {
let document = window.Document();
Ok(Text::new_with_proto(text, &document, proto, can_gc))
}
fn SplitText(&self, offset: u32, can_gc: CanGc) -> Fallible<DomRoot<Text>> {
let cdata = self.upcast::<CharacterData>();
let length = cdata.Length();
if offset > length {
return Err(Error::IndexSize);
}
let count = length - offset;
let new_data = cdata.SubstringData(offset, count).unwrap();
let node = self.upcast::<Node>();
let owner_doc = node.owner_doc();
let new_node = owner_doc.CreateTextNode(new_data, can_gc);
let parent = node.GetParentNode();
if let Some(ref parent) = parent {
parent
.InsertBefore(new_node.upcast(), node.GetNextSibling().as_deref())
.unwrap();
node.ranges()
.move_to_following_text_sibling_above(node, offset, new_node.upcast());
parent.ranges().increment_at(parent, node.index() + 1);
}
cdata.DeleteData(offset, count).unwrap();
Ok(new_node)
}
fn WholeText(&self) -> DOMString {
let first = self
.upcast::<Node>()
.inclusively_preceding_siblings()
.take_while(|node| node.is::<Text>())
.last()
.unwrap();
let nodes = first
.inclusively_following_siblings()
.take_while(|node| node.is::<Text>());
let mut text = String::new();
for ref node in nodes {
let cdata = node.downcast::<CharacterData>().unwrap();
text.push_str(&cdata.data());
}
DOMString::from(text)
}
}