diff options
Diffstat (limited to 'dom/html/HTMLTextAreaElement.cpp')
-rw-r--r-- | dom/html/HTMLTextAreaElement.cpp | 1192 |
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 |