/* 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/. */ #![allow(unrooted_must_root)] use dom::bindings::codegen::Bindings::NodeBinding::NodeMethods; use dom::bindings::inheritance::Castable; use dom::bindings::js::{JS, Root}; use dom::bindings::str::DOMString; use dom::comment::Comment; use dom::document::Document; use dom::documenttype::DocumentType; use dom::element::{Element, ElementCreator}; use dom::htmlscriptelement::HTMLScriptElement; use dom::node::Node; use dom::processinginstruction::ProcessingInstruction; use dom::servoxmlparser; use dom::servoxmlparser::ServoXMLParser; use dom::text::Text; use html5ever; use msg::constellation_msg::PipelineId; use parse::Parser; use std::borrow::Cow; use string_cache::{Atom, QualName, Namespace}; use url::Url; use xml5ever::tendril::StrTendril; use xml5ever::tokenizer::{Attribute, QName}; use xml5ever::tree_builder::{NextParserState, NodeOrText, TreeSink}; impl<'a> TreeSink for servoxmlparser::Sink { type Handle = JS; fn parse_error(&mut self, msg: Cow<'static, str>) { debug!("Parse error: {}", msg); } fn get_document(&mut self) -> JS { JS::from_ref(self.document.upcast()) } fn elem_name(&self, target: &JS) -> QName { let elem = target.downcast::() .expect("tried to get name of non-Element in XML parsing"); QName { prefix: elem.prefix().as_ref().map_or(atom!(""), |p| Atom::from(&**p)), namespace_url: elem.namespace().0.clone(), local: elem.local_name().clone(), } } fn create_element(&mut self, name: QName, attrs: Vec) -> JS { let prefix = if name.prefix == atom!("") { None } else { Some(name.prefix) }; let name = QualName { ns: Namespace(name.namespace_url), local: name.local, }; let elem = Element::create(name, prefix, &*self.document, ElementCreator::ParserCreated); for attr in attrs { let name = QualName { ns: Namespace(attr.name.namespace_url), local: attr.name.local, }; elem.set_attribute_from_parser(name, DOMString::from(String::from(attr.value)), None); } JS::from_ref(elem.upcast()) } fn create_comment(&mut self, text: StrTendril) -> JS { let comment = Comment::new(DOMString::from(String::from(text)), &*self.document); JS::from_ref(comment.upcast()) } fn append(&mut self, parent: JS, child: NodeOrText>) { let child = match child { NodeOrText::AppendNode(n) => Root::from_ref(&*n), NodeOrText::AppendText(t) => { let s: String = t.into(); let text = Text::new(DOMString::from(s), &self.document); Root::upcast(text) } }; assert!(parent.AppendChild(child.r()).is_ok()); } fn append_doctype_to_document(&mut self, name: StrTendril, public_id: StrTendril, system_id: StrTendril) { let doc = &*self.document; let doctype = DocumentType::new( DOMString::from(String::from(name)), Some(DOMString::from(String::from(public_id))), Some(DOMString::from(String::from(system_id))), doc); doc.upcast::().AppendChild(doctype.upcast()).expect("Appending failed"); } fn create_pi(&mut self, target: StrTendril, data: StrTendril) -> JS { let doc = &*self.document; let pi = ProcessingInstruction::new( DOMString::from(String::from(target)), DOMString::from(String::from(data)), doc); JS::from_ref(pi.upcast()) } fn mark_script_already_started(&mut self, node: Self::Handle) { let script = node.downcast::(); if let Some(script) = script { script.set_already_started(true); } } fn complete_script(&mut self, node: Self::Handle) -> NextParserState { let script = node.downcast::(); if let Some(script) = script { return match script.prepare() { html5ever::tree_builder::NextParserState::Continue => NextParserState::Continue, html5ever::tree_builder::NextParserState::Suspend => NextParserState::Suspend }; } NextParserState::Continue } } pub enum ParseContext { Owner(Option) } pub fn parse_xml(document: &Document, input: DOMString, url: Url, context: ParseContext) { let parser = match context { ParseContext::Owner(owner) => ServoXMLParser::new(Some(url), document, owner), }; parser.parse_chunk(String::from(input)); }