blob: 8f3ed8c4dd9979b7662c14e3ca396bb8d72dafbf [file] [log] [blame]
/*
* Copyright (C) 2022 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. AND ITS CONTRIBUTORS ``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 ITS 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 "ElementInternals.h"
#include "CustomStateSet.h"
#include "AXObjectCache.h"
#include "ElementInlines.h"
#include "ElementRareData.h"
#include "HTMLFormElement.h"
#include "HTMLMaybeFormAssociatedCustomElement.h"
#include "ScriptWrappableInlines.h"
#include "ShadowRoot.h"
#include <wtf/TZoneMallocInlines.h>
namespace WebCore {
using namespace HTMLNames;
WTF_MAKE_TZONE_ALLOCATED_IMPL(ElementInternals);
RefPtr<ShadowRoot> ElementInternals::shadowRoot() const
{
RefPtr element = m_element.get();
if (!element)
return nullptr;
RefPtr shadowRoot = element->shadowRoot();
if (!shadowRoot)
return nullptr;
if (!shadowRoot->isAvailableToElementInternals())
return nullptr;
return shadowRoot;
}
ExceptionOr<RefPtr<HTMLFormElement>> ElementInternals::form() const
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->form();
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<void> ElementInternals::setFormValue(CustomElementFormValue&& value, std::optional<CustomElementFormValue>&& state)
{
if (RefPtr element = elementAsFormAssociatedCustom()) {
element->setFormValue(WTF::move(value), WTF::move(state));
return { };
}
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<void> ElementInternals::setValidity(ValidityStateFlags validityStateFlags, String&& message, HTMLElement* validationAnchor)
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->setValidity(validityStateFlags, WTF::move(message), validationAnchor);
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<bool> ElementInternals::willValidate() const
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->willValidate();
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<ValidityState&> ElementInternals::validity()
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->validity();
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<String> ElementInternals::validationMessage() const
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->validationMessage();
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<bool> ElementInternals::reportValidity()
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->reportValidity();
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<bool> ElementInternals::checkValidity()
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->checkValidity();
return Exception { ExceptionCode::NotSupportedError };
}
ExceptionOr<RefPtr<NodeList>> ElementInternals::labels()
{
if (RefPtr element = elementAsFormAssociatedCustom())
return element->asHTMLElement().labels();
return Exception { ExceptionCode::NotSupportedError };
}
FormAssociatedCustomElement* ElementInternals::elementAsFormAssociatedCustom() const
{
if (RefPtr element = dynamicDowncast<HTMLMaybeFormAssociatedCustomElement>(m_element.get()))
return element->formAssociatedCustomElementForElementInternals();
return nullptr;
}
static const AtomString& computeValueForAttribute(Element& element, const QualifiedName& name)
{
auto& value = element.attributeWithoutSynchronization(name);
if (CheckedPtr defaultARIA = element.customElementDefaultARIAIfExists(); value.isNull() && defaultARIA)
return defaultARIA->valueForAttribute(element, name);
return value;
}
void ElementInternals::setAttributeWithoutSynchronization(const QualifiedName& name, const AtomString& value)
{
RefPtr element = m_element.get();
auto oldValue = computeValueForAttribute(*element, name);
element->checkedCustomElementDefaultARIA()->setValueForAttribute(name, value);
if (CheckedPtr cache = element->document().existingAXObjectCache())
cache->deferAttributeChangeIfNeeded(*element, name, oldValue, computeValueForAttribute(*element, name));
}
const AtomString& ElementInternals::attributeWithoutSynchronization(const QualifiedName& name) const
{
RefPtr element = m_element.get();
CheckedPtr defaultARIA = element->customElementDefaultARIAIfExists();
return defaultARIA ? defaultARIA->valueForAttribute(*element, name) : nullAtom();
}
RefPtr<Element> ElementInternals::getElementAttributeForBindings(const QualifiedName& name) const
{
RefPtr element = m_element.get();
CheckedPtr defaultARIA = m_element->customElementDefaultARIAIfExists();
return defaultARIA ? defaultARIA->elementForAttribute(*element, name) : nullptr;
}
void ElementInternals::setElementAttribute(const QualifiedName& name, Element* value)
{
RefPtr element = m_element.get();
auto oldValue = computeValueForAttribute(*element, name);
element->checkedCustomElementDefaultARIA()->setElementForAttribute(name, value);
if (CheckedPtr cache = element->document().existingAXObjectCache())
cache->deferAttributeChangeIfNeeded(*element, name, oldValue, computeValueForAttribute(*element, name));
}
std::optional<Vector<Ref<Element>>> ElementInternals::getElementsArrayAttributeForBindings(const QualifiedName& name) const
{
RefPtr element = m_element.get();
CheckedPtr defaultARIA = m_element->customElementDefaultARIAIfExists();
if (!defaultARIA)
return std::nullopt;
return defaultARIA->elementsForAttribute(*element, name);
}
void ElementInternals::setElementsArrayAttribute(const QualifiedName& name, std::optional<Vector<Ref<Element>>>&& value)
{
RefPtr element = m_element.get();
auto oldValue = computeValueForAttribute(*element, name);
element->checkedCustomElementDefaultARIA()->setElementsForAttribute(name, WTF::move(value));
if (CheckedPtr cache = element->document().existingAXObjectCache())
cache->deferAttributeChangeIfNeeded(*element, name, oldValue, computeValueForAttribute(*element, name));
}
CustomStateSet& ElementInternals::states()
{
return m_element->ensureCustomStateSet();
}
} // namespace WebCore