HTMLOptionElement.cpp [plain text]
#include "config.h"
#include "HTMLOptionElement.h"
#include "Document.h"
#include "ElementAncestorIterator.h"
#include "HTMLDataListElement.h"
#include "HTMLNames.h"
#include "HTMLOptGroupElement.h"
#include "HTMLParserIdioms.h"
#include "HTMLSelectElement.h"
#include "NodeRenderStyle.h"
#include "NodeTraversal.h"
#include "RenderMenuList.h"
#include "RenderTheme.h"
#include "ScriptElement.h"
#include "StyleResolver.h"
#include "Text.h"
#include <wtf/IsoMallocInlines.h>
#include <wtf/Ref.h>
namespace WebCore {
WTF_MAKE_ISO_ALLOCATED_IMPL(HTMLOptionElement);
using namespace HTMLNames;
HTMLOptionElement::HTMLOptionElement(const QualifiedName& tagName, Document& document)
: HTMLElement(tagName, document)
, m_disabled(false)
, m_isSelected(false)
{
ASSERT(hasTagName(optionTag));
setHasCustomStyleResolveCallbacks();
}
Ref<HTMLOptionElement> HTMLOptionElement::create(Document& document)
{
return adoptRef(*new HTMLOptionElement(optionTag, document));
}
Ref<HTMLOptionElement> HTMLOptionElement::create(const QualifiedName& tagName, Document& document)
{
return adoptRef(*new HTMLOptionElement(tagName, document));
}
ExceptionOr<Ref<HTMLOptionElement>> HTMLOptionElement::createForJSConstructor(Document& document, const String& text, const String& value, bool defaultSelected, bool selected)
{
auto element = create(document);
if (!text.isEmpty()) {
auto appendResult = element->appendChild(Text::create(document, text));
if (appendResult.hasException())
return appendResult.releaseException();
}
if (!value.isNull())
element->setValue(value);
if (defaultSelected)
element->setAttributeWithoutSynchronization(selectedAttr, emptyAtom());
element->setSelected(selected);
return element;
}
bool HTMLOptionElement::isFocusable() const
{
if (!supportsFocus())
return false;
auto* style = const_cast<HTMLOptionElement&>(*this).computedStyle();
return style && style->display() != DisplayType::None;
}
bool HTMLOptionElement::matchesDefaultPseudoClass() const
{
return hasAttributeWithoutSynchronization(selectedAttr);
}
String HTMLOptionElement::text() const
{
String text = collectOptionInnerText();
return stripLeadingAndTrailingHTMLSpaces(document().displayStringModifiedByEncoding(text)).simplifyWhiteSpace(isHTMLSpace);
}
void HTMLOptionElement::setText(const String &text)
{
Ref<HTMLOptionElement> protectedThis(*this);
RefPtr<HTMLSelectElement> select = ownerSelectElement();
bool selectIsMenuList = select && select->usesMenuList();
int oldSelectedIndex = selectIsMenuList ? select->selectedIndex() : -1;
RefPtr<Node> child = firstChild();
if (is<Text>(child) && !child->nextSibling())
downcast<Text>(*child).setData(text);
else {
removeChildren();
appendChild(Text::create(document(), text));
}
if (selectIsMenuList && select->selectedIndex() != oldSelectedIndex)
select->setSelectedIndex(oldSelectedIndex);
}
bool HTMLOptionElement::accessKeyAction(bool)
{
RefPtr<HTMLSelectElement> select = ownerSelectElement();
if (select) {
select->accessKeySetSelectedIndex(index());
return true;
}
return false;
}
int HTMLOptionElement::index() const
{
RefPtr<HTMLSelectElement> selectElement = ownerSelectElement();
if (!selectElement)
return 0;
int optionIndex = 0;
for (auto& item : selectElement->listItems()) {
if (!is<HTMLOptionElement>(*item))
continue;
if (item == this)
return optionIndex;
++optionIndex;
}
return 0;
}
void HTMLOptionElement::parseAttribute(const QualifiedName& name, const AtomString& value)
{
#if ENABLE(DATALIST_ELEMENT)
if (name == valueAttr) {
for (auto& dataList : ancestorsOfType<HTMLDataListElement>(*this))
dataList.optionElementChildrenChanged();
} else
#endif
if (name == disabledAttr) {
bool oldDisabled = m_disabled;
m_disabled = !value.isNull();
if (oldDisabled != m_disabled) {
invalidateStyleForSubtree();
if (renderer() && renderer()->style().hasAppearance())
renderer()->theme().stateChanged(*renderer(), ControlStates::EnabledState);
}
} else if (name == selectedAttr) {
invalidateStyleForSubtree();
m_isSelected = !value.isNull();
} else
HTMLElement::parseAttribute(name, value);
}
String HTMLOptionElement::value() const
{
const AtomString& value = attributeWithoutSynchronization(valueAttr);
if (!value.isNull())
return value;
return stripLeadingAndTrailingHTMLSpaces(collectOptionInnerText()).simplifyWhiteSpace(isHTMLSpace);
}
void HTMLOptionElement::setValue(const String& value)
{
setAttributeWithoutSynchronization(valueAttr, value);
}
bool HTMLOptionElement::selected() const
{
if (RefPtr<HTMLSelectElement> select = ownerSelectElement())
select->updateListItemSelectedStates();
return m_isSelected;
}
void HTMLOptionElement::setSelected(bool selected)
{
if (m_isSelected == selected)
return;
setSelectedState(selected);
if (RefPtr<HTMLSelectElement> select = ownerSelectElement())
select->optionSelectionStateChanged(*this, selected);
}
void HTMLOptionElement::setSelectedState(bool selected)
{
if (m_isSelected == selected)
return;
m_isSelected = selected;
invalidateStyleForSubtree();
if (RefPtr<HTMLSelectElement> select = ownerSelectElement())
select->invalidateSelectedItems();
}
void HTMLOptionElement::childrenChanged(const ChildChange& change)
{
#if ENABLE(DATALIST_ELEMENT)
for (auto& dataList : ancestorsOfType<HTMLDataListElement>(*this))
dataList.optionElementChildrenChanged();
#endif
if (RefPtr<HTMLSelectElement> select = ownerSelectElement())
select->optionElementChildrenChanged();
HTMLElement::childrenChanged(change);
}
HTMLSelectElement* HTMLOptionElement::ownerSelectElement() const
{
return const_cast<HTMLSelectElement*>(ancestorsOfType<HTMLSelectElement>(*this).first());
}
String HTMLOptionElement::label() const
{
String label = attributeWithoutSynchronization(labelAttr);
if (!label.isNull())
return stripLeadingAndTrailingHTMLSpaces(label);
return stripLeadingAndTrailingHTMLSpaces(collectOptionInnerText()).simplifyWhiteSpace(isHTMLSpace);
}
String HTMLOptionElement::displayLabel() const
{
if (document().inQuirksMode())
return stripLeadingAndTrailingHTMLSpaces(collectOptionInnerText()).simplifyWhiteSpace(isHTMLSpace);
return label();
}
void HTMLOptionElement::setLabel(const String& label)
{
setAttributeWithoutSynchronization(labelAttr, label);
}
void HTMLOptionElement::willResetComputedStyle()
{
if (auto select = ownerSelectElement()) {
if (auto renderer = select->renderer())
renderer->repaint();
}
}
String HTMLOptionElement::textIndentedToRespectGroupLabel() const
{
RefPtr<ContainerNode> parent = parentNode();
if (is<HTMLOptGroupElement>(parent))
return " " + displayLabel();
return displayLabel();
}
bool HTMLOptionElement::isDisabledFormControl() const
{
if (ownElementDisabled())
return true;
if (!is<HTMLOptGroupElement>(parentNode()))
return false;
return downcast<HTMLOptGroupElement>(*parentNode()).isDisabledFormControl();
}
Node::InsertedIntoAncestorResult HTMLOptionElement::insertedIntoAncestor(InsertionType insertionType, ContainerNode& parentOfInsertedTree)
{
if (RefPtr<HTMLSelectElement> select = ownerSelectElement()) {
select->setRecalcListItems();
select->updateValidity();
if (m_isSelected)
select->optionSelectionStateChanged(*this, true);
select->scrollToSelection();
}
return HTMLElement::insertedIntoAncestor(insertionType, parentOfInsertedTree);
}
String HTMLOptionElement::collectOptionInnerText() const
{
StringBuilder text;
for (RefPtr<Node> node = firstChild(); node; ) {
if (is<Text>(*node))
text.append(node->nodeValue());
if (is<Element>(*node) && isScriptElement(downcast<Element>(*node)))
node = NodeTraversal::nextSkippingChildren(*node, this);
else
node = NodeTraversal::next(*node, this);
}
return text.toString();
}
}