blob: 240ef78a5d4dd7f553da6aa640af6f014cbbac2d [file] [log] [blame]
/*
* Copyright (C) 2015, 2016 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "CustomElementRegistry.h"
#include "CustomElementReactionQueue.h"
#include "Document.h"
#include "DocumentQuirks.h"
#include "ElementRareData.h"
#include "ElementTraversal.h"
#include "HTMLElementFactory.h"
#include "JSCustomElementInterface.h"
#include "JSDOMPromiseDeferred.h"
#include "LocalDOMWindow.h"
#include "MathMLNames.h"
#include "QualifiedName.h"
#include "ShadowRoot.h"
#include "TypedElementDescendantIteratorInlines.h"
#include <JavaScriptCore/JSCJSValueInlines.h>
#include <wtf/text/AtomString.h>
namespace WebCore {
Ref<CustomElementRegistry> CustomElementRegistry::create(ScriptExecutionContext& scriptExecutionContext, LocalDOMWindow& window)
{
return adoptRef(*new CustomElementRegistry(scriptExecutionContext, window));
}
Ref<CustomElementRegistry> CustomElementRegistry::create(ScriptExecutionContext& scriptExecutionContext)
{
return adoptRef(*new CustomElementRegistry(scriptExecutionContext));
}
CustomElementRegistry::CustomElementRegistry(ScriptExecutionContext& scriptExecutionContext, LocalDOMWindow& window)
: ContextDestructionObserver(&scriptExecutionContext)
, m_window(window)
{
}
CustomElementRegistry::CustomElementRegistry(ScriptExecutionContext& scriptExecutionContext)
: ContextDestructionObserver(&scriptExecutionContext)
{
}
CustomElementRegistry::~CustomElementRegistry() = default;
Document* CustomElementRegistry::document() const
{
return m_window ? m_window->document() : nullptr;
}
void CustomElementRegistry::didAssociateWithDocument(Document& document)
{
m_associatedDocuments.add(document);
}
// https://dom.spec.whatwg.org/#concept-shadow-including-tree-order
static void enqueueUpgradeInShadowIncludingTreeOrder(ContainerNode& node, JSCustomElementInterface& elementInterface, CustomElementRegistry& registry)
{
for (RefPtr element = ElementTraversal::firstWithin(node); element; element = ElementTraversal::next(*element)) {
if (element->isCustomElementUpgradeCandidate() && CustomElementRegistry::registryForElement(*element) == &registry && element->tagQName().matches(elementInterface.name()))
element->enqueueToUpgrade(elementInterface);
if (RefPtr shadowRoot = element->shadowRoot()) {
if (shadowRoot->mode() != ShadowRootMode::UserAgent)
enqueueUpgradeInShadowIncludingTreeOrder(*shadowRoot, elementInterface, registry);
}
}
}
RefPtr<DeferredPromise> CustomElementRegistry::addElementDefinition(Ref<JSCustomElementInterface>&& elementInterface)
{
static MainThreadNeverDestroyed<const AtomString> extendsLi("extends-li"_s);
AtomString localName = elementInterface->name().localName();
ASSERT(!m_nameMap.contains(localName));
m_nameMap.add(localName, elementInterface.copyRef());
{
Locker locker { m_constructorMapLock };
m_constructorMap.add(elementInterface->constructor(), elementInterface.ptr());
}
if (elementInterface->isShadowDisabled())
m_disabledShadowSet.add(localName);
if (RefPtr document = this->document()) { // Global custom element registry
// ungap/@custom-elements detection for quirk (rdar://problem/111008826).
if (localName == extendsLi.get())
document->quirks().setNeedsConfigurableIndexedPropertiesQuirk();
enqueueUpgradeInShadowIncludingTreeOrder(*document, elementInterface.get(), *this);
}
for (Ref document : m_associatedDocuments) {
if (document->hasBrowsingContext())
enqueueUpgradeInShadowIncludingTreeOrder(document, elementInterface.get(), *this);
}
return m_promiseMap.take(localName);
}
JSCustomElementInterface* CustomElementRegistry::findInterface(const Element& element) const
{
return findInterface(element.tagQName());
}
JSCustomElementInterface* CustomElementRegistry::findInterface(const QualifiedName& name) const
{
if (name.namespaceURI() != HTMLNames::xhtmlNamespaceURI)
return nullptr;
return m_nameMap.get(name.localName());
}
JSCustomElementInterface* CustomElementRegistry::findInterface(const AtomString& name) const
{
return m_nameMap.get(name);
}
RefPtr<JSCustomElementInterface> CustomElementRegistry::findInterface(const JSC::JSObject* constructor) const
{
Locker locker { m_constructorMapLock };
return m_constructorMap.get(constructor);
}
bool CustomElementRegistry::containsConstructor(const JSC::JSObject* constructor) const
{
Locker locker { m_constructorMapLock };
return m_constructorMap.contains(constructor);
}
JSC::JSValue CustomElementRegistry::get(const AtomString& name)
{
if (RefPtr elementInterface = m_nameMap.get(name))
return elementInterface->constructor();
return JSC::jsUndefined();
}
String CustomElementRegistry::getName(JSC::JSValue constructorValue)
{
auto* constructor = constructorValue.getObject();
if (!constructor)
return String { };
RefPtr elementInterface = findInterface(constructor);
if (!elementInterface)
return String { };
return elementInterface->name().localName();
}
static void upgradeElementsInShadowIncludingDescendants(CustomElementRegistry& registry, ContainerNode& root)
{
for (Ref element : descendantsOfType<Element>(root)) {
if (element->isCustomElementUpgradeCandidate() && CustomElementRegistry::registryForElement(element) == &registry)
CustomElementReactionQueue::tryToUpgradeElement(element);
if (RefPtr shadowRoot = element->shadowRoot())
upgradeElementsInShadowIncludingDescendants(registry, *shadowRoot);
}
}
void CustomElementRegistry::upgrade(Node& root)
{
auto* containerNode = dynamicDowncast<ContainerNode>(root);
if (!containerNode)
return;
RefPtr element = dynamicDowncast<Element>(*containerNode);
if (element && element->isCustomElementUpgradeCandidate())
CustomElementReactionQueue::tryToUpgradeElement(*element);
upgradeElementsInShadowIncludingDescendants(*this, *containerNode);
}
ExceptionOr<void> CustomElementRegistry::initialize(Node& root)
{
if (!isScoped() && (is<Document>(root) || this != root.document().customElementRegistry()))
return Exception { ExceptionCode::NotSupportedError };
auto* containerRoot = dynamicDowncast<ContainerNode>(root);
if (!containerRoot) {
ASSERT(!root.usesNullCustomElementRegistry()); // Flag is only set on ShadowRoot and Element.
return { };
}
if (RefPtr document = dynamicDowncast<Document>(*containerRoot); document && document->usesNullCustomElementRegistry()) {
document->clearUsesNullCustomElementRegistry();
document->setCustomElementRegistry(*this);
} else if (RefPtr shadowRoot = dynamicDowncast<ShadowRoot>(*containerRoot); shadowRoot && shadowRoot->usesNullCustomElementRegistry()) {
ASSERT(shadowRoot->hasScopedCustomElementRegistry());
shadowRoot->clearUsesNullCustomElementRegistry();
shadowRoot->setCustomElementRegistry(*this);
} else if (auto* documentFragment = dynamicDowncast<DocumentFragment>(*containerRoot); documentFragment && documentFragment->usesNullCustomElementRegistry())
documentFragment->clearUsesNullCustomElementRegistry();
RefPtr registryOfTreeScope = root.isInTreeScope() ? root.treeScope().customElementRegistry() : nullptr;
auto updateRegistryIfNeeded = [&](Element& element) {
if (!element.usesNullCustomElementRegistry())
return;
element.clearUsesNullCustomElementRegistry();
if (this != registryOfTreeScope)
addToScopedCustomElementRegistryMap(element, *this);
};
auto upgradeElementIfPossible = [&](Element& element) {
if (element.isCustomElementUpgradeCandidate() && CustomElementRegistry::registryForElement(element) == this)
CustomElementReactionQueue::tryToUpgradeElement(element);
};
if (RefPtr element = dynamicDowncast<Element>(*containerRoot)) {
updateRegistryIfNeeded(*element);
upgradeElementIfPossible(*element);
}
for (Ref element : descendantsOfType<Element>(*containerRoot)) {
updateRegistryIfNeeded(element);
upgradeElementIfPossible(element);
}
return { };
}
void CustomElementRegistry::addToScopedCustomElementRegistryMap(Element& element, CustomElementRegistry& registry)
{
ASSERT(!element.usesScopedCustomElementRegistryMap() || scopedCustomElementRegistryMap().get(element) == &registry);
if (element.usesScopedCustomElementRegistryMap())
return;
element.setUsesScopedCustomElementRegistryMap();
registry.didAssociateWithDocument(element.protectedDocument().get());
auto result = scopedCustomElementRegistryMap().add(element, registry);
ASSERT_UNUSED(result, result.isNewEntry);
}
void CustomElementRegistry::removeFromScopedCustomElementRegistryMap(Element& element)
{
ASSERT(element.usesScopedCustomElementRegistryMap());
element.clearUsesScopedCustomElementRegistryMap();
auto didRemove = scopedCustomElementRegistryMap().remove(element);
ASSERT_UNUSED(didRemove, didRemove);
}
WeakHashMap<Element, Ref<CustomElementRegistry>, WeakPtrImplWithEventTargetData>& CustomElementRegistry::scopedCustomElementRegistryMap()
{
static NeverDestroyed<WeakHashMap<Element, Ref<CustomElementRegistry>, WeakPtrImplWithEventTargetData>> map;
return map.get();
}
template<typename Visitor>
void CustomElementRegistry::visitJSCustomElementInterfaces(Visitor& visitor) const
{
Locker locker { m_constructorMapLock };
for (const auto& iterator : m_constructorMap)
iterator.value->visitJSFunctions(visitor);
}
template void CustomElementRegistry::visitJSCustomElementInterfaces(JSC::AbstractSlotVisitor&) const;
template void CustomElementRegistry::visitJSCustomElementInterfaces(JSC::SlotVisitor&) const;
}