| OLD | NEW |
| (Empty) |
| 1 /* | |
| 2 * Copyright (C) 2013 Google Inc. All rights reserved. | |
| 3 * | |
| 4 * Redistribution and use in source and binary forms, with or without | |
| 5 * modification, are permitted provided that the following conditions | |
| 6 * are met: | |
| 7 * | |
| 8 * 1. Redistributions of source code must retain the above copyright | |
| 9 * notice, this list of conditions and the following disclaimer. | |
| 10 * 2. Redistributions in binary form must reproduce the above copyright | |
| 11 * notice, this list of conditions and the following disclaimer | |
| 12 * in the documentation and/or other materials provided with the | |
| 13 * distribution. | |
| 14 * 3. Neither the name of Google Inc. nor the names of its contributors | |
| 15 * may be used to endorse or promote products derived from this | |
| 16 * software without specific prior written permission. | |
| 17 * | |
| 18 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |
| 19 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |
| 20 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |
| 21 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |
| 22 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |
| 23 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | |
| 24 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | |
| 25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | |
| 26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | |
| 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | |
| 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
| 29 */ | |
| 30 | |
| 31 #include "config.h" | |
| 32 #include "core/dom/CustomElementRegistrationContext.h" | |
| 33 | |
| 34 #include "HTMLNames.h" | |
| 35 #include "SVGNames.h" | |
| 36 #include "bindings/v8/ExceptionState.h" | |
| 37 #include "core/dom/CustomElement.h" | |
| 38 #include "core/dom/CustomElementCallbackScheduler.h" | |
| 39 #include "core/dom/CustomElementDefinition.h" | |
| 40 #include "core/dom/Element.h" | |
| 41 #include "core/html/HTMLElement.h" | |
| 42 #include "core/html/HTMLUnknownElement.h" | |
| 43 #include "core/svg/SVGUnknownElement.h" | |
| 44 #include "wtf/RefPtr.h" | |
| 45 | |
| 46 namespace WebCore { | |
| 47 | |
| 48 void CustomElementRegistrationContext::registerElement(Document* document, Custo
mElementConstructorBuilder* constructorBuilder, const AtomicString& type, Custom
Element::NameSet validNames, ExceptionState& es) | |
| 49 { | |
| 50 CustomElementDefinition* definition = m_registry.registerElement(document, c
onstructorBuilder, type, validNames, es); | |
| 51 | |
| 52 if (!definition) | |
| 53 return; | |
| 54 | |
| 55 // Upgrade elements that were waiting for this definition. | |
| 56 const CustomElementUpgradeCandidateMap::ElementSet& upgradeCandidates = m_ca
ndidates.takeUpgradeCandidatesFor(definition->descriptor()); | |
| 57 for (CustomElementUpgradeCandidateMap::ElementSet::const_iterator it = upgra
deCandidates.begin(); it != upgradeCandidates.end(); ++it) | |
| 58 didResolveElement(definition, *it); | |
| 59 } | |
| 60 | |
| 61 PassRefPtr<Element> CustomElementRegistrationContext::createCustomTagElement(Doc
ument& document, const QualifiedName& tagName, CreationMode mode) | |
| 62 { | |
| 63 ASSERT(CustomElement::isValidName(tagName.localName())); | |
| 64 | |
| 65 RefPtr<Element> element; | |
| 66 | |
| 67 if (HTMLNames::xhtmlNamespaceURI == tagName.namespaceURI()) { | |
| 68 element = HTMLElement::create(tagName, document); | |
| 69 } else if (SVGNames::svgNamespaceURI == tagName.namespaceURI()) { | |
| 70 element = SVGUnknownElement::create(tagName, document); | |
| 71 } else { | |
| 72 // XML elements are not custom elements, so return early. | |
| 73 return Element::create(tagName, &document); | |
| 74 } | |
| 75 | |
| 76 element->setCustomElementState(mode == CreatedByParser ? Element::WaitingFor
Parser : Element::WaitingForUpgrade); | |
| 77 resolve(element.get(), nullAtom); | |
| 78 return element.release(); | |
| 79 } | |
| 80 | |
| 81 void CustomElementRegistrationContext::didGiveTypeExtension(Element* element, co
nst AtomicString& type) | |
| 82 { | |
| 83 resolve(element, type); | |
| 84 } | |
| 85 | |
| 86 void CustomElementRegistrationContext::resolve(Element* element, const AtomicStr
ing& typeExtension) | |
| 87 { | |
| 88 // If an element has a custom tag name it takes precedence over | |
| 89 // the "is" attribute (if any). | |
| 90 const AtomicString& type = CustomElement::isValidName(element->localName()) | |
| 91 ? element->localName() | |
| 92 : typeExtension; | |
| 93 ASSERT(!type.isNull()); | |
| 94 | |
| 95 CustomElementDescriptor descriptor(type, element->namespaceURI(), element->l
ocalName()); | |
| 96 CustomElementDefinition* definition = m_registry.find(descriptor); | |
| 97 if (definition) | |
| 98 didResolveElement(definition, element); | |
| 99 else | |
| 100 didCreateUnresolvedElement(descriptor, element); | |
| 101 } | |
| 102 | |
| 103 void CustomElementRegistrationContext::didResolveElement(CustomElementDefinition
* definition, Element* element) | |
| 104 { | |
| 105 CustomElement::define(element, definition); | |
| 106 } | |
| 107 | |
| 108 void CustomElementRegistrationContext::didCreateUnresolvedElement(const CustomEl
ementDescriptor& descriptor, Element* element) | |
| 109 { | |
| 110 ASSERT(element->customElementState() == Element::WaitingForParser || element
->customElementState() == Element::WaitingForUpgrade); | |
| 111 m_candidates.add(descriptor, element); | |
| 112 } | |
| 113 | |
| 114 PassRefPtr<CustomElementRegistrationContext> CustomElementRegistrationContext::c
reate() | |
| 115 { | |
| 116 return adoptRef(new CustomElementRegistrationContext()); | |
| 117 } | |
| 118 | |
| 119 void CustomElementRegistrationContext::setIsAttributeAndTypeExtension(Element* e
lement, const AtomicString& type) | |
| 120 { | |
| 121 ASSERT(element); | |
| 122 ASSERT(!type.isEmpty()); | |
| 123 element->setAttribute(HTMLNames::isAttr, type); | |
| 124 setTypeExtension(element, type); | |
| 125 } | |
| 126 | |
| 127 void CustomElementRegistrationContext::setTypeExtension(Element* element, const
AtomicString& type, CreationMode mode) | |
| 128 { | |
| 129 if (!element->isHTMLElement() && !element->isSVGElement()) | |
| 130 return; | |
| 131 | |
| 132 if (element->isCustomElement()) { | |
| 133 // This can happen if: | |
| 134 // 1. The element has a custom tag, which takes precedence over | |
| 135 // type extensions. | |
| 136 // 2. Undoing a command (eg ReplaceNodeWithSpan) recycles an | |
| 137 // element but tries to overwrite its attribute list. | |
| 138 return; | |
| 139 } | |
| 140 | |
| 141 // Custom tags take precedence over type extensions | |
| 142 ASSERT(!CustomElement::isValidName(element->localName())); | |
| 143 | |
| 144 element->setCustomElementState(mode == CreatedByParser ? Element::WaitingFor
Parser : Element::WaitingForUpgrade); | |
| 145 | |
| 146 if (CustomElementRegistrationContext* context = element->document().registra
tionContext()) | |
| 147 context->didGiveTypeExtension(element, type); | |
| 148 } | |
| 149 | |
| 150 } // namespace WebCore | |
| OLD | NEW |