summaryrefslogtreecommitdiffstats
path: root/dom/html/HTMLTextAreaElement.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'dom/html/HTMLTextAreaElement.cpp')
-rw-r--r--dom/html/HTMLTextAreaElement.cpp1192
1 files changed, 1192 insertions, 0 deletions
diff --git a/dom/html/HTMLTextAreaElement.cpp b/dom/html/HTMLTextAreaElement.cpp
new file mode 100644
index 0000000000..700212351b
--- /dev/null
+++ b/dom/html/HTMLTextAreaElement.cpp
@@ -0,0 +1,1192 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "mozilla/dom/HTMLTextAreaElement.h"
+
+#include "mozAutoDocUpdate.h"
+#include "mozilla/AsyncEventDispatcher.h"
+#include "mozilla/Attributes.h"
+#include "mozilla/dom/FormData.h"
+#include "mozilla/dom/HTMLTextAreaElementBinding.h"
+#include "mozilla/dom/MutationEventBinding.h"
+#include "mozilla/EventDispatcher.h"
+#include "mozilla/MappedDeclarations.h"
+#include "mozilla/MouseEvents.h"
+#include "mozilla/PresState.h"
+#include "mozilla/TextControlState.h"
+#include "nsAttrValueInlines.h"
+#include "nsBaseCommandController.h"
+#include "nsContentCID.h"
+#include "nsContentCreatorFunctions.h"
+#include "nsError.h"
+#include "nsFocusManager.h"
+#include "nsIConstraintValidation.h"
+#include "nsIControllers.h"
+#include "mozilla/dom/Document.h"
+#include "nsIFormControlFrame.h"
+#include "nsIFormControl.h"
+#include "nsIFrame.h"
+#include "nsITextControlFrame.h"
+#include "nsLayoutUtils.h"
+#include "nsLinebreakConverter.h"
+#include "nsMappedAttributes.h"
+#include "nsPIDOMWindow.h"
+#include "nsPresContext.h"
+#include "nsReadableUtils.h"
+#include "nsStyleConsts.h"
+#include "nsTextControlFrame.h"
+#include "nsThreadUtils.h"
+#include "nsXULControllers.h"
+
+NS_IMPL_NS_NEW_HTML_ELEMENT_CHECK_PARSER(TextArea)
+
+namespace mozilla::dom {
+
+HTMLTextAreaElement::HTMLTextAreaElement(
+ already_AddRefed<mozilla::dom::NodeInfo>&& aNodeInfo,
+ FromParser aFromParser)
+ : TextControlElement(std::move(aNodeInfo), aFromParser,
+ FormControlType::Textarea),
+ mValueChanged(false),
+ mLastValueChangeWasInteractive(false),
+ mHandlingSelect(false),
+ mDoneAddingChildren(!aFromParser),
+ mInhibitStateRestoration(!!(aFromParser & FROM_PARSER_FRAGMENT)),
+ mDisabledChanged(false),
+ mCanShowInvalidUI(true),
+ mCanShowValidUI(true),
+ mIsPreviewEnabled(false),
+ mAutocompleteAttrState(nsContentUtils::eAutocompleteAttrState_Unknown),
+ mState(TextControlState::Construct(this)) {
+ AddMutationObserver(this);
+
+ // Set up our default state. By default we're enabled (since we're
+ // a control type that can be disabled but not actually disabled
+ // right now), optional, and valid. We are NOT readwrite by default
+ // until someone calls UpdateEditableState on us, apparently! Also
+ // by default we don't have to show validity UI and so forth.
+ AddStatesSilently(ElementState::ENABLED | ElementState::OPTIONAL_ |
+ ElementState::VALID | ElementState::VALUE_EMPTY);
+}
+
+HTMLTextAreaElement::~HTMLTextAreaElement() {
+ mState->Destroy();
+ mState = nullptr;
+}
+
+NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLTextAreaElement)
+
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLTextAreaElement,
+ TextControlElement)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mValidity)
+ NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mControllers)
+ if (tmp->mState) {
+ tmp->mState->Traverse(cb);
+ }
+NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
+
+NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLTextAreaElement,
+ TextControlElement)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mValidity)
+ NS_IMPL_CYCLE_COLLECTION_UNLINK(mControllers)
+ if (tmp->mState) {
+ tmp->mState->Unlink();
+ }
+NS_IMPL_CYCLE_COLLECTION_UNLINK_END
+
+NS_IMPL_ISUPPORTS_CYCLE_COLLECTION_INHERITED(HTMLTextAreaElement,
+ TextControlElement,
+ nsIMutationObserver,
+ nsIConstraintValidation)
+
+// nsIDOMHTMLTextAreaElement
+
+nsresult HTMLTextAreaElement::Clone(dom::NodeInfo* aNodeInfo,
+ nsINode** aResult) const {
+ *aResult = nullptr;
+ RefPtr<HTMLTextAreaElement> it = new (aNodeInfo->NodeInfoManager())
+ HTMLTextAreaElement(do_AddRef(aNodeInfo));
+
+ nsresult rv = const_cast<HTMLTextAreaElement*>(this)->CopyInnerTo(it);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ if (mValueChanged) {
+ // Set our value on the clone.
+ nsAutoString value;
+ GetValueInternal(value, true);
+
+ // SetValueInternal handles setting mValueChanged for us
+ if (NS_WARN_IF(
+ NS_FAILED(rv = it->SetValueInternal(
+ value, {ValueSetterOption::SetValueChanged})))) {
+ return rv;
+ }
+ }
+
+ it->SetLastValueChangeWasInteractive(mLastValueChangeWasInteractive);
+ it.forget(aResult);
+ return NS_OK;
+}
+
+// nsIContent
+
+void HTMLTextAreaElement::Select() {
+ if (FocusState() != FocusTristate::eUnfocusable) {
+ if (RefPtr<nsFocusManager> fm = nsFocusManager::GetFocusManager()) {
+ fm->SetFocus(this, nsIFocusManager::FLAG_NOSCROLL);
+ }
+ }
+
+ SetSelectionRange(0, UINT32_MAX, mozilla::dom::Optional<nsAString>(),
+ IgnoreErrors());
+}
+
+NS_IMETHODIMP
+HTMLTextAreaElement::SelectAll(nsPresContext* aPresContext) {
+ nsIFormControlFrame* formControlFrame = GetFormControlFrame(true);
+
+ if (formControlFrame) {
+ formControlFrame->SetFormProperty(nsGkAtoms::select, u""_ns);
+ }
+
+ return NS_OK;
+}
+
+bool HTMLTextAreaElement::IsHTMLFocusable(bool aWithMouse, bool* aIsFocusable,
+ int32_t* aTabIndex) {
+ if (nsGenericHTMLFormControlElementWithState::IsHTMLFocusable(
+ aWithMouse, aIsFocusable, aTabIndex)) {
+ return true;
+ }
+
+ // disabled textareas are not focusable
+ *aIsFocusable = !IsDisabled();
+ return false;
+}
+
+int32_t HTMLTextAreaElement::TabIndexDefault() { return 0; }
+
+void HTMLTextAreaElement::GetType(nsAString& aType) {
+ aType.AssignLiteral("textarea");
+}
+
+void HTMLTextAreaElement::GetValue(nsAString& aValue) {
+ GetValueInternal(aValue, true);
+ MOZ_ASSERT(aValue.FindChar(static_cast<char16_t>('\r')) == -1);
+}
+
+void HTMLTextAreaElement::GetValueInternal(nsAString& aValue,
+ bool aIgnoreWrap) const {
+ MOZ_ASSERT(mState);
+ mState->GetValue(aValue, aIgnoreWrap);
+}
+
+bool HTMLTextAreaElement::ValueEquals(const nsAString& aValue) const {
+ MOZ_ASSERT(mState);
+ return mState->ValueEquals(aValue);
+}
+
+nsIEditor* HTMLTextAreaElement::GetEditorForBindings() {
+ if (!GetPrimaryFrame()) {
+ GetPrimaryFrame(FlushType::Frames);
+ }
+ return GetTextEditor();
+}
+
+TextEditor* HTMLTextAreaElement::GetTextEditor() {
+ MOZ_ASSERT(mState);
+ return mState->GetTextEditor();
+}
+
+TextEditor* HTMLTextAreaElement::GetTextEditorWithoutCreation() {
+ MOZ_ASSERT(mState);
+ return mState->GetTextEditorWithoutCreation();
+}
+
+nsISelectionController* HTMLTextAreaElement::GetSelectionController() {
+ MOZ_ASSERT(mState);
+ return mState->GetSelectionController();
+}
+
+nsFrameSelection* HTMLTextAreaElement::GetConstFrameSelection() {
+ MOZ_ASSERT(mState);
+ return mState->GetConstFrameSelection();
+}
+
+nsresult HTMLTextAreaElement::BindToFrame(nsTextControlFrame* aFrame) {
+ MOZ_ASSERT(!nsContentUtils::IsSafeToRunScript());
+ MOZ_ASSERT(mState);
+ return mState->BindToFrame(aFrame);
+}
+
+void HTMLTextAreaElement::UnbindFromFrame(nsTextControlFrame* aFrame) {
+ MOZ_ASSERT(mState);
+ if (aFrame) {
+ mState->UnbindFromFrame(aFrame);
+ }
+}
+
+nsresult HTMLTextAreaElement::CreateEditor() {
+ MOZ_ASSERT(mState);
+ return mState->PrepareEditor();
+}
+
+void HTMLTextAreaElement::SetPreviewValue(const nsAString& aValue) {
+ MOZ_ASSERT(mState);
+ mState->SetPreviewText(aValue, true);
+}
+
+void HTMLTextAreaElement::GetPreviewValue(nsAString& aValue) {
+ MOZ_ASSERT(mState);
+ mState->GetPreviewText(aValue);
+}
+
+void HTMLTextAreaElement::EnablePreview() {
+ if (mIsPreviewEnabled) {
+ return;
+ }
+
+ mIsPreviewEnabled = true;
+ // Reconstruct the frame to append an anonymous preview node
+ nsLayoutUtils::PostRestyleEvent(this, RestyleHint{0},
+ nsChangeHint_ReconstructFrame);
+}
+
+bool HTMLTextAreaElement::IsPreviewEnabled() { return mIsPreviewEnabled; }
+
+nsresult HTMLTextAreaElement::SetValueInternal(
+ const nsAString& aValue, const ValueSetterOptions& aOptions) {
+ MOZ_ASSERT(mState);
+
+ // Need to set the value changed flag here if our value has in fact changed
+ // (i.e. if ValueSetterOption::SetValueChanged is in aOptions), so that
+ // retrieves the correct value if needed.
+ if (aOptions.contains(ValueSetterOption::SetValueChanged)) {
+ SetValueChanged(true);
+ }
+
+ if (!mState->SetValue(aValue, aOptions)) {
+ return NS_ERROR_OUT_OF_MEMORY;
+ }
+
+ return NS_OK;
+}
+
+void HTMLTextAreaElement::SetValue(const nsAString& aValue,
+ ErrorResult& aError) {
+ // If the value has been set by a script, we basically want to keep the
+ // current change event state. If the element is ready to fire a change
+ // event, we should keep it that way. Otherwise, we should make sure the
+ // element will not fire any event because of the script interaction.
+ //
+ // NOTE: this is currently quite expensive work (too much string
+ // manipulation). We should probably optimize that.
+ nsAutoString currentValue;
+ GetValueInternal(currentValue, true);
+
+ nsresult rv = SetValueInternal(
+ aValue,
+ {ValueSetterOption::ByContentAPI, ValueSetterOption::SetValueChanged,
+ ValueSetterOption::MoveCursorToEndIfValueChanged});
+ if (NS_WARN_IF(NS_FAILED(rv))) {
+ aError.Throw(rv);
+ return;
+ }
+
+ if (mFocusedValue.Equals(currentValue)) {
+ GetValueInternal(mFocusedValue, true);
+ }
+}
+
+void HTMLTextAreaElement::SetUserInput(const nsAString& aValue,
+ nsIPrincipal& aSubjectPrincipal) {
+ SetValueInternal(aValue, {ValueSetterOption::BySetUserInputAPI,
+ ValueSetterOption::SetValueChanged,
+ ValueSetterOption::MoveCursorToEndIfValueChanged});
+}
+
+void HTMLTextAreaElement::SetValueChanged(bool aValueChanged) {
+ MOZ_ASSERT(mState);
+
+ bool previousValue = mValueChanged;
+ mValueChanged = aValueChanged;
+ if (!aValueChanged && !mState->IsEmpty()) {
+ mState->EmptyValue();
+ }
+ if (mValueChanged == previousValue) {
+ return;
+ }
+ UpdateTooLongValidityState();
+ UpdateTooShortValidityState();
+ // We need to do this unconditionally because the validity ui bits depend on
+ // this.
+ UpdateState(true);
+}
+
+void HTMLTextAreaElement::SetLastValueChangeWasInteractive(
+ bool aWasInteractive) {
+ if (aWasInteractive == mLastValueChangeWasInteractive) {
+ return;
+ }
+ mLastValueChangeWasInteractive = aWasInteractive;
+ const bool wasValid = IsValid();
+ UpdateTooLongValidityState();
+ UpdateTooShortValidityState();
+ if (wasValid != IsValid()) {
+ UpdateState(true);
+ }
+}
+
+void HTMLTextAreaElement::GetDefaultValue(nsAString& aDefaultValue,
+ ErrorResult& aError) const {
+ if (!nsContentUtils::GetNodeTextContent(this, false, aDefaultValue,
+ fallible)) {
+ aError.Throw(NS_ERROR_OUT_OF_MEMORY);
+ }
+}
+
+void HTMLTextAreaElement::SetDefaultValue(const nsAString& aDefaultValue,
+ ErrorResult& aError) {
+ // setting the value of an textarea element using `.defaultValue = "foo"`
+ // must be interpreted as a two-step operation:
+ // 1. clearing all child nodes
+ // 2. adding a new text node with the new content
+ // Step 1 must therefore collapse the Selection to 0.
+ // Calling `SetNodeTextContent()` with an empty string will do that for us.
+ nsContentUtils::SetNodeTextContent(this, EmptyString(), true);
+ nsresult rv = nsContentUtils::SetNodeTextContent(this, aDefaultValue, true);
+ if (NS_SUCCEEDED(rv) && !mValueChanged) {
+ Reset();
+ }
+ if (NS_FAILED(rv)) {
+ aError.Throw(rv);
+ }
+}
+
+bool HTMLTextAreaElement::ParseAttribute(int32_t aNamespaceID,
+ nsAtom* aAttribute,
+ const nsAString& aValue,
+ nsIPrincipal* aMaybeScriptedPrincipal,
+ nsAttrValue& aResult) {
+ if (aNamespaceID == kNameSpaceID_None) {
+ if (aAttribute == nsGkAtoms::maxlength ||
+ aAttribute == nsGkAtoms::minlength) {
+ return aResult.ParseNonNegativeIntValue(aValue);
+ } else if (aAttribute == nsGkAtoms::cols) {
+ aResult.ParseIntWithFallback(aValue, DEFAULT_COLS);
+ return true;
+ } else if (aAttribute == nsGkAtoms::rows) {
+ aResult.ParseIntWithFallback(aValue, DEFAULT_ROWS_TEXTAREA);
+ return true;
+ } else if (aAttribute == nsGkAtoms::autocomplete) {
+ aResult.ParseAtomArray(aValue);
+ return true;
+ }
+ }
+ return TextControlElement::ParseAttribute(aNamespaceID, aAttribute, aValue,
+ aMaybeScriptedPrincipal, aResult);
+}
+
+void HTMLTextAreaElement::MapAttributesIntoRule(
+ const nsMappedAttributes* aAttributes, MappedDeclarations& aDecls) {
+ // wrap=off
+ if (!aDecls.PropertyIsSet(eCSSProperty_white_space)) {
+ const nsAttrValue* value = aAttributes->GetAttr(nsGkAtoms::wrap);
+ if (value && value->Type() == nsAttrValue::eString &&
+ value->Equals(nsGkAtoms::OFF, eIgnoreCase)) {
+ aDecls.SetKeywordValue(eCSSProperty_white_space, StyleWhiteSpace::Pre);
+ }
+ }
+
+ nsGenericHTMLFormControlElementWithState::MapDivAlignAttributeInto(
+ aAttributes, aDecls);
+ nsGenericHTMLFormControlElementWithState::MapCommonAttributesInto(aAttributes,
+ aDecls);
+}
+
+nsChangeHint HTMLTextAreaElement::GetAttributeChangeHint(
+ const nsAtom* aAttribute, int32_t aModType) const {
+ nsChangeHint retval =
+ nsGenericHTMLFormControlElementWithState::GetAttributeChangeHint(
+ aAttribute, aModType);
+
+ const bool isAdditionOrRemoval =
+ aModType == MutationEvent_Binding::ADDITION ||
+ aModType == MutationEvent_Binding::REMOVAL;
+
+ if (aAttribute == nsGkAtoms::rows || aAttribute == nsGkAtoms::cols) {
+ retval |= NS_STYLE_HINT_REFLOW;
+ } else if (aAttribute == nsGkAtoms::wrap) {
+ retval |= nsChangeHint_ReconstructFrame;
+ } else if (aAttribute == nsGkAtoms::placeholder && isAdditionOrRemoval) {
+ retval |= nsChangeHint_ReconstructFrame;
+ }
+ return retval;
+}
+
+NS_IMETHODIMP_(bool)
+HTMLTextAreaElement::IsAttributeMapped(const nsAtom* aAttribute) const {
+ static const MappedAttributeEntry attributes[] = {{nsGkAtoms::wrap},
+ {nullptr}};
+
+ static const MappedAttributeEntry* const map[] = {
+ attributes,
+ sDivAlignAttributeMap,
+ sCommonAttributeMap,
+ };
+
+ return FindAttributeDependence(aAttribute, map);
+}
+
+nsMapRuleToAttributesFunc HTMLTextAreaElement::GetAttributeMappingFunction()
+ const {
+ return &MapAttributesIntoRule;
+}
+
+bool HTMLTextAreaElement::IsDisabledForEvents(WidgetEvent* aEvent) {
+ nsIFormControlFrame* formControlFrame = GetFormControlFrame(false);
+ nsIFrame* formFrame = do_QueryFrame(formControlFrame);
+ return IsElementDisabledForEvents(aEvent, formFrame);
+}
+
+void HTMLTextAreaElement::GetEventTargetParent(EventChainPreVisitor& aVisitor) {
+ aVisitor.mCanHandle = false;
+ if (IsDisabledForEvents(aVisitor.mEvent)) {
+ return;
+ }
+
+ // Don't dispatch a second select event if we are already handling
+ // one.
+ if (aVisitor.mEvent->mMessage == eFormSelect) {
+ if (mHandlingSelect) {
+ return;
+ }
+ mHandlingSelect = true;
+ }
+
+ if (aVisitor.mEvent->mMessage == eBlur) {
+ // Set mWantsPreHandleEvent and fire change event in PreHandleEvent to
+ // prevent it breaks event target chain creation.
+ aVisitor.mWantsPreHandleEvent = true;
+ }
+
+ nsGenericHTMLFormControlElementWithState::GetEventTargetParent(aVisitor);
+}
+
+nsresult HTMLTextAreaElement::PreHandleEvent(EventChainVisitor& aVisitor) {
+ if (aVisitor.mEvent->mMessage == eBlur) {
+ // Fire onchange (if necessary), before we do the blur, bug 370521.
+ FireChangeEventIfNeeded();
+ }
+ return nsGenericHTMLFormControlElementWithState::PreHandleEvent(aVisitor);
+}
+
+void HTMLTextAreaElement::FireChangeEventIfNeeded() {
+ nsString value;
+ GetValueInternal(value, true);
+
+ if (mFocusedValue.Equals(value)) {
+ return;
+ }
+
+ // Dispatch the change event.
+ mFocusedValue = value;
+ nsContentUtils::DispatchTrustedEvent(
+ OwnerDoc(), static_cast<nsIContent*>(this), u"change"_ns, CanBubble::eYes,
+ Cancelable::eNo);
+}
+
+nsresult HTMLTextAreaElement::PostHandleEvent(EventChainPostVisitor& aVisitor) {
+ if (aVisitor.mEvent->mMessage == eFormSelect) {
+ mHandlingSelect = false;
+ }
+
+ if (aVisitor.mEvent->mMessage == eFocus ||
+ aVisitor.mEvent->mMessage == eBlur) {
+ if (aVisitor.mEvent->mMessage == eFocus) {
+ // If the invalid UI is shown, we should show it while focusing (and
+ // update). Otherwise, we should not.
+ GetValueInternal(mFocusedValue, true);
+ mCanShowInvalidUI = !IsValid() && ShouldShowValidityUI();
+
+ // If neither invalid UI nor valid UI is shown, we shouldn't show the
+ // valid UI while typing.
+ mCanShowValidUI = ShouldShowValidityUI();
+ } else { // eBlur
+ mCanShowInvalidUI = true;
+ mCanShowValidUI = true;
+ }
+
+ UpdateState(true);
+ }
+
+ return NS_OK;
+}
+
+void HTMLTextAreaElement::DoneAddingChildren(bool aHaveNotified) {
+ if (!mValueChanged) {
+ if (!mDoneAddingChildren) {
+ // Reset now that we're done adding children if the content sink tried to
+ // sneak some text in without calling AppendChildTo.
+ Reset();
+ }
+
+ if (!mInhibitStateRestoration) {
+ GenerateStateKey();
+ RestoreFormControlState();
+ }
+ }
+
+ mDoneAddingChildren = true;
+}
+
+bool HTMLTextAreaElement::IsDoneAddingChildren() { return mDoneAddingChildren; }
+
+// Controllers Methods
+
+nsIControllers* HTMLTextAreaElement::GetControllers(ErrorResult& aError) {
+ if (!mControllers) {
+ mControllers = new nsXULControllers();
+ if (!mControllers) {
+ aError.Throw(NS_ERROR_FAILURE);
+ return nullptr;
+ }
+
+ RefPtr<nsBaseCommandController> commandController =
+ nsBaseCommandController::CreateEditorController();
+ if (!commandController) {
+ aError.Throw(NS_ERROR_FAILURE);
+ return nullptr;
+ }
+
+ mControllers->AppendController(commandController);
+
+ commandController = nsBaseCommandController::CreateEditingController();
+ if (!commandController) {
+ aError.Throw(NS_ERROR_FAILURE);
+ return nullptr;
+ }
+
+ mControllers->AppendController(commandController);
+ }
+
+ return mControllers;
+}
+
+nsresult HTMLTextAreaElement::GetControllers(nsIControllers** aResult) {
+ NS_ENSURE_ARG_POINTER(aResult);
+
+ ErrorResult error;
+ *aResult = GetControllers(error);
+ NS_IF_ADDREF(*aResult);
+
+ return error.StealNSResult();
+}
+
+uint32_t HTMLTextAreaElement::GetTextLength() {
+ nsAutoString val;
+ GetValue(val);
+ return val.Length();
+}
+
+Nullable<uint32_t> HTMLTextAreaElement::GetSelectionStart(ErrorResult& aError) {
+ uint32_t selStart, selEnd;
+ GetSelectionRange(&selStart, &selEnd, aError);
+ return Nullable<uint32_t>(selStart);
+}
+
+void HTMLTextAreaElement::SetSelectionStart(
+ const Nullable<uint32_t>& aSelectionStart, ErrorResult& aError) {
+ MOZ_ASSERT(mState);
+ mState->SetSelectionStart(aSelectionStart, aError);
+}
+
+Nullable<uint32_t> HTMLTextAreaElement::GetSelectionEnd(ErrorResult& aError) {
+ uint32_t selStart, selEnd;
+ GetSelectionRange(&selStart, &selEnd, aError);
+ return Nullable<uint32_t>(selEnd);
+}
+
+void HTMLTextAreaElement::SetSelectionEnd(
+ const Nullable<uint32_t>& aSelectionEnd, ErrorResult& aError) {
+ MOZ_ASSERT(mState);
+ mState->SetSelectionEnd(aSelectionEnd, aError);
+}
+
+void HTMLTextAreaElement::GetSelectionRange(uint32_t* aSelectionStart,
+ uint32_t* aSelectionEnd,
+ ErrorResult& aRv) {
+ MOZ_ASSERT(mState);
+ return mState->GetSelectionRange(aSelectionStart, aSelectionEnd, aRv);
+}
+
+void HTMLTextAreaElement::GetSelectionDirection(nsAString& aDirection,
+ ErrorResult& aError) {
+ MOZ_ASSERT(mState);
+ mState->GetSelectionDirectionString(aDirection, aError);
+}
+
+void HTMLTextAreaElement::SetSelectionDirection(const nsAString& aDirection,
+ ErrorResult& aError) {
+ MOZ_ASSERT(mState);
+ mState->SetSelectionDirection(aDirection, aError);
+}
+
+void HTMLTextAreaElement::SetSelectionRange(
+ uint32_t aSelectionStart, uint32_t aSelectionEnd,
+ const Optional<nsAString>& aDirection, ErrorResult& aError) {
+ MOZ_ASSERT(mState);
+ mState->SetSelectionRange(aSelectionStart, aSelectionEnd, aDirection, aError);
+}
+
+void HTMLTextAreaElement::SetRangeText(const nsAString& aReplacement,
+ ErrorResult& aRv) {
+ MOZ_ASSERT(mState);
+ mState->SetRangeText(aReplacement, aRv);
+}
+
+void HTMLTextAreaElement::SetRangeText(const nsAString& aReplacement,
+ uint32_t aStart, uint32_t aEnd,
+ SelectionMode aSelectMode,
+ ErrorResult& aRv) {
+ MOZ_ASSERT(mState);
+ mState->SetRangeText(aReplacement, aStart, aEnd, aSelectMode, aRv);
+}
+
+void HTMLTextAreaElement::GetValueFromSetRangeText(nsAString& aValue) {
+ GetValueInternal(aValue, false);
+}
+
+nsresult HTMLTextAreaElement::SetValueFromSetRangeText(
+ const nsAString& aValue) {
+ return SetValueInternal(aValue, {ValueSetterOption::ByContentAPI,
+ ValueSetterOption::BySetRangeTextAPI,
+ ValueSetterOption::SetValueChanged});
+}
+
+void HTMLTextAreaElement::SetDirectionFromValue(bool aNotify,
+ const nsAString* aKnownValue) {
+ nsAutoString value;
+ if (!aKnownValue) {
+ GetValue(value);
+ aKnownValue = &value;
+ }
+ SetDirectionalityFromValue(this, *aKnownValue, aNotify);
+}
+
+nsresult HTMLTextAreaElement::Reset() {
+ nsAutoString resetVal;
+ GetDefaultValue(resetVal, IgnoreErrors());
+ SetValueChanged(false);
+
+ nsresult rv = SetValueInternal(resetVal, ValueSetterOption::ByInternalAPI);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+HTMLTextAreaElement::SubmitNamesValues(FormData* aFormData) {
+ //
+ // Get the name (if no name, no submit)
+ //
+ nsAutoString name;
+ GetAttr(kNameSpaceID_None, nsGkAtoms::name, name);
+ if (name.IsEmpty()) {
+ return NS_OK;
+ }
+
+ //
+ // Get the value
+ //
+ nsAutoString value;
+ GetValueInternal(value, false);
+
+ //
+ // Submit
+ //
+ return aFormData->AddNameValuePair(name, value);
+}
+
+void HTMLTextAreaElement::SaveState() {
+ // Only save if value != defaultValue (bug 62713)
+ PresState* state = nullptr;
+ if (mValueChanged) {
+ state = GetPrimaryPresState();
+ if (state) {
+ nsAutoString value;
+ GetValueInternal(value, true);
+
+ if (NS_FAILED(nsLinebreakConverter::ConvertStringLineBreaks(
+ value, nsLinebreakConverter::eLinebreakPlatform,
+ nsLinebreakConverter::eLinebreakContent))) {
+ NS_ERROR("Converting linebreaks failed!");
+ return;
+ }
+
+ state->contentData() =
+ TextContentData(value, mLastValueChangeWasInteractive);
+ }
+ }
+
+ if (mDisabledChanged) {
+ if (!state) {
+ state = GetPrimaryPresState();
+ }
+ if (state) {
+ // We do not want to save the real disabled state but the disabled
+ // attribute.
+ state->disabled() = HasAttr(kNameSpaceID_None, nsGkAtoms::disabled);
+ state->disabledSet() = true;
+ }
+ }
+}
+
+bool HTMLTextAreaElement::RestoreState(PresState* aState) {
+ const PresContentData& state = aState->contentData();
+
+ if (state.type() == PresContentData::TTextContentData) {
+ ErrorResult rv;
+ SetValue(state.get_TextContentData().value(), rv);
+ ENSURE_SUCCESS(rv, false);
+ if (state.get_TextContentData().lastValueChangeWasInteractive()) {
+ SetLastValueChangeWasInteractive(true);
+ }
+ }
+ if (aState->disabledSet() && !aState->disabled()) {
+ SetDisabled(false, IgnoreErrors());
+ }
+
+ return false;
+}
+
+ElementState HTMLTextAreaElement::IntrinsicState() const {
+ ElementState state =
+ nsGenericHTMLFormControlElementWithState::IntrinsicState();
+
+ if (IsCandidateForConstraintValidation()) {
+ if (IsValid()) {
+ state |= ElementState::VALID;
+ } else {
+ state |= ElementState::INVALID;
+ // :-moz-ui-invalid always apply if the element suffers from a custom
+ // error.
+ if (GetValidityState(VALIDITY_STATE_CUSTOM_ERROR) ||
+ (mCanShowInvalidUI && ShouldShowValidityUI())) {
+ state |= ElementState::USER_INVALID;
+ }
+ }
+
+ // :-moz-ui-valid applies if all the following are true:
+ // 1. The element is not focused, or had either :-moz-ui-valid or
+ // :-moz-ui-invalid applying before it was focused ;
+ // 2. The element is either valid or isn't allowed to have
+ // :-moz-ui-invalid applying ;
+ // 3. The element has already been modified or the user tried to submit the
+ // form owner while invalid.
+ if (mCanShowValidUI && ShouldShowValidityUI() &&
+ (IsValid() ||
+ (state.HasState(ElementState::USER_INVALID) && !mCanShowInvalidUI))) {
+ state |= ElementState::USER_VALID;
+ }
+ }
+
+ if (IsValueEmpty() && HasAttr(nsGkAtoms::placeholder)) {
+ state |= ElementState::PLACEHOLDER_SHOWN;
+ }
+
+ return state;
+}
+
+nsresult HTMLTextAreaElement::BindToTree(BindContext& aContext,
+ nsINode& aParent) {
+ nsresult rv =
+ nsGenericHTMLFormControlElementWithState::BindToTree(aContext, aParent);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ // Set direction based on value if dir=auto
+ if (HasDirAuto()) {
+ SetDirectionFromValue(false);
+ }
+
+ // If there is a disabled fieldset in the parent chain, the element is now
+ // barred from constraint validation and can't suffer from value missing.
+ UpdateValueMissingValidityState();
+ UpdateBarredFromConstraintValidation();
+
+ // And now make sure our state is up to date
+ UpdateState(false);
+
+ return rv;
+}
+
+void HTMLTextAreaElement::UnbindFromTree(bool aNullParent) {
+ nsGenericHTMLFormControlElementWithState::UnbindFromTree(aNullParent);
+
+ // We might be no longer disabled because of parent chain changed.
+ UpdateValueMissingValidityState();
+ UpdateBarredFromConstraintValidation();
+
+ // And now make sure our state is up to date
+ UpdateState(false);
+}
+
+void HTMLTextAreaElement::BeforeSetAttr(int32_t aNameSpaceID, nsAtom* aName,
+ const nsAttrValue* aValue,
+ bool aNotify) {
+ if (aNotify && aName == nsGkAtoms::disabled &&
+ aNameSpaceID == kNameSpaceID_None) {
+ mDisabledChanged = true;
+ }
+
+ return nsGenericHTMLFormControlElementWithState::BeforeSetAttr(
+ aNameSpaceID, aName, aValue, aNotify);
+}
+
+void HTMLTextAreaElement::CharacterDataChanged(nsIContent* aContent,
+ const CharacterDataChangeInfo&) {
+ ContentChanged(aContent);
+}
+
+void HTMLTextAreaElement::ContentAppended(nsIContent* aFirstNewContent) {
+ ContentChanged(aFirstNewContent);
+}
+
+void HTMLTextAreaElement::ContentInserted(nsIContent* aChild) {
+ ContentChanged(aChild);
+}
+
+void HTMLTextAreaElement::ContentRemoved(nsIContent* aChild,
+ nsIContent* aPreviousSibling) {
+ ContentChanged(aChild);
+}
+
+void HTMLTextAreaElement::ContentChanged(nsIContent* aContent) {
+ if (!mValueChanged && mDoneAddingChildren &&
+ nsContentUtils::IsInSameAnonymousTree(this, aContent)) {
+ if (mState->IsSelectionCached()) {
+ // In case the content is *replaced*, i.e. by calling
+ // `.textContent = "foo";`,
+ // firstly the old content is removed, then the new content is added.
+ // As per wpt, this must collapse the selection to 0.
+ // Removing and adding of an element is routed through here, but due to
+ // the script runner `Reset()` is only invoked after the append operation.
+ // Therefore, `Reset()` would adjust the Selection to the new value, not
+ // to 0.
+ // By forcing a selection update here, the selection is reset in order to
+ // comply with the wpt.
+ auto& props = mState->GetSelectionProperties();
+ nsAutoString resetVal;
+ GetDefaultValue(resetVal, IgnoreErrors());
+ props.SetMaxLength(resetVal.Length());
+ props.SetStart(props.GetStart());
+ props.SetEnd(props.GetEnd());
+ }
+ // We should wait all ranges finish handling the mutation before updating
+ // the anonymous subtree with a call of Reset.
+ nsContentUtils::AddScriptRunner(NS_NewRunnableFunction(
+ "ResetHTMLTextAreaElementIfValueHasNotChangedYet",
+ [self = RefPtr{this}]() {
+ // However, if somebody has already changed the value, we don't need
+ // to keep doing this.
+ if (!self->mValueChanged) {
+ self->Reset();
+ }
+ }));
+ }
+}
+
+void HTMLTextAreaElement::AfterSetAttr(int32_t aNameSpaceID, nsAtom* aName,
+ const nsAttrValue* aValue,
+ const nsAttrValue* aOldValue,
+ nsIPrincipal* aSubjectPrincipal,
+ bool aNotify) {
+ if (aNameSpaceID == kNameSpaceID_None) {
+ if (aName == nsGkAtoms::required || aName == nsGkAtoms::disabled ||
+ aName == nsGkAtoms::readonly) {
+ if (aName == nsGkAtoms::disabled) {
+ // This *has* to be called *before* validity state check because
+ // UpdateBarredFromConstraintValidation and
+ // UpdateValueMissingValidityState depend on our disabled state.
+ UpdateDisabledState(aNotify);
+ }
+
+ if (aName == nsGkAtoms::required) {
+ // This *has* to be called *before* UpdateValueMissingValidityState
+ // because UpdateValueMissingValidityState depends on our required
+ // state.
+ UpdateRequiredState(!!aValue, aNotify);
+ }
+
+ UpdateValueMissingValidityState();
+
+ // This *has* to be called *after* validity has changed.
+ if (aName == nsGkAtoms::readonly || aName == nsGkAtoms::disabled) {
+ UpdateBarredFromConstraintValidation();
+ }
+ } else if (aName == nsGkAtoms::autocomplete) {
+ // Clear the cached @autocomplete attribute state.
+ mAutocompleteAttrState = nsContentUtils::eAutocompleteAttrState_Unknown;
+ } else if (aName == nsGkAtoms::maxlength) {
+ UpdateTooLongValidityState();
+ } else if (aName == nsGkAtoms::minlength) {
+ UpdateTooShortValidityState();
+ } else if (aName == nsGkAtoms::placeholder) {
+ if (nsTextControlFrame* f = do_QueryFrame(GetPrimaryFrame())) {
+ f->PlaceholderChanged(aOldValue, aValue);
+ }
+ } else if (aName == nsGkAtoms::dir && aValue &&
+ aValue->Equals(nsGkAtoms::_auto, eIgnoreCase)) {
+ SetDirectionFromValue(aNotify);
+ }
+ }
+
+ return nsGenericHTMLFormControlElementWithState::AfterSetAttr(
+ aNameSpaceID, aName, aValue, aOldValue, aSubjectPrincipal, aNotify);
+}
+
+nsresult HTMLTextAreaElement::CopyInnerTo(Element* aDest) {
+ nsresult rv = nsGenericHTMLFormControlElementWithState::CopyInnerTo(aDest);
+ NS_ENSURE_SUCCESS(rv, rv);
+
+ if (aDest->OwnerDoc()->IsStaticDocument()) {
+ nsAutoString value;
+ GetValueInternal(value, true);
+ ErrorResult ret;
+ static_cast<HTMLTextAreaElement*>(aDest)->SetValue(value, ret);
+ return ret.StealNSResult();
+ }
+ return NS_OK;
+}
+
+bool HTMLTextAreaElement::IsMutable() const {
+ return (!HasAttr(kNameSpaceID_None, nsGkAtoms::readonly) && !IsDisabled());
+}
+
+void HTMLTextAreaElement::SetCustomValidity(const nsAString& aError) {
+ ConstraintValidation::SetCustomValidity(aError);
+
+ UpdateState(true);
+}
+
+bool HTMLTextAreaElement::IsTooLong() {
+ if (!mValueChanged || !mLastValueChangeWasInteractive ||
+ !HasAttr(nsGkAtoms::maxlength)) {
+ return false;
+ }
+
+ int32_t maxLength = MaxLength();
+
+ // Maxlength of -1 means parsing error.
+ if (maxLength == -1) {
+ return false;
+ }
+
+ int32_t textLength = GetTextLength();
+
+ return textLength > maxLength;
+}
+
+bool HTMLTextAreaElement::IsTooShort() {
+ if (!mValueChanged || !mLastValueChangeWasInteractive ||
+ !HasAttr(nsGkAtoms::minlength)) {
+ return false;
+ }
+
+ int32_t minLength = MinLength();
+
+ // Minlength of -1 means parsing error.
+ if (minLength == -1) {
+ return false;
+ }
+
+ int32_t textLength = GetTextLength();
+
+ return textLength && textLength < minLength;
+}
+
+bool HTMLTextAreaElement::IsValueMissing() const {
+ if (!Required() || !IsMutable()) {
+ return false;
+ }
+ return IsValueEmpty();
+}
+
+void HTMLTextAreaElement::UpdateTooLongValidityState() {
+ SetValidityState(VALIDITY_STATE_TOO_LONG, IsTooLong());
+}
+
+void HTMLTextAreaElement::UpdateTooShortValidityState() {
+ SetValidityState(VALIDITY_STATE_TOO_SHORT, IsTooShort());
+}
+
+void HTMLTextAreaElement::UpdateValueMissingValidityState() {
+ SetValidityState(VALIDITY_STATE_VALUE_MISSING, IsValueMissing());
+}
+
+void HTMLTextAreaElement::UpdateBarredFromConstraintValidation() {
+ SetBarredFromConstraintValidation(
+ HasAttr(kNameSpaceID_None, nsGkAtoms::readonly) ||
+ HasFlag(ELEMENT_IS_DATALIST_OR_HAS_DATALIST_ANCESTOR) || IsDisabled());
+}
+
+nsresult HTMLTextAreaElement::GetValidationMessage(
+ nsAString& aValidationMessage, ValidityStateType aType) {
+ nsresult rv = NS_OK;
+
+ switch (aType) {
+ case VALIDITY_STATE_TOO_LONG: {
+ nsAutoString message;
+ int32_t maxLength = MaxLength();
+ int32_t textLength = GetTextLength();
+ nsAutoString strMaxLength;
+ nsAutoString strTextLength;
+
+ strMaxLength.AppendInt(maxLength);
+ strTextLength.AppendInt(textLength);
+
+ rv = nsContentUtils::FormatMaybeLocalizedString(
+ message, nsContentUtils::eDOM_PROPERTIES, "FormValidationTextTooLong",
+ OwnerDoc(), strMaxLength, strTextLength);
+ aValidationMessage = message;
+ } break;
+ case VALIDITY_STATE_TOO_SHORT: {
+ nsAutoString message;
+ int32_t minLength = MinLength();
+ int32_t textLength = GetTextLength();
+ nsAutoString strMinLength;
+ nsAutoString strTextLength;
+
+ strMinLength.AppendInt(minLength);
+ strTextLength.AppendInt(textLength);
+
+ rv = nsContentUtils::FormatMaybeLocalizedString(
+ message, nsContentUtils::eDOM_PROPERTIES,
+ "FormValidationTextTooShort", OwnerDoc(), strMinLength,
+ strTextLength);
+ aValidationMessage = message;
+ } break;
+ case VALIDITY_STATE_VALUE_MISSING: {
+ nsAutoString message;
+ rv = nsContentUtils::GetMaybeLocalizedString(
+ nsContentUtils::eDOM_PROPERTIES, "FormValidationValueMissing",
+ OwnerDoc(), message);
+ aValidationMessage = message;
+ } break;
+ default:
+ rv =
+ ConstraintValidation::GetValidationMessage(aValidationMessage, aType);
+ }
+
+ return rv;
+}
+
+bool HTMLTextAreaElement::IsSingleLineTextControl() const { return false; }
+
+bool HTMLTextAreaElement::IsTextArea() const { return true; }
+
+bool HTMLTextAreaElement::IsPasswordTextControl() const { return false; }
+
+int32_t HTMLTextAreaElement::GetCols() { return Cols(); }
+
+int32_t HTMLTextAreaElement::GetWrapCols() {
+ nsHTMLTextWrap wrapProp;
+ TextControlElement::GetWrapPropertyEnum(this, wrapProp);
+ if (wrapProp == TextControlElement::eHTMLTextWrap_Off) {
+ // do not wrap when wrap=off
+ return 0;
+ }
+
+ // Otherwise we just wrap at the given number of columns
+ return GetCols();
+}
+
+int32_t HTMLTextAreaElement::GetRows() {
+ const nsAttrValue* attr = GetParsedAttr(nsGkAtoms::rows);
+ if (attr && attr->Type() == nsAttrValue::eInteger) {
+ int32_t rows = attr->GetIntegerValue();
+ return (rows <= 0) ? DEFAULT_ROWS_TEXTAREA : rows;
+ }
+
+ return DEFAULT_ROWS_TEXTAREA;
+}
+
+void HTMLTextAreaElement::GetDefaultValueFromContent(nsAString& aValue) {
+ GetDefaultValue(aValue, IgnoreErrors());
+}
+
+bool HTMLTextAreaElement::ValueChanged() const { return mValueChanged; }
+
+void HTMLTextAreaElement::GetTextEditorValue(nsAString& aValue,
+ bool aIgnoreWrap) const {
+ MOZ_ASSERT(mState);
+ mState->GetValue(aValue, aIgnoreWrap);
+}
+
+void HTMLTextAreaElement::InitializeKeyboardEventListeners() {
+ MOZ_ASSERT(mState);
+ mState->InitializeKeyboardEventListeners();
+}
+
+void HTMLTextAreaElement::OnValueChanged(ValueChangeKind aKind,
+ bool aNewValueEmpty,
+ const nsAString* aKnownNewValue) {
+ if (aKind != ValueChangeKind::Internal) {
+ mLastValueChangeWasInteractive = aKind == ValueChangeKind::UserInteraction;
+ }
+
+ const bool emptyBefore = IsValueEmpty();
+ if (aNewValueEmpty) {
+ AddStates(ElementState::VALUE_EMPTY);
+ } else {
+ RemoveStates(ElementState::VALUE_EMPTY);
+ }
+
+ // Update the validity state
+ const bool validBefore = IsValid();
+ UpdateTooLongValidityState();
+ UpdateTooShortValidityState();
+ UpdateValueMissingValidityState();
+
+ if (HasDirAuto()) {
+ SetDirectionFromValue(true, aKnownNewValue);
+ }
+
+ if (validBefore != IsValid() ||
+ (emptyBefore != IsValueEmpty() && HasAttr(nsGkAtoms::placeholder))) {
+ UpdateState(true);
+ }
+}
+
+bool HTMLTextAreaElement::HasCachedSelection() {
+ MOZ_ASSERT(mState);
+ return mState->IsSelectionCached();
+}
+
+void HTMLTextAreaElement::FieldSetDisabledChanged(bool aNotify) {
+ // This *has* to be called before UpdateBarredFromConstraintValidation and
+ // UpdateValueMissingValidityState because these two functions depend on our
+ // disabled state.
+ nsGenericHTMLFormControlElementWithState::FieldSetDisabledChanged(aNotify);
+
+ UpdateValueMissingValidityState();
+ UpdateBarredFromConstraintValidation();
+ UpdateState(aNotify);
+}
+
+JSObject* HTMLTextAreaElement::WrapNode(JSContext* aCx,
+ JS::Handle<JSObject*> aGivenProto) {
+ return HTMLTextAreaElement_Binding::Wrap(aCx, this, aGivenProto);
+}
+
+void HTMLTextAreaElement::GetAutocomplete(DOMString& aValue) {
+ const nsAttrValue* attributeVal = GetParsedAttr(nsGkAtoms::autocomplete);
+
+ mAutocompleteAttrState = nsContentUtils::SerializeAutocompleteAttribute(
+ attributeVal, aValue, mAutocompleteAttrState);
+}
+
+} // namespace mozilla::dom