| /* |
| * 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, 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved. |
| * (C) 2006 Alexey Proskuryakov (ap@nypop.com) |
| * |
| * 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 "HTMLFormElement.h" |
| |
| #include "Attribute.h" |
| #include "Document.h" |
| #include "ElementTraversal.h" |
| #include "Event.h" |
| #include "EventNames.h" |
| #include "FormController.h" |
| #include "FormData.h" |
| #include "FormState.h" |
| #include "Frame.h" |
| #include "FrameLoader.h" |
| #include "FrameLoaderClient.h" |
| #include "HTMLCollection.h" |
| #include "HTMLImageElement.h" |
| #include "HTMLInputElement.h" |
| #include "HTMLNames.h" |
| #include "HTMLTableElement.h" |
| #include "Page.h" |
| #include "RenderTextControl.h" |
| #include "ScriptController.h" |
| #include "Settings.h" |
| #include <limits> |
| #include <wtf/Ref.h> |
| |
| namespace WebCore { |
| |
| using namespace HTMLNames; |
| |
| HTMLFormElement::HTMLFormElement(const QualifiedName& tagName, Document& document) |
| : HTMLElement(tagName, document) |
| , m_associatedElementsBeforeIndex(0) |
| , m_associatedElementsAfterIndex(0) |
| , m_wasUserSubmitted(false) |
| , m_isSubmittingOrPreparingForSubmission(false) |
| , m_shouldSubmit(false) |
| , m_isInResetFunction(false) |
| , m_wasDemoted(false) |
| { |
| ASSERT(hasTagName(formTag)); |
| } |
| |
| PassRefPtr<HTMLFormElement> HTMLFormElement::create(Document& document) |
| { |
| return adoptRef(new HTMLFormElement(formTag, document)); |
| } |
| |
| PassRefPtr<HTMLFormElement> HTMLFormElement::create(const QualifiedName& tagName, Document& document) |
| { |
| return adoptRef(new HTMLFormElement(tagName, document)); |
| } |
| |
| HTMLFormElement::~HTMLFormElement() |
| { |
| document().formController().willDeleteForm(this); |
| if (!shouldAutocomplete()) |
| document().unregisterForPageCacheSuspensionCallbacks(this); |
| |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) |
| m_associatedElements[i]->formWillBeDestroyed(); |
| for (unsigned i = 0; i < m_imageElements.size(); ++i) |
| m_imageElements[i]->m_form = 0; |
| } |
| |
| bool HTMLFormElement::formWouldHaveSecureSubmission(const String& url) |
| { |
| return document().completeURL(url).protocolIs("https"); |
| } |
| |
| bool HTMLFormElement::rendererIsNeeded(const RenderStyle& style) |
| { |
| if (!m_wasDemoted) |
| return HTMLElement::rendererIsNeeded(style); |
| |
| auto parent = parentNode(); |
| auto parentRenderer = parent->renderer(); |
| |
| if (!parentRenderer) |
| return false; |
| |
| // FIXME: Shouldn't we also check for table caption (see |formIsTablePart| below). |
| bool parentIsTableElementPart = (parentRenderer->isTable() && isHTMLTableElement(parent)) |
| || (parentRenderer->isTableRow() && parent->hasTagName(trTag)) |
| || (parentRenderer->isTableSection() && parent->hasTagName(tbodyTag)) |
| || (parentRenderer->isRenderTableCol() && parent->hasTagName(colTag)) |
| || (parentRenderer->isTableCell() && parent->hasTagName(trTag)); |
| |
| if (!parentIsTableElementPart) |
| return true; |
| |
| EDisplay display = style.display(); |
| bool formIsTablePart = display == TABLE || display == INLINE_TABLE || display == TABLE_ROW_GROUP |
| || display == TABLE_HEADER_GROUP || display == TABLE_FOOTER_GROUP || display == TABLE_ROW |
| || display == TABLE_COLUMN_GROUP || display == TABLE_COLUMN || display == TABLE_CELL |
| || display == TABLE_CAPTION; |
| |
| return formIsTablePart; |
| } |
| |
| Node::InsertionNotificationRequest HTMLFormElement::insertedInto(ContainerNode& insertionPoint) |
| { |
| HTMLElement::insertedInto(insertionPoint); |
| if (insertionPoint.inDocument()) |
| document().didAssociateFormControl(this); |
| return InsertionDone; |
| } |
| |
| static inline Node* findRoot(Node* n) |
| { |
| Node* root = n; |
| for (; n; n = n->parentNode()) |
| root = n; |
| return root; |
| } |
| |
| void HTMLFormElement::removedFrom(ContainerNode& insertionPoint) |
| { |
| Node* root = findRoot(this); |
| Vector<FormAssociatedElement*> associatedElements(m_associatedElements); |
| for (unsigned i = 0; i < associatedElements.size(); ++i) |
| associatedElements[i]->formRemovedFromTree(root); |
| HTMLElement::removedFrom(insertionPoint); |
| } |
| |
| void HTMLFormElement::handleLocalEvents(Event& event) |
| { |
| Node* targetNode = event.target()->toNode(); |
| if (event.eventPhase() != Event::CAPTURING_PHASE && targetNode && targetNode != this && (event.type() == eventNames().submitEvent || event.type() == eventNames().resetEvent)) { |
| event.stopPropagation(); |
| return; |
| } |
| HTMLElement::handleLocalEvents(event); |
| } |
| |
| unsigned HTMLFormElement::length() const |
| { |
| unsigned len = 0; |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) |
| if (m_associatedElements[i]->isEnumeratable()) |
| ++len; |
| return len; |
| } |
| |
| Node* HTMLFormElement::item(unsigned index) |
| { |
| return elements()->item(index); |
| } |
| |
| void HTMLFormElement::submitImplicitly(Event* event, bool fromImplicitSubmissionTrigger) |
| { |
| int submissionTriggerCount = 0; |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| FormAssociatedElement* formAssociatedElement = m_associatedElements[i]; |
| if (!formAssociatedElement->isFormControlElement()) |
| continue; |
| HTMLFormControlElement* formElement = static_cast<HTMLFormControlElement*>(formAssociatedElement); |
| if (formElement->isSuccessfulSubmitButton()) { |
| if (formElement->renderer()) { |
| formElement->dispatchSimulatedClick(event); |
| return; |
| } |
| } else if (formElement->canTriggerImplicitSubmission()) |
| ++submissionTriggerCount; |
| } |
| if (fromImplicitSubmissionTrigger && submissionTriggerCount == 1) |
| prepareForSubmission(event); |
| } |
| |
| static inline HTMLFormControlElement* submitElementFromEvent(const Event* event) |
| { |
| for (Node* node = event->target()->toNode(); node; node = node->parentNode()) { |
| if (node->isElementNode() && toElement(node)->isFormControlElement()) |
| return static_cast<HTMLFormControlElement*>(node); |
| } |
| return 0; |
| } |
| |
| bool HTMLFormElement::validateInteractively(Event* event) |
| { |
| ASSERT(event); |
| if (!document().page() || !document().page()->settings().interactiveFormValidationEnabled() || noValidate()) |
| return true; |
| |
| HTMLFormControlElement* submitElement = submitElementFromEvent(event); |
| if (submitElement && submitElement->formNoValidate()) |
| return true; |
| |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| if (m_associatedElements[i]->isFormControlElement()) |
| static_cast<HTMLFormControlElement*>(m_associatedElements[i])->hideVisibleValidationMessage(); |
| } |
| |
| Vector<RefPtr<FormAssociatedElement>> unhandledInvalidControls; |
| if (!checkInvalidControlsAndCollectUnhandled(unhandledInvalidControls)) |
| return true; |
| // Because the form has invalid controls, we abort the form submission and |
| // show a validation message on a focusable form control. |
| |
| // Needs to update layout now because we'd like to call isFocusable(), which |
| // has !renderer()->needsLayout() assertion. |
| document().updateLayoutIgnorePendingStylesheets(); |
| |
| Ref<HTMLFormElement> protect(*this); |
| |
| // Focus on the first focusable control and show a validation message. |
| for (unsigned i = 0; i < unhandledInvalidControls.size(); ++i) { |
| HTMLElement& element = unhandledInvalidControls[i]->asHTMLElement(); |
| if (element.inDocument() && element.isFocusable()) { |
| element.scrollIntoViewIfNeeded(false); |
| element.focus(); |
| if (element.isFormControlElement()) |
| toHTMLFormControlElement(element).updateVisibleValidationMessage(); |
| break; |
| } |
| } |
| |
| // Warn about all of unfocusable controls. |
| if (document().frame()) { |
| for (unsigned i = 0; i < unhandledInvalidControls.size(); ++i) { |
| FormAssociatedElement& control = *unhandledInvalidControls[i]; |
| HTMLElement& element = control.asHTMLElement(); |
| if (element.inDocument() && element.isFocusable()) |
| continue; |
| String message("An invalid form control with name='%name' is not focusable."); |
| message.replace("%name", control.name()); |
| document().addConsoleMessage(RenderingMessageSource, ErrorMessageLevel, message); |
| } |
| } |
| |
| return false; |
| } |
| |
| bool HTMLFormElement::prepareForSubmission(Event* event) |
| { |
| Frame* frame = document().frame(); |
| if (m_isSubmittingOrPreparingForSubmission || !frame) |
| return m_isSubmittingOrPreparingForSubmission; |
| |
| m_isSubmittingOrPreparingForSubmission = true; |
| m_shouldSubmit = false; |
| |
| // Interactive validation must be done before dispatching the submit event. |
| if (!validateInteractively(event)) { |
| m_isSubmittingOrPreparingForSubmission = false; |
| return false; |
| } |
| |
| StringPairVector controlNamesAndValues; |
| getTextFieldValues(controlNamesAndValues); |
| RefPtr<FormState> formState = FormState::create(this, controlNamesAndValues, &document(), NotSubmittedByJavaScript); |
| frame->loader().client().dispatchWillSendSubmitEvent(formState.release()); |
| |
| if (dispatchEvent(Event::create(eventNames().submitEvent, true, true))) |
| m_shouldSubmit = true; |
| |
| m_isSubmittingOrPreparingForSubmission = false; |
| |
| if (m_shouldSubmit) |
| submit(event, true, true, NotSubmittedByJavaScript); |
| |
| return m_shouldSubmit; |
| } |
| |
| void HTMLFormElement::submit() |
| { |
| submit(0, false, true, NotSubmittedByJavaScript); |
| } |
| |
| void HTMLFormElement::submitFromJavaScript() |
| { |
| submit(0, false, ScriptController::processingUserGesture(), SubmittedByJavaScript); |
| } |
| |
| void HTMLFormElement::getTextFieldValues(StringPairVector& fieldNamesAndValues) const |
| { |
| ASSERT_ARG(fieldNamesAndValues, fieldNamesAndValues.isEmpty()); |
| |
| fieldNamesAndValues.reserveCapacity(m_associatedElements.size()); |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| FormAssociatedElement& control = *m_associatedElements[i]; |
| HTMLElement& element = control.asHTMLElement(); |
| if (!isHTMLInputElement(element)) |
| continue; |
| HTMLInputElement& input = toHTMLInputElement(element); |
| if (!input.isTextField()) |
| continue; |
| fieldNamesAndValues.append(std::make_pair(input.name().string(), input.value())); |
| } |
| } |
| |
| void HTMLFormElement::submit(Event* event, bool activateSubmitButton, bool processingUserGesture, FormSubmissionTrigger formSubmissionTrigger) |
| { |
| FrameView* view = document().view(); |
| Frame* frame = document().frame(); |
| if (!view || !frame) |
| return; |
| |
| if (m_isSubmittingOrPreparingForSubmission) { |
| m_shouldSubmit = true; |
| return; |
| } |
| |
| m_isSubmittingOrPreparingForSubmission = true; |
| m_wasUserSubmitted = processingUserGesture; |
| |
| HTMLFormControlElement* firstSuccessfulSubmitButton = 0; |
| bool needButtonActivation = activateSubmitButton; // do we need to activate a submit button? |
| |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| FormAssociatedElement* associatedElement = m_associatedElements[i]; |
| if (!associatedElement->isFormControlElement()) |
| continue; |
| if (needButtonActivation) { |
| HTMLFormControlElement* control = static_cast<HTMLFormControlElement*>(associatedElement); |
| if (control->isActivatedSubmit()) |
| needButtonActivation = false; |
| else if (firstSuccessfulSubmitButton == 0 && control->isSuccessfulSubmitButton()) |
| firstSuccessfulSubmitButton = control; |
| } |
| } |
| |
| if (needButtonActivation && firstSuccessfulSubmitButton) |
| firstSuccessfulSubmitButton->setActivatedSubmit(true); |
| |
| bool lockHistory = !processingUserGesture; |
| frame->loader().submitForm(FormSubmission::create(this, m_attributes, event, lockHistory, formSubmissionTrigger)); |
| |
| if (needButtonActivation && firstSuccessfulSubmitButton) |
| firstSuccessfulSubmitButton->setActivatedSubmit(false); |
| |
| m_shouldSubmit = false; |
| m_isSubmittingOrPreparingForSubmission = false; |
| } |
| |
| void HTMLFormElement::reset() |
| { |
| Frame* frame = document().frame(); |
| if (m_isInResetFunction || !frame) |
| return; |
| |
| m_isInResetFunction = true; |
| |
| if (!dispatchEvent(Event::create(eventNames().resetEvent, true, true))) { |
| m_isInResetFunction = false; |
| return; |
| } |
| |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| if (m_associatedElements[i]->isFormControlElement()) |
| static_cast<HTMLFormControlElement*>(m_associatedElements[i])->reset(); |
| } |
| |
| m_isInResetFunction = false; |
| } |
| |
| void HTMLFormElement::parseAttribute(const QualifiedName& name, const AtomicString& value) |
| { |
| if (name == actionAttr) |
| m_attributes.parseAction(value); |
| else if (name == targetAttr) |
| m_attributes.setTarget(value); |
| else if (name == methodAttr) |
| m_attributes.updateMethodType(value); |
| else if (name == enctypeAttr) |
| m_attributes.updateEncodingType(value); |
| else if (name == accept_charsetAttr) |
| m_attributes.setAcceptCharset(value); |
| else if (name == autocompleteAttr) { |
| if (!shouldAutocomplete()) |
| document().registerForPageCacheSuspensionCallbacks(this); |
| else |
| document().unregisterForPageCacheSuspensionCallbacks(this); |
| } |
| else |
| HTMLElement::parseAttribute(name, value); |
| } |
| |
| template<class T, size_t n> static void removeFromVector(Vector<T*, n> & vec, T* item) |
| { |
| size_t size = vec.size(); |
| for (size_t i = 0; i != size; ++i) |
| if (vec[i] == item) { |
| vec.remove(i); |
| break; |
| } |
| } |
| |
| unsigned HTMLFormElement::formElementIndexWithFormAttribute(Element* element, unsigned rangeStart, unsigned rangeEnd) |
| { |
| if (m_associatedElements.isEmpty()) |
| return 0; |
| |
| ASSERT(rangeStart <= rangeEnd); |
| |
| if (rangeStart == rangeEnd) |
| return rangeStart; |
| |
| unsigned left = rangeStart; |
| unsigned right = rangeEnd - 1; |
| unsigned short position; |
| |
| // Does binary search on m_associatedElements in order to find the index |
| // to be inserted. |
| while (left != right) { |
| unsigned middle = left + ((right - left) / 2); |
| ASSERT(middle < m_associatedElementsBeforeIndex || middle >= m_associatedElementsAfterIndex); |
| position = element->compareDocumentPosition(&m_associatedElements[middle]->asHTMLElement()); |
| if (position & DOCUMENT_POSITION_FOLLOWING) |
| right = middle; |
| else |
| left = middle + 1; |
| } |
| |
| ASSERT(left < m_associatedElementsBeforeIndex || left >= m_associatedElementsAfterIndex); |
| position = element->compareDocumentPosition(&m_associatedElements[left]->asHTMLElement()); |
| if (position & DOCUMENT_POSITION_FOLLOWING) |
| return left; |
| return left + 1; |
| } |
| |
| unsigned HTMLFormElement::formElementIndex(FormAssociatedElement* associatedElement) |
| { |
| ASSERT(associatedElement); |
| |
| HTMLElement& associatedHTMLElement = associatedElement->asHTMLElement(); |
| |
| // Treats separately the case where this element has the form attribute |
| // for performance consideration. |
| if (associatedHTMLElement.fastHasAttribute(formAttr)) { |
| unsigned short position = compareDocumentPosition(&associatedHTMLElement); |
| if (position & DOCUMENT_POSITION_PRECEDING) { |
| ++m_associatedElementsBeforeIndex; |
| ++m_associatedElementsAfterIndex; |
| return HTMLFormElement::formElementIndexWithFormAttribute(&associatedHTMLElement, 0, m_associatedElementsBeforeIndex - 1); |
| } |
| if (position & DOCUMENT_POSITION_FOLLOWING && !(position & DOCUMENT_POSITION_CONTAINED_BY)) |
| return HTMLFormElement::formElementIndexWithFormAttribute(&associatedHTMLElement, m_associatedElementsAfterIndex, m_associatedElements.size()); |
| } |
| |
| // Check for the special case where this element is the very last thing in |
| // the form's tree of children; we don't want to walk the entire tree in that |
| // common case that occurs during parsing; instead we'll just return a value |
| // that says "add this form element to the end of the array". |
| if (ElementTraversal::next(&associatedHTMLElement, this)) { |
| unsigned i = m_associatedElementsBeforeIndex; |
| for (Element* element = this; element; element = ElementTraversal::next(element, this)) { |
| if (element == &associatedHTMLElement) { |
| ++m_associatedElementsAfterIndex; |
| return i; |
| } |
| if (!element->isFormControlElement() && !element->hasTagName(objectTag)) |
| continue; |
| if (!element->isHTMLElement() || toHTMLElement(element)->form() != this) |
| continue; |
| ++i; |
| } |
| } |
| return m_associatedElementsAfterIndex++; |
| } |
| |
| void HTMLFormElement::registerFormElement(FormAssociatedElement* e) |
| { |
| m_associatedElements.insert(formElementIndex(e), e); |
| } |
| |
| void HTMLFormElement::removeFormElement(FormAssociatedElement* e) |
| { |
| unsigned index; |
| for (index = 0; index < m_associatedElements.size(); ++index) { |
| if (m_associatedElements[index] == e) |
| break; |
| } |
| ASSERT_WITH_SECURITY_IMPLICATION(index < m_associatedElements.size()); |
| if (index < m_associatedElementsBeforeIndex) |
| --m_associatedElementsBeforeIndex; |
| if (index < m_associatedElementsAfterIndex) |
| --m_associatedElementsAfterIndex; |
| removeFromPastNamesMap(e); |
| removeFromVector(m_associatedElements, e); |
| } |
| |
| bool HTMLFormElement::isURLAttribute(const Attribute& attribute) const |
| { |
| return attribute.name() == actionAttr || HTMLElement::isURLAttribute(attribute); |
| } |
| |
| void HTMLFormElement::registerImgElement(HTMLImageElement* e) |
| { |
| ASSERT(m_imageElements.find(e) == notFound); |
| m_imageElements.append(e); |
| } |
| |
| void HTMLFormElement::removeImgElement(HTMLImageElement* e) |
| { |
| ASSERT(m_imageElements.find(e) != notFound); |
| removeFromPastNamesMap(e); |
| removeFromVector(m_imageElements, e); |
| } |
| |
| PassRefPtr<HTMLCollection> HTMLFormElement::elements() |
| { |
| return ensureCachedHTMLCollection(FormControls); |
| } |
| |
| String HTMLFormElement::name() const |
| { |
| return getNameAttribute(); |
| } |
| |
| bool HTMLFormElement::noValidate() const |
| { |
| return fastHasAttribute(novalidateAttr); |
| } |
| |
| // FIXME: This function should be removed because it does not do the same thing as the |
| // JavaScript binding for action, which treats action as a URL attribute. Last time I |
| // (Darin Adler) removed this, someone added it back, so I am leaving it in for now. |
| String HTMLFormElement::action() const |
| { |
| return getAttribute(actionAttr); |
| } |
| |
| void HTMLFormElement::setAction(const String &value) |
| { |
| setAttribute(actionAttr, value); |
| } |
| |
| void HTMLFormElement::setEnctype(const String &value) |
| { |
| setAttribute(enctypeAttr, value); |
| } |
| |
| String HTMLFormElement::method() const |
| { |
| return FormSubmission::Attributes::methodString(m_attributes.method()); |
| } |
| |
| void HTMLFormElement::setMethod(const String &value) |
| { |
| setAttribute(methodAttr, value); |
| } |
| |
| String HTMLFormElement::target() const |
| { |
| return getAttribute(targetAttr); |
| } |
| |
| bool HTMLFormElement::wasUserSubmitted() const |
| { |
| return m_wasUserSubmitted; |
| } |
| |
| HTMLFormControlElement* HTMLFormElement::defaultButton() const |
| { |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| if (!m_associatedElements[i]->isFormControlElement()) |
| continue; |
| HTMLFormControlElement* control = static_cast<HTMLFormControlElement*>(m_associatedElements[i]); |
| if (control->isSuccessfulSubmitButton()) |
| return control; |
| } |
| |
| return 0; |
| } |
| |
| bool HTMLFormElement::checkValidity() |
| { |
| Vector<RefPtr<FormAssociatedElement>> controls; |
| return !checkInvalidControlsAndCollectUnhandled(controls); |
| } |
| |
| bool HTMLFormElement::checkInvalidControlsAndCollectUnhandled(Vector<RefPtr<FormAssociatedElement>>& unhandledInvalidControls) |
| { |
| Ref<HTMLFormElement> protect(*this); |
| // Copy m_associatedElements because event handlers called from |
| // HTMLFormControlElement::checkValidity() might change m_associatedElements. |
| Vector<RefPtr<FormAssociatedElement>> elements; |
| elements.reserveCapacity(m_associatedElements.size()); |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) |
| elements.append(m_associatedElements[i]); |
| bool hasInvalidControls = false; |
| for (unsigned i = 0; i < elements.size(); ++i) { |
| if (elements[i]->form() == this && elements[i]->isFormControlElement()) { |
| HTMLFormControlElement* control = static_cast<HTMLFormControlElement*>(elements[i].get()); |
| if (!control->checkValidity(&unhandledInvalidControls) && control->form() == this) |
| hasInvalidControls = true; |
| } |
| } |
| return hasInvalidControls; |
| } |
| |
| #ifndef NDEBUG |
| void HTMLFormElement::assertItemCanBeInPastNamesMap(FormNamedItem* item) const |
| { |
| ASSERT_WITH_SECURITY_IMPLICATION(item); |
| HTMLElement& element = item->asHTMLElement(); |
| ASSERT_WITH_SECURITY_IMPLICATION(element.form() == this); |
| |
| if (item->isFormAssociatedElement()) { |
| ASSERT_WITH_SECURITY_IMPLICATION(m_associatedElements.find(static_cast<FormAssociatedElement*>(item)) != notFound); |
| return; |
| } |
| |
| ASSERT_WITH_SECURITY_IMPLICATION(element.hasTagName(imgTag)); |
| ASSERT_WITH_SECURITY_IMPLICATION(m_imageElements.find(&toHTMLImageElement(element)) != notFound); |
| } |
| #else |
| inline void HTMLFormElement::assertItemCanBeInPastNamesMap(FormNamedItem*) const |
| { |
| } |
| #endif |
| |
| HTMLElement* HTMLFormElement::elementFromPastNamesMap(const AtomicString& pastName) const |
| { |
| if (pastName.isEmpty() || !m_pastNamesMap) |
| return nullptr; |
| FormNamedItem* item = m_pastNamesMap->get(pastName.impl()); |
| if (!item) |
| return nullptr; |
| assertItemCanBeInPastNamesMap(item); |
| return &item->asHTMLElement(); |
| } |
| |
| void HTMLFormElement::addToPastNamesMap(FormNamedItem* item, const AtomicString& pastName) |
| { |
| assertItemCanBeInPastNamesMap(item); |
| if (pastName.isEmpty()) |
| return; |
| if (!m_pastNamesMap) |
| m_pastNamesMap = adoptPtr(new PastNamesMap); |
| m_pastNamesMap->set(pastName.impl(), item); |
| } |
| |
| void HTMLFormElement::removeFromPastNamesMap(FormNamedItem* item) |
| { |
| ASSERT(item); |
| if (!m_pastNamesMap) |
| return; |
| |
| PastNamesMap::iterator end = m_pastNamesMap->end(); |
| for (PastNamesMap::iterator it = m_pastNamesMap->begin(); it != end; ++it) { |
| if (it->value == item) |
| it->value = 0; // Keep looping. Single element can have multiple names. |
| } |
| } |
| |
| bool HTMLFormElement::hasNamedElement(const AtomicString& name) |
| { |
| return elements()->hasNamedItem(name) || elementFromPastNamesMap(name); |
| } |
| |
| // FIXME: Use RefPtr<HTMLElement> for namedItems. elements()->namedItems never return non-HTMLElement nodes. |
| void HTMLFormElement::getNamedElements(const AtomicString& name, Vector<Ref<Element>>& namedItems) |
| { |
| // http://www.whatwg.org/specs/web-apps/current-work/multipage/forms.html#dom-form-nameditem |
| elements()->namedItems(name, namedItems); |
| |
| HTMLElement* elementFromPast = elementFromPastNamesMap(name); |
| if (namedItems.size() == 1 && &namedItems.first().get() != elementFromPast) |
| addToPastNamesMap(toHTMLElement(&namedItems.first().get())->asFormNamedItem(), name); |
| else if (elementFromPast && namedItems.isEmpty()) |
| namedItems.append(*elementFromPast); |
| } |
| |
| void HTMLFormElement::documentDidResumeFromPageCache() |
| { |
| ASSERT(!shouldAutocomplete()); |
| |
| for (unsigned i = 0; i < m_associatedElements.size(); ++i) { |
| if (m_associatedElements[i]->isFormControlElement()) |
| static_cast<HTMLFormControlElement*>(m_associatedElements[i])->reset(); |
| } |
| } |
| |
| void HTMLFormElement::didMoveToNewDocument(Document* oldDocument) |
| { |
| if (!shouldAutocomplete()) { |
| if (oldDocument) |
| oldDocument->unregisterForPageCacheSuspensionCallbacks(this); |
| document().registerForPageCacheSuspensionCallbacks(this); |
| } |
| |
| HTMLElement::didMoveToNewDocument(oldDocument); |
| } |
| |
| bool HTMLFormElement::shouldAutocomplete() const |
| { |
| return !equalIgnoringCase(fastGetAttribute(autocompleteAttr), "off"); |
| } |
| |
| void HTMLFormElement::finishParsingChildren() |
| { |
| HTMLElement::finishParsingChildren(); |
| document().formController().restoreControlStateIn(*this); |
| } |
| |
| void HTMLFormElement::copyNonAttributePropertiesFromElement(const Element& source) |
| { |
| m_wasDemoted = static_cast<const HTMLFormElement&>(source).m_wasDemoted; |
| HTMLElement::copyNonAttributePropertiesFromElement(source); |
| } |
| |
| HTMLFormElement* HTMLFormElement::findClosestFormAncestor(const Element& startElement) |
| { |
| for (Element* element = startElement.parentElement(); element; element = element->parentElement()) { |
| if (isHTMLFormElement(element)) |
| return toHTMLFormElement(element); |
| } |
| return 0; |
| } |
| |
| } // namespace |