| /* |
| * Copyright (C) 1999 Lars Knoll (knoll@kde.org) |
| * (C) 1999 Antti Koivisto (koivisto@kde.org) |
| * (C) 2001 Dirk Mueller (mueller@kde.org) |
| * Copyright (C) 2004-2018 Apple Inc. All rights reserved. |
| * (C) 2006 Alexey Proskuryakov (ap@nypop.com) |
| * Copyright (C) 2007 Samuel Weinig (sam@webkit.org) |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| * |
| */ |
| |
| #include "config.h" |
| #include "HTMLButtonElement.h" |
| |
| #include "DOMFormData.h" |
| #include "EventNames.h" |
| #include "HTMLFormElement.h" |
| #include "HTMLNames.h" |
| #include "KeyboardEvent.h" |
| #include "RenderButton.h" |
| #include <wtf/IsoMallocInlines.h> |
| #include <wtf/SetForScope.h> |
| #include <wtf/StdLibExtras.h> |
| |
| namespace WebCore { |
| |
| WTF_MAKE_ISO_ALLOCATED_IMPL(HTMLButtonElement); |
| |
| using namespace HTMLNames; |
| |
| inline HTMLButtonElement::HTMLButtonElement(const QualifiedName& tagName, Document& document, HTMLFormElement* form) |
| : HTMLFormControlElement(tagName, document, form) |
| , m_type(SUBMIT) |
| , m_isActivatedSubmit(false) |
| { |
| ASSERT(hasTagName(buttonTag)); |
| } |
| |
| Ref<HTMLButtonElement> HTMLButtonElement::create(const QualifiedName& tagName, Document& document, HTMLFormElement* form) |
| { |
| return adoptRef(*new HTMLButtonElement(tagName, document, form)); |
| } |
| |
| void HTMLButtonElement::setType(const AtomicString& type) |
| { |
| setAttributeWithoutSynchronization(typeAttr, type); |
| } |
| |
| RenderPtr<RenderElement> HTMLButtonElement::createElementRenderer(RenderStyle&& style, const RenderTreePosition&) |
| { |
| return createRenderer<RenderButton>(*this, WTFMove(style)); |
| } |
| |
| const AtomicString& HTMLButtonElement::formControlType() const |
| { |
| switch (m_type) { |
| case SUBMIT: { |
| static NeverDestroyed<const AtomicString> submit("submit", AtomicString::ConstructFromLiteral); |
| return submit; |
| } |
| case BUTTON: { |
| static NeverDestroyed<const AtomicString> button("button", AtomicString::ConstructFromLiteral); |
| return button; |
| } |
| case RESET: { |
| static NeverDestroyed<const AtomicString> reset("reset", AtomicString::ConstructFromLiteral); |
| return reset; |
| } |
| } |
| |
| ASSERT_NOT_REACHED(); |
| return emptyAtom(); |
| } |
| |
| bool HTMLButtonElement::isPresentationAttribute(const QualifiedName& name) const |
| { |
| if (name == alignAttr) { |
| // Don't map 'align' attribute. This matches what Firefox and IE do, but not Opera. |
| // See http://bugs.webkit.org/show_bug.cgi?id=12071 |
| return false; |
| } |
| |
| return HTMLFormControlElement::isPresentationAttribute(name); |
| } |
| |
| void HTMLButtonElement::parseAttribute(const QualifiedName& name, const AtomicString& value) |
| { |
| if (name == typeAttr) { |
| Type oldType = m_type; |
| if (equalLettersIgnoringASCIICase(value, "reset")) |
| m_type = RESET; |
| else if (equalLettersIgnoringASCIICase(value, "button")) |
| m_type = BUTTON; |
| else |
| m_type = SUBMIT; |
| if (oldType != m_type) { |
| setNeedsWillValidateCheck(); |
| if (form() && (oldType == SUBMIT || m_type == SUBMIT)) |
| form()->resetDefaultButton(); |
| } |
| } else |
| HTMLFormControlElement::parseAttribute(name, value); |
| } |
| |
| void HTMLButtonElement::defaultEventHandler(Event& event) |
| { |
| if (event.type() == eventNames().DOMActivateEvent && !isDisabledFormControl()) { |
| RefPtr<HTMLFormElement> protectedForm(form()); |
| |
| if (protectedForm) { |
| // Update layout before processing form actions in case the style changes |
| // the Form or button relationships. |
| document().updateLayoutIgnorePendingStylesheets(); |
| |
| if (auto currentForm = form()) { |
| if (m_type == SUBMIT) { |
| SetForScope<bool> activatedSubmitState(m_isActivatedSubmit, true); |
| currentForm->prepareForSubmission(event); |
| } |
| |
| if (m_type == RESET) |
| currentForm->reset(); |
| } |
| |
| if (m_type == SUBMIT || m_type == RESET) |
| event.setDefaultHandled(); |
| } |
| } |
| |
| if (is<KeyboardEvent>(event)) { |
| KeyboardEvent& keyboardEvent = downcast<KeyboardEvent>(event); |
| if (keyboardEvent.type() == eventNames().keydownEvent && keyboardEvent.keyIdentifier() == "U+0020") { |
| setActive(true, true); |
| // No setDefaultHandled() - IE dispatches a keypress in this case. |
| return; |
| } |
| if (keyboardEvent.type() == eventNames().keypressEvent) { |
| switch (keyboardEvent.charCode()) { |
| case '\r': |
| dispatchSimulatedClick(&keyboardEvent); |
| keyboardEvent.setDefaultHandled(); |
| return; |
| case ' ': |
| // Prevent scrolling down the page. |
| keyboardEvent.setDefaultHandled(); |
| return; |
| } |
| } |
| if (keyboardEvent.type() == eventNames().keyupEvent && keyboardEvent.keyIdentifier() == "U+0020") { |
| if (active()) |
| dispatchSimulatedClick(&keyboardEvent); |
| keyboardEvent.setDefaultHandled(); |
| return; |
| } |
| } |
| |
| HTMLFormControlElement::defaultEventHandler(event); |
| } |
| |
| bool HTMLButtonElement::willRespondToMouseClickEvents() |
| { |
| return !isDisabledFormControl(); |
| } |
| |
| bool HTMLButtonElement::isSuccessfulSubmitButton() const |
| { |
| // HTML spec says that buttons must have names to be considered successful. |
| // However, other browsers do not impose this constraint. |
| return m_type == SUBMIT && !isDisabledFormControl(); |
| } |
| |
| bool HTMLButtonElement::matchesDefaultPseudoClass() const |
| { |
| return isSuccessfulSubmitButton() && form() && form()->defaultButton() == this; |
| } |
| |
| bool HTMLButtonElement::isActivatedSubmit() const |
| { |
| return m_isActivatedSubmit; |
| } |
| |
| void HTMLButtonElement::setActivatedSubmit(bool flag) |
| { |
| m_isActivatedSubmit = flag; |
| } |
| |
| bool HTMLButtonElement::appendFormData(DOMFormData& formData, bool) |
| { |
| if (m_type != SUBMIT || name().isEmpty() || !m_isActivatedSubmit) |
| return false; |
| formData.append(name(), value()); |
| return true; |
| } |
| |
| void HTMLButtonElement::accessKeyAction(bool sendMouseEvents) |
| { |
| focus(); |
| |
| dispatchSimulatedClick(0, sendMouseEvents ? SendMouseUpDownEvents : SendNoEvents); |
| } |
| |
| bool HTMLButtonElement::isURLAttribute(const Attribute& attribute) const |
| { |
| return attribute.name() == formactionAttr || HTMLFormControlElement::isURLAttribute(attribute); |
| } |
| |
| const AtomicString& HTMLButtonElement::value() const |
| { |
| return attributeWithoutSynchronization(valueAttr); |
| } |
| |
| bool HTMLButtonElement::computeWillValidate() const |
| { |
| return m_type == SUBMIT && HTMLFormControlElement::computeWillValidate(); |
| } |
| |
| } // namespace |