This source file includes following definitions.
- numberOfLineBreaks
- computeLengthForSubmission
- m_isDirty
- create
- didAddUserAgentShadowRoot
- formControlType
- saveFormControlState
- restoreFormControlState
- childrenChanged
- isPresentationAttribute
- collectStyleForPresentationAttribute
- parseAttribute
- createRenderer
- appendFormData
- resetImpl
- hasCustomFocusLogic
- isKeyboardFocusable
- shouldShowFocusRingOnMouseFocus
- updateFocusAppearance
- defaultEventHandler
- handleFocusEvent
- subtreeHasChanged
- handleBeforeTextInsertedEvent
- sanitizeUserInputValue
- updateValue
- value
- setValue
- setNonDirtyValue
- setValueCommon
- defaultValue
- setDefaultValue
- maxLength
- setMaxLength
- suggestedValue
- setSuggestedValue
- validationMessage
- valueMissing
- tooLong
- tooLong
- isValidValue
- accessKeyAction
- setCols
- setRows
- shouldUseInputMethod
- matchesReadOnlyPseudoClass
- matchesReadWritePseudoClass
- updatePlaceholderText
- isInteractiveContent
- supportsAutofocus
#include "config.h"
#include "core/html/HTMLTextAreaElement.h"
#include "CSSValueKeywords.h"
#include "HTMLNames.h"
#include "bindings/v8/ExceptionState.h"
#include "bindings/v8/ExceptionStatePlaceholder.h"
#include "core/dom/Document.h"
#include "core/dom/ExceptionCode.h"
#include "core/dom/Text.h"
#include "core/dom/shadow/ShadowRoot.h"
#include "core/editing/FrameSelection.h"
#include "core/editing/SpellChecker.h"
#include "core/editing/TextIterator.h"
#include "core/events/BeforeTextInsertedEvent.h"
#include "core/events/Event.h"
#include "core/frame/FrameHost.h"
#include "core/frame/LocalFrame.h"
#include "core/html/FormDataList.h"
#include "core/html/forms/FormController.h"
#include "core/html/shadow/ShadowElementNames.h"
#include "core/html/shadow/TextControlInnerElements.h"
#include "core/page/Chrome.h"
#include "core/page/ChromeClient.h"
#include "core/rendering/RenderTextControlMultiLine.h"
#include "platform/text/PlatformLocale.h"
#include "wtf/StdLibExtras.h"
#include "wtf/text/StringBuilder.h"
namespace WebCore {
using namespace HTMLNames;
static const int defaultRows = 2;
static const int defaultCols = 20;
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)
{
setFormControlValueMatchesRenderer(true);
ScriptWrappable::init(this);
}
PassRefPtr<HTMLTextAreaElement> HTMLTextAreaElement::create(Document& document, HTMLFormElement* form)
{
RefPtr<HTMLTextAreaElement> textArea = adoptRef(new HTMLTextAreaElement(document, form));
textArea->ensureUserAgentShadowRoot();
return textArea.release();
}
void HTMLTextAreaElement::didAddUserAgentShadowRoot(ShadowRoot& root)
{
root.appendChild(TextControlInnerTextElement::create(document()));
}
const AtomicString& HTMLTextAreaElement::formControlType() const
{
DEFINE_STATIC_LOCAL(const AtomicString, textarea, ("textarea", AtomicString::ConstructFromLiteral));
return textarea;
}
FormControlState HTMLTextAreaElement::saveFormControlState() const
{
return m_isDirty ? FormControlState(value()) : FormControlState();
}
void HTMLTextAreaElement::restoreFormControlState(const FormControlState& state)
{
setValue(state[0]);
}
void HTMLTextAreaElement::childrenChanged(bool changedByParser, Node* beforeChange, Node* afterChange, int childCountDelta)
{
HTMLElement::childrenChanged(changedByParser, beforeChange, afterChange, childCountDelta);
setLastChangeWasNotUserEdit();
if (m_isDirty)
setInnerTextValue(value());
else
setNonDirtyValue(defaultValue());
}
bool HTMLTextAreaElement::isPresentationAttribute(const QualifiedName& name) const
{
if (name == alignAttr) {
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& value)
{
if (name == rowsAttr) {
int rows = value.toInt();
if (rows <= 0)
rows = defaultRows;
if (m_rows != rows) {
m_rows = rows;
if (renderer())
renderer()->setNeedsLayoutAndPrefWidthsRecalc();
}
} else if (name == colsAttr) {
int cols = value.toInt();
if (cols <= 0)
cols = defaultCols;
if (m_cols != cols) {
m_cols = cols;
if (renderer())
renderer()->setNeedsLayoutAndPrefWidthsRecalc();
}
} else if (name == wrapAttr) {
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 (renderer())
renderer()->setNeedsLayoutAndPrefWidthsRecalc();
}
} else if (name == accesskeyAttr) {
} else if (name == maxlengthAttr)
setNeedsValidityCheck();
else
HTMLTextFormControlElement::parseAttribute(name, value);
}
RenderObject* HTMLTextAreaElement::createRenderer(RenderStyle*)
{
return new RenderTextControlMultiLine(this);
}
bool HTMLTextAreaElement::appendFormData(FormDataList& encoding, bool)
{
if (name().isEmpty())
return false;
document().updateLayout();
const String& text = (m_wrap == HardWrap) ? valueWithHardLineBreaks() : value();
encoding.appendData(name(), text);
const AtomicString& dirnameAttrValue = fastGetAttribute(dirnameAttr);
if (!dirnameAttrValue.isNull())
encoding.appendData(dirnameAttrValue, directionForFormData());
return true;
}
void HTMLTextAreaElement::resetImpl()
{
setNonDirtyValue(defaultValue());
}
bool HTMLTextAreaElement::hasCustomFocusLogic() const
{
return true;
}
bool HTMLTextAreaElement::isKeyboardFocusable() const
{
return isFocusable();
}
bool HTMLTextAreaElement::shouldShowFocusRingOnMouseFocus() const
{
return true;
}
void HTMLTextAreaElement::updateFocusAppearance(bool restorePreviousSelection)
{
if (!restorePreviousSelection || !hasCachedSelection()) {
setSelectionRange(0, 0);
} else
restoreCachedSelection();
if (document().frame())
document().frame()->selection().revealSelection();
}
void HTMLTextAreaElement::defaultEventHandler(Event* event)
{
if (renderer() && (event->isMouseEvent() || event->isDragEvent() || event->hasInterface(EventNames::WheelEvent) || event->type() == EventTypeNames::blur))
forwardEvent(event);
else if (renderer() && event->isBeforeTextInsertedEvent())
handleBeforeTextInsertedEvent(static_cast<BeforeTextInsertedEvent*>(event));
HTMLTextFormControlElement::defaultEventHandler(event);
}
void HTMLTextAreaElement::handleFocusEvent(Element*, FocusType)
{
if (LocalFrame* frame = document().frame())
frame->spellChecker().didBeginEditing(this);
}
void HTMLTextAreaElement::subtreeHasChanged()
{
setChangedSinceLastFormControlChangeEvent(true);
setFormControlValueMatchesRenderer(false);
setNeedsValidityCheck();
if (!focused())
return;
calculateAndAdjustDirectionality();
ASSERT(document().isActive());
document().frameHost()->chrome().client().didChangeValueInTextField(*this);
}
void HTMLTextAreaElement::handleBeforeTextInsertedEvent(BeforeTextInsertedEvent* event) const
{
ASSERT(event);
ASSERT(renderer());
int signedMaxLength = maxLength();
if (signedMaxLength < 0)
return;
unsigned unsignedMaxLength = static_cast<unsigned>(signedMaxLength);
const String& currentValue = innerTextValue();
unsigned currentLength = computeLengthForSubmission(currentValue);
if (currentLength + computeLengthForSubmission(event->text()) < unsignedMaxLength)
return;
unsigned selectionLength = focused() ? computeLengthForSubmission(plainText(document().frame()->selection().selection().toNormalizedRange().get())) : 0;
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)
{
if (maxLength > 0 && U16_IS_LEAD(proposedValue[maxLength - 1]))
--maxLength;
return proposedValue.left(maxLength);
}
void HTMLTextAreaElement::updateValue() const
{
if (formControlValueMatchesRenderer())
return;
ASSERT(renderer());
m_value = innerTextValue();
const_cast<HTMLTextAreaElement*>(this)->setFormControlValueMatchesRenderer(true);
const_cast<HTMLTextAreaElement*>(this)->notifyFormStateChanged();
m_isDirty = true;
const_cast<HTMLTextAreaElement*>(this)->updatePlaceholderVisibility(false);
}
String HTMLTextAreaElement::value() const
{
updateValue();
return m_value;
}
void HTMLTextAreaElement::setValue(const String& value, TextFieldEventBehavior eventBehavior)
{
RefPtr<HTMLTextAreaElement> protector(this);
setValueCommon(value, eventBehavior);
m_isDirty = true;
setNeedsValidityCheck();
}
void HTMLTextAreaElement::setNonDirtyValue(const String& value)
{
setValueCommon(value, DispatchNoEvent);
m_isDirty = false;
setNeedsValidityCheck();
}
void HTMLTextAreaElement::setValueCommon(const String& newValue, TextFieldEventBehavior eventBehavior)
{
String normalizedValue = newValue.isNull() ? "" : newValue;
normalizedValue.replace("\r\n", "\n");
normalizedValue.replace('\r', '\n');
if (normalizedValue == value())
return;
m_value = normalizedValue;
setInnerTextValue(m_value);
if (eventBehavior == DispatchNoEvent)
setLastChangeWasNotUserEdit();
updatePlaceholderVisibility(false);
setNeedsStyleRecalc(SubtreeStyleChange);
setFormControlValueMatchesRenderer(true);
m_suggestedValue = String();
if (document().focusedElement() == this) {
unsigned endOfString = m_value.length();
setSelectionRange(endOfString, endOfString);
}
notifyFormStateChanged();
if (eventBehavior == DispatchNoEvent) {
setTextAsOfLastFormControlChangeEvent(normalizedValue);
} else {
if (eventBehavior == DispatchInputAndChangeEvent)
dispatchFormControlInputEvent();
dispatchFormControlChangeEvent();
}
}
String HTMLTextAreaElement::defaultValue() const
{
StringBuilder value;
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)
{
RefPtr<Node> protectFromMutationEvents(this);
Vector<RefPtr<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);
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
{
bool ok;
int value = getAttribute(maxlengthAttr).string().toInt(&ok);
return ok && value >= 0 ? value : -1;
}
void HTMLTextAreaElement::setMaxLength(int newValue, ExceptionState& exceptionState)
{
if (newValue < 0)
exceptionState.throwDOMException(IndexSizeError, "The value provided (" + String::number(newValue) + ") is not positive or 0.");
else
setIntegralAttribute(maxlengthAttr, newValue);
}
String HTMLTextAreaElement::suggestedValue() const
{
return m_suggestedValue;
}
void HTMLTextAreaElement::setSuggestedValue(const String& value)
{
m_suggestedValue = value;
if (!value.isNull())
setInnerTextValue(m_suggestedValue);
else
setInnerTextValue(m_value);
updatePlaceholderVisibility(false);
setNeedsStyleRecalc(SubtreeStyleChange);
}
String HTMLTextAreaElement::validationMessage() const
{
if (!willValidate())
return String();
if (customError())
return customValidationMessage();
if (valueMissing())
return locale().queryString(blink::WebLocalizedString::ValidationValueMissing);
if (tooLong())
return locale().validationMessageTooLongText(computeLengthForSubmission(value()), maxLength());
return String();
}
bool HTMLTextAreaElement::valueMissing() const
{
return willValidate() && valueMissing(value());
}
bool HTMLTextAreaElement::tooLong() const
{
return willValidate() && tooLong(value(), CheckDirtyFlag);
}
bool HTMLTextAreaElement::tooLong(const String& value, NeedsToCheckDirtyFlag check) const
{
if (check == CheckDirtyFlag && !lastChangeWasUserEdit())
return false;
int max = maxLength();
if (max < 0)
return false;
return computeLengthForSubmission(value) > static_cast<unsigned>(max);
}
bool HTMLTextAreaElement::isValidValue(const String& candidate) const
{
return !valueMissing(candidate) && !tooLong(candidate, IgnoreDirtyFlag);
}
void HTMLTextAreaElement::accessKeyAction(bool)
{
focus();
}
void HTMLTextAreaElement::setCols(int cols)
{
setIntegralAttribute(colsAttr, cols);
}
void HTMLTextAreaElement::setRows(int rows)
{
setIntegralAttribute(rowsAttr, rows);
}
bool HTMLTextAreaElement::shouldUseInputMethod()
{
return true;
}
bool HTMLTextAreaElement::matchesReadOnlyPseudoClass() const
{
return isReadOnly();
}
bool HTMLTextAreaElement::matchesReadWritePseudoClass() const
{
return !isReadOnly();
}
void HTMLTextAreaElement::updatePlaceholderText()
{
HTMLElement* placeholder = placeholderElement();
String placeholderText = strippedPlaceholder();
if (placeholderText.isEmpty()) {
if (placeholder)
userAgentShadowRoot()->removeChild(placeholder);
return;
}
if (!placeholder) {
RefPtr<HTMLDivElement> newElement = HTMLDivElement::create(document());
placeholder = newElement.get();
placeholder->setShadowPseudoId(AtomicString("-webkit-input-placeholder", AtomicString::ConstructFromLiteral));
placeholder->setAttribute(idAttr, ShadowElementNames::placeholder());
userAgentShadowRoot()->insertBefore(placeholder, innerTextElement()->nextSibling());
}
placeholder->setTextContent(placeholderText);
}
bool HTMLTextAreaElement::isInteractiveContent() const
{
return true;
}
bool HTMLTextAreaElement::supportsAutofocus() const
{
return true;
}
}