blob: 924d4db8765e4c888dcfb13aaf9e9c88bdff7a45 [file] [log] [blame]
/*
* 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, 2010 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 "core/html/HTMLTextAreaElement.h"
#include "bindings/core/v8/ExceptionState.h"
#include "bindings/core/v8/ExceptionStatePlaceholder.h"
#include "core/CSSValueKeywords.h"
#include "core/HTMLNames.h"
#include "core/dom/Document.h"
#include "core/dom/ExceptionCode.h"
#include "core/dom/StyleChangeReason.h"
#include "core/dom/Text.h"
#include "core/dom/shadow/ShadowRoot.h"
#include "core/editing/FrameSelection.h"
#include "core/editing/iterators/TextIterator.h"
#include "core/editing/spellcheck/SpellChecker.h"
#include "core/events/BeforeTextInsertedEvent.h"
#include "core/events/Event.h"
#include "core/frame/FrameHost.h"
#include "core/frame/LocalFrame.h"
#include "core/frame/UseCounter.h"
#include "core/html/FormData.h"
#include "core/html/forms/FormController.h"
#include "core/html/parser/HTMLParserIdioms.h"
#include "core/html/shadow/ShadowElementNames.h"
#include "core/html/shadow/TextControlInnerElements.h"
#include "core/layout/LayoutTextControlMultiLine.h"
#include "core/page/ChromeClient.h"
#include "platform/text/PlatformLocale.h"
#include "wtf/StdLibExtras.h"
#include "wtf/text/StringBuilder.h"
namespace blink {
using namespace HTMLNames;
static const unsigned defaultRows = 2;
static const unsigned defaultCols = 20;
// On submission, LF characters are converted into CRLF.
// This function returns number of characters considering this.
static unsigned numberOfLineBreaks(const String& text)
{
unsigned length = text.length();
unsigned count = 0;
for (unsigned i = 0; i < length; i++) {
if (text[i] == '\n')
count++;
}
return count;
}
static inline unsigned computeLengthForSubmission(const String& text)
{
return text.length() + numberOfLineBreaks(text);
}
HTMLTextAreaElement::HTMLTextAreaElement(Document& document, HTMLFormElement* form)
: HTMLTextFormControlElement(textareaTag, document, form)
, m_rows(defaultRows)
, m_cols(defaultCols)
, m_wrap(SoftWrap)
, m_isDirty(false)
, m_valueIsUpToDate(true)
, m_isPlaceholderVisible(false)
{
}
HTMLTextAreaElement* HTMLTextAreaElement::create(Document& document, HTMLFormElement* form)
{
HTMLTextAreaElement* textArea = new HTMLTextAreaElement(document, form);
textArea->ensureUserAgentShadowRoot();
return textArea;
}
void HTMLTextAreaElement::didAddUserAgentShadowRoot(ShadowRoot& root)
{
root.appendChild(TextControlInnerEditorElement::create(document()));
}
const AtomicString& HTMLTextAreaElement::formControlType() const
{
DEFINE_STATIC_LOCAL(const AtomicString, textarea, ("textarea"));
return textarea;
}
FormControlState HTMLTextAreaElement::saveFormControlState() const
{
return m_isDirty ? FormControlState(value()) : FormControlState();
}
void HTMLTextAreaElement::restoreFormControlState(const FormControlState& state)
{
setValue(state[0]);
}
void HTMLTextAreaElement::childrenChanged(const ChildrenChange& change)
{
HTMLElement::childrenChanged(change);
setLastChangeWasNotUserEdit();
if (m_isDirty)
setInnerEditorValue(value());
else
setNonDirtyValue(defaultValue());
}
bool HTMLTextAreaElement::isPresentationAttribute(const QualifiedName& name) const
{
if (name == alignAttr) {
// Don't map 'align' attribute. This matches what Firefox, Opera and IE do.
// See http://bugs.webkit.org/show_bug.cgi?id=7075
return false;
}
if (name == wrapAttr)
return true;
return HTMLTextFormControlElement::isPresentationAttribute(name);
}
void HTMLTextAreaElement::collectStyleForPresentationAttribute(const QualifiedName& name, const AtomicString& value, MutableStylePropertySet* style)
{
if (name == wrapAttr) {
if (shouldWrapText()) {
addPropertyToPresentationAttributeStyle(style, CSSPropertyWhiteSpace, CSSValuePreWrap);
addPropertyToPresentationAttributeStyle(style, CSSPropertyWordWrap, CSSValueBreakWord);
} else {
addPropertyToPresentationAttributeStyle(style, CSSPropertyWhiteSpace, CSSValuePre);
addPropertyToPresentationAttributeStyle(style, CSSPropertyWordWrap, CSSValueNormal);
}
} else {
HTMLTextFormControlElement::collectStyleForPresentationAttribute(name, value, style);
}
}
void HTMLTextAreaElement::parseAttribute(const QualifiedName& name, const AtomicString& oldValue, const AtomicString& value)
{
if (name == rowsAttr) {
unsigned rows = 0;
if (value.isEmpty() || !parseHTMLNonNegativeInteger(value, rows) || rows <= 0)
rows = defaultRows;
if (m_rows != rows) {
m_rows = rows;
if (layoutObject())
layoutObject()->setNeedsLayoutAndPrefWidthsRecalcAndFullPaintInvalidation(LayoutInvalidationReason::AttributeChanged);
}
} else if (name == colsAttr) {
unsigned cols = 0;
if (value.isEmpty() || !parseHTMLNonNegativeInteger(value, cols) || cols <= 0)
cols = defaultCols;
if (m_cols != cols) {
m_cols = cols;
if (LayoutObject* layoutObject = this->layoutObject())
layoutObject->setNeedsLayoutAndPrefWidthsRecalcAndFullPaintInvalidation(LayoutInvalidationReason::AttributeChanged);
}
} else if (name == wrapAttr) {
// The virtual/physical values were a Netscape extension of HTML 3.0, now deprecated.
// The soft/hard /off values are a recommendation for HTML 4 extension by IE and NS 4.
WrapMethod wrap;
if (equalIgnoringCase(value, "physical") || equalIgnoringCase(value, "hard") || equalIgnoringCase(value, "on"))
wrap = HardWrap;
else if (equalIgnoringCase(value, "off"))
wrap = NoWrap;
else
wrap = SoftWrap;
if (wrap != m_wrap) {
m_wrap = wrap;
if (LayoutObject* layoutObject = this->layoutObject())
layoutObject->setNeedsLayoutAndPrefWidthsRecalcAndFullPaintInvalidation(LayoutInvalidationReason::AttributeChanged);
}
} else if (name == accesskeyAttr) {
// ignore for the moment
} else if (name == maxlengthAttr) {
UseCounter::count(document(), UseCounter::TextAreaMaxLength);
setNeedsValidityCheck();
} else if (name == minlengthAttr) {
UseCounter::count(document(), UseCounter::TextAreaMinLength);
setNeedsValidityCheck();
} else {
HTMLTextFormControlElement::parseAttribute(name, oldValue, value);
}
}
LayoutObject* HTMLTextAreaElement::createLayoutObject(const ComputedStyle&)
{
return new LayoutTextControlMultiLine(this);
}
void HTMLTextAreaElement::appendToFormData(FormData& formData)
{
if (name().isEmpty())
return;
document().updateStyleAndLayout();
const String& text = (m_wrap == HardWrap) ? valueWithHardLineBreaks() : value();
formData.append(name(), text);
const AtomicString& dirnameAttrValue = fastGetAttribute(dirnameAttr);
if (!dirnameAttrValue.isNull())
formData.append(dirnameAttrValue, directionForFormData());
}
void HTMLTextAreaElement::resetImpl()
{
setNonDirtyValue(defaultValue());
}
bool HTMLTextAreaElement::hasCustomFocusLogic() const
{
return true;
}
bool HTMLTextAreaElement::isKeyboardFocusable() const
{
// If a given text area can be focused at all, then it will always be keyboard focusable.
return isFocusable();
}
bool HTMLTextAreaElement::shouldShowFocusRingOnMouseFocus() const
{
return true;
}
void HTMLTextAreaElement::updateFocusAppearance(SelectionBehaviorOnFocus selectionBehavior)
{
switch (selectionBehavior) {
case SelectionBehaviorOnFocus::Reset: // Fallthrough.
case SelectionBehaviorOnFocus::Restore:
restoreCachedSelection();
break;
case SelectionBehaviorOnFocus::None:
return;
}
if (document().frame())
document().frame()->selection().revealSelection();
}
void HTMLTextAreaElement::defaultEventHandler(Event* event)
{
if (layoutObject() && (event->isMouseEvent() || event->isDragEvent() || event->hasInterface(EventNames::WheelEvent) || event->type() == EventTypeNames::blur))
forwardEvent(event);
else if (layoutObject() && event->isBeforeTextInsertedEvent())
handleBeforeTextInsertedEvent(static_cast<BeforeTextInsertedEvent*>(event));
HTMLTextFormControlElement::defaultEventHandler(event);
}
void HTMLTextAreaElement::handleFocusEvent(Element*, WebFocusType)
{
if (LocalFrame* frame = document().frame())
frame->spellChecker().didBeginEditing(this);
}
void HTMLTextAreaElement::subtreeHasChanged()
{
#if ENABLE(ASSERT)
// The innerEditor should have either Text nodes or a placeholder break
// element. If we see other nodes, it's a bug in editing code and we should
// fix it.
Element* innerEditor = innerEditorElement();
for (Node& node : NodeTraversal::descendantsOf(*innerEditor)) {
if (node.isTextNode())
continue;
ASSERT(isHTMLBRElement(node));
ASSERT(&node == innerEditor->lastChild());
}
#endif
addPlaceholderBreakElementIfNecessary();
setChangedSinceLastFormControlChangeEvent(true);
m_valueIsUpToDate = false;
setNeedsValidityCheck();
setAutofilled(false);
updatePlaceholderVisibility();
if (!focused())
return;
// When typing in a textarea, childrenChanged is not called, so we need to force the directionality check.
calculateAndAdjustDirectionality();
ASSERT(document().isActive());
document().frameHost()->chromeClient().didChangeValueInTextField(*this);
}
void HTMLTextAreaElement::handleBeforeTextInsertedEvent(BeforeTextInsertedEvent* event) const
{
ASSERT(event);
ASSERT(layoutObject());
int signedMaxLength = maxLength();
if (signedMaxLength < 0)
return;
unsigned unsignedMaxLength = static_cast<unsigned>(signedMaxLength);
const String& currentValue = innerEditorValue();
unsigned currentLength = computeLengthForSubmission(currentValue);
if (currentLength + computeLengthForSubmission(event->text()) < unsignedMaxLength)
return;
// selectionLength represents the selection length of this text field to be
// removed by this insertion.
// If the text field has no focus, we don't need to take account of the
// selection length. The selection is the source of text drag-and-drop in
// that case, and nothing in the text field will be removed.
unsigned selectionLength = 0;
if (focused()) {
selectionLength = computeLengthForSubmission(document().frame()->selection().selectedText());
}
ASSERT(currentLength >= selectionLength);
unsigned baseLength = currentLength - selectionLength;
unsigned appendableLength = unsignedMaxLength > baseLength ? unsignedMaxLength - baseLength : 0;
event->setText(sanitizeUserInputValue(event->text(), appendableLength));
}
String HTMLTextAreaElement::sanitizeUserInputValue(const String& proposedValue, unsigned maxLength)
{
unsigned submissionLength = 0;
unsigned i = 0;
for (; i < proposedValue.length(); ++i) {
submissionLength += proposedValue[i] == '\n' ? 2 : 1;
if (submissionLength == maxLength) {
++i;
break;
}
if (submissionLength > maxLength)
break;
}
if (i > 0 && U16_IS_LEAD(proposedValue[i - 1]))
--i;
return proposedValue.left(i);
}
void HTMLTextAreaElement::updateValue() const
{
if (m_valueIsUpToDate)
return;
m_value = innerEditorValue();
const_cast<HTMLTextAreaElement*>(this)->m_valueIsUpToDate = true;
const_cast<HTMLTextAreaElement*>(this)->notifyFormStateChanged();
m_isDirty = true;
const_cast<HTMLTextAreaElement*>(this)->updatePlaceholderVisibility();
}
String HTMLTextAreaElement::value() const
{
updateValue();
return m_value;
}
void HTMLTextAreaElement::setValue(const String& value, TextFieldEventBehavior eventBehavior)
{
setValueCommon(value, eventBehavior);
m_isDirty = true;
if (document().focusedElement() == this)
document().frameHost()->chromeClient().didUpdateTextOfFocusedElementByNonUserInput();
}
void HTMLTextAreaElement::setNonDirtyValue(const String& value)
{
setValueCommon(value, DispatchNoEvent, SetSeletion);
m_isDirty = false;
}
void HTMLTextAreaElement::setValueCommon(const String& newValue, TextFieldEventBehavior eventBehavior, SetValueCommonOption setValueOption)
{
// Code elsewhere normalizes line endings added by the user via the keyboard or pasting.
// We normalize line endings coming from JavaScript here.
String normalizedValue = newValue.isNull() ? "" : newValue;
normalizedValue.replace("\r\n", "\n");
normalizedValue.replace('\r', '\n');
// Return early because we don't want to trigger other side effects
// when the value isn't changing.
// FIXME: Simple early return doesn't match the Firefox ever.
// Remove these lines.
if (normalizedValue == value()) {
if (setValueOption == SetSeletion) {
setNeedsValidityCheck();
if (isFinishedParsingChildren()) {
// Set the caret to the end of the text value except for initialize.
unsigned endOfString = m_value.length();
setSelectionRange(endOfString, endOfString, SelectionHasNoDirection, NotDispatchSelectEvent, ChangeSelectionIfFocused);
}
}
return;
}
m_value = normalizedValue;
setInnerEditorValue(m_value);
if (eventBehavior == DispatchNoEvent)
setLastChangeWasNotUserEdit();
updatePlaceholderVisibility();
setNeedsStyleRecalc(SubtreeStyleChange, StyleChangeReasonForTracing::create(StyleChangeReason::ControlValue));
m_suggestedValue = String();
setNeedsValidityCheck();
if (isFinishedParsingChildren()) {
// Set the caret to the end of the text value except for initialize.
unsigned endOfString = m_value.length();
setSelectionRange(endOfString, endOfString, SelectionHasNoDirection, NotDispatchSelectEvent, ChangeSelectionIfFocused);
}
notifyFormStateChanged();
if (eventBehavior == DispatchNoEvent) {
setTextAsOfLastFormControlChangeEvent(normalizedValue);
} else {
if (eventBehavior == DispatchInputAndChangeEvent)
dispatchFormControlInputEvent();
dispatchFormControlChangeEvent();
}
}
void HTMLTextAreaElement::setInnerEditorValue(const String& value)
{
HTMLTextFormControlElement::setInnerEditorValue(value);
m_valueIsUpToDate = true;
}
String HTMLTextAreaElement::defaultValue() const
{
StringBuilder value;
// Since there may be comments, ignore nodes other than text nodes.
for (Node* n = firstChild(); n; n = n->nextSibling()) {
if (n->isTextNode())
value.append(toText(n)->data());
}
return value.toString();
}
void HTMLTextAreaElement::setDefaultValue(const String& defaultValue)
{
// To preserve comments, remove only the text nodes, then add a single text node.
HeapVector<Member<Node>> textNodes;
for (Node* n = firstChild(); n; n = n->nextSibling()) {
if (n->isTextNode())
textNodes.append(n);
}
size_t size = textNodes.size();
for (size_t i = 0; i < size; ++i)
removeChild(textNodes[i].get(), IGNORE_EXCEPTION);
// Normalize line endings.
String value = defaultValue;
value.replace("\r\n", "\n");
value.replace('\r', '\n');
insertBefore(document().createTextNode(value), firstChild(), IGNORE_EXCEPTION);
if (!m_isDirty)
setNonDirtyValue(value);
}
int HTMLTextAreaElement::maxLength() const
{
int value;
if (!parseHTMLInteger(getAttribute(maxlengthAttr), value))
return -1;
return value >= 0 ? value : -1;
}
int HTMLTextAreaElement::minLength() const
{
int value;
if (!parseHTMLInteger(getAttribute(minlengthAttr), value))
return -1;
return value >= 0 ? value : -1;
}
void HTMLTextAreaElement::setMaxLength(int newValue, ExceptionState& exceptionState)
{
int min = minLength();
if (newValue < 0)
exceptionState.throwDOMException(IndexSizeError, "The value provided (" + String::number(newValue) + ") is not positive or 0.");
else if (min >= 0 && newValue < min)
exceptionState.throwDOMException(IndexSizeError, ExceptionMessages::indexExceedsMinimumBound("maxLength", newValue, min));
else
setIntegralAttribute(maxlengthAttr, newValue);
}
void HTMLTextAreaElement::setMinLength(int newValue, ExceptionState& exceptionState)
{
int max = maxLength();
if (newValue < 0)
exceptionState.throwDOMException(IndexSizeError, "The value provided (" + String::number(newValue) + ") is not positive or 0.");
else if (max >= 0 && newValue > max)
exceptionState.throwDOMException(IndexSizeError, ExceptionMessages::indexExceedsMaximumBound("minLength", newValue, max));
else
setIntegralAttribute(minlengthAttr, newValue);
}
String HTMLTextAreaElement::suggestedValue() const
{
return m_suggestedValue;
}
void HTMLTextAreaElement::setSuggestedValue(const String& value)
{
m_suggestedValue = value;
if (!value.isNull())
setInnerEditorValue(m_suggestedValue);
else
setInnerEditorValue(m_value);
updatePlaceholderVisibility();
setNeedsStyleRecalc(SubtreeStyleChange, StyleChangeReasonForTracing::create(StyleChangeReason::ControlValue));
}
String HTMLTextAreaElement::validationMessage() const
{
if (!willValidate())
return String();
if (customError())
return customValidationMessage();
if (valueMissing())
return locale().queryString(WebLocalizedString::ValidationValueMissing);
if (tooLong())
return locale().validationMessageTooLongText(computeLengthForSubmission(value()), maxLength());
if (tooShort())
return locale().validationMessageTooShortText(computeLengthForSubmission(value()), minLength());
return String();
}
bool HTMLTextAreaElement::valueMissing() const
{
// We should not call value() for performance.
return willValidate() && valueMissing(nullptr);
}
bool HTMLTextAreaElement::valueMissing(const String* value) const
{
return isRequiredFormControl() && !isDisabledOrReadOnly() && (value ? *value : this->value()).isEmpty();
}
bool HTMLTextAreaElement::tooLong() const
{
// We should not call value() for performance.
return willValidate() && tooLong(nullptr, CheckDirtyFlag);
}
bool HTMLTextAreaElement::tooShort() const
{
// We should not call value() for performance.
return willValidate() && tooShort(nullptr, CheckDirtyFlag);
}
bool HTMLTextAreaElement::tooLong(const String* value, NeedsToCheckDirtyFlag check) const
{
// Return false for the default value or value set by script even if it is
// longer than maxLength.
if (check == CheckDirtyFlag && !lastChangeWasUserEdit())
return false;
int max = maxLength();
if (max < 0)
return false;
return computeLengthForSubmission(value ? *value : this->value()) > static_cast<unsigned>(max);
}
bool HTMLTextAreaElement::tooShort(const String* value, NeedsToCheckDirtyFlag check) const
{
// Return false for the default value or value set by script even if it is
// shorter than minLength.
if (check == CheckDirtyFlag && !lastChangeWasUserEdit())
return false;
int min = minLength();
if (min <= 0)
return false;
// An empty string is excluded from minlength check.
unsigned len = computeLengthForSubmission(value ? *value : this->value());
return len > 0 && len < static_cast<unsigned>(min);
}
bool HTMLTextAreaElement::isValidValue(const String& candidate) const
{
return !valueMissing(&candidate) && !tooLong(&candidate, IgnoreDirtyFlag) && !tooShort(&candidate, IgnoreDirtyFlag);
}
void HTMLTextAreaElement::accessKeyAction(bool)
{
focus();
}
void HTMLTextAreaElement::setCols(unsigned cols)
{
setUnsignedIntegralAttribute(colsAttr, cols);
}
void HTMLTextAreaElement::setRows(unsigned rows)
{
setUnsignedIntegralAttribute(rowsAttr, rows);
}
bool HTMLTextAreaElement::matchesReadOnlyPseudoClass() const
{
return isReadOnly();
}
bool HTMLTextAreaElement::matchesReadWritePseudoClass() const
{
return !isReadOnly();
}
void HTMLTextAreaElement::setPlaceholderVisibility(bool visible)
{
m_isPlaceholderVisible = visible;
}
void HTMLTextAreaElement::updatePlaceholderText()
{
HTMLElement* placeholder = placeholderElement();
const AtomicString& placeholderText = fastGetAttribute(placeholderAttr);
if (placeholderText.isEmpty()) {
if (placeholder)
userAgentShadowRoot()->removeChild(placeholder);
return;
}
if (!placeholder) {
HTMLDivElement* newElement = HTMLDivElement::create(document());
placeholder = newElement;
placeholder->setShadowPseudoId(AtomicString("-webkit-input-placeholder"));
placeholder->setAttribute(idAttr, ShadowElementNames::placeholder());
placeholder->setInlineStyleProperty(CSSPropertyDisplay, isPlaceholderVisible() ? CSSValueBlock : CSSValueNone, true);
userAgentShadowRoot()->insertBefore(placeholder, innerEditorElement());
}
placeholder->setTextContent(placeholderText);
}
bool HTMLTextAreaElement::isInteractiveContent() const
{
return true;
}
bool HTMLTextAreaElement::supportsAutofocus() const
{
return true;
}
const AtomicString& HTMLTextAreaElement::defaultAutocapitalize() const
{
DEFINE_STATIC_LOCAL(const AtomicString, sentences, ("sentences"));
return sentences;
}
void HTMLTextAreaElement::copyNonAttributePropertiesFromElement(const Element& source)
{
const HTMLTextAreaElement& sourceElement = static_cast<const HTMLTextAreaElement&>(source);
setValueCommon(sourceElement.value(), DispatchNoEvent, SetSeletion);
m_isDirty = sourceElement.m_isDirty;
HTMLTextFormControlElement::copyNonAttributePropertiesFromElement(source);
}
} // namespace blink