/* 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::DocumentBinding::DocumentMethods; use dom::bindings::codegen::Bindings::HTMLTemplateElementBinding; use dom::bindings::codegen::Bindings::HTMLTemplateElementBinding::HTMLTemplateElementMethods; use dom::bindings::codegen::Bindings::NodeBinding::NodeMethods; use dom::bindings::codegen::InheritTypes::{ElementTypeId, EventTargetTypeId}; use dom::bindings::codegen::InheritTypes::{HTMLElementCast, HTMLElementTypeId}; use dom::bindings::codegen::InheritTypes::{HTMLTemplateElementCast, HTMLTemplateElementDerived}; use dom::bindings::codegen::InheritTypes::{NodeCast, NodeTypeId}; use dom::bindings::js::{JS, MutNullableHeap, Root}; use dom::document::Document; use dom::documentfragment::DocumentFragment; use dom::eventtarget::EventTarget; use dom::htmlelement::HTMLElement; use dom::node::{CloneChildrenFlag, Node, document_from_node}; use dom::virtualmethods::VirtualMethods; use util::str::DOMString; #[dom_struct] pub struct HTMLTemplateElement { htmlelement: HTMLElement, /// https://html.spec.whatwg.org/multipage/#template-contents contents: MutNullableHeap>, } impl HTMLTemplateElementDerived for EventTarget { fn is_htmltemplateelement(&self) -> bool { *self.type_id() == EventTargetTypeId::Node( NodeTypeId::Element(ElementTypeId::HTMLElement(HTMLElementTypeId::HTMLTemplateElement))) } } impl HTMLTemplateElement { fn new_inherited(localName: DOMString, prefix: Option, document: &Document) -> HTMLTemplateElement { HTMLTemplateElement { htmlelement: HTMLElement::new_inherited(HTMLElementTypeId::HTMLTemplateElement, localName, prefix, document), contents: MutNullableHeap::new(None), } } #[allow(unrooted_must_root)] pub fn new(localName: DOMString, prefix: Option, document: &Document) -> Root { let element = HTMLTemplateElement::new_inherited(localName, prefix, document); Node::reflect_node(box element, document, HTMLTemplateElementBinding::Wrap) } } impl HTMLTemplateElementMethods for HTMLTemplateElement { /// https://html.spec.whatwg.org/multipage/#dom-template-content fn Content(&self) -> Root { self.contents.or_init(|| { let doc = document_from_node(self); doc.appropriate_template_contents_owner_document().CreateDocumentFragment() }) } } impl VirtualMethods for HTMLTemplateElement { fn super_type(&self) -> Option<&VirtualMethods> { Some(HTMLElementCast::from_ref(self) as &VirtualMethods) } /// https://html.spec.whatwg.org/multipage/#template-adopting-steps fn adopting_steps(&self, old_doc: &Document) { self.super_type().unwrap().adopting_steps(old_doc); // Step 1. let doc = document_from_node(self).appropriate_template_contents_owner_document(); // Step 2. Node::adopt(NodeCast::from_ref(&*self.Content()), &doc); } /// https://html.spec.whatwg.org/multipage/#the-template-element:concept-node-clone-ext fn cloning_steps(&self, copy: &Node, maybe_doc: Option<&Document>, clone_children: CloneChildrenFlag) { self.super_type().unwrap().cloning_steps(copy, maybe_doc, clone_children); if clone_children == CloneChildrenFlag::DoNotCloneChildren { // Step 1. return; } let copy = HTMLTemplateElementCast::to_ref(copy).unwrap(); // Steps 2-3. let copy_contents = NodeCast::from_root(copy.Content()); let copy_contents_doc = copy_contents.owner_doc(); for child in NodeCast::from_root(self.Content()).children() { let copy_child = Node::clone( &child, Some(©_contents_doc), CloneChildrenFlag::CloneChildren); copy_contents.AppendChild(©_child).unwrap(); } } }