Skip to content

Commit

Permalink
Add validation and error summary to comment form
Browse files Browse the repository at this point in the history
- Add check for empty comment on submit
- Add new error for when the comment is empty
- Add error type var to display the correct error
- Add tests

Ticket: https://phabricator.wikimedia.org/T367387
  • Loading branch information
Abban committed Sep 3, 2024
1 parent f9bca1d commit 0253598
Show file tree
Hide file tree
Showing 2 changed files with 86 additions and 7 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -16,14 +16,15 @@
<div v-else>
<p>{{ $t( 'donation_comment_popup_explanation' ) }}</p>

<ScrollTarget target-id="comment-scroll-target"/>
<TextField
input-type="textarea"
v-model="comment"
name="comment"
input-id="comment"
placeholder=""
:label="$t( 'donation_comment_popup_label' )"
:error-message="$t( 'donation_comment_popup_error' )"
:error-message="$t( commentError )"
:show-error="commentErrored"
:autofocus="true"
/>
Expand All @@ -45,6 +46,18 @@
{{ $t( 'donation_comment_popup_is_public' ) }}
</CheckboxField>

<ErrorSummary
:is-visible="commentErrored"
:items="[
{
validity: commentErrored ? Validity.INVALID : Validity.VALID,
message: $t( commentError ),
focusElement: 'comment',
scrollElement: 'comment-scroll-target'
},
]"
/>

<FormSummary :show-border="false">
<template #summary-buttons>
<FormButton
Expand All @@ -67,7 +80,7 @@
</template>

<script setup lang="ts">
import { computed, inject, onMounted, ref } from 'vue';
import { computed, inject, onMounted, ref, watch } from 'vue';
import { trackDynamicForm, trackFormSubmission } from '@src/util/tracking';
import { addressTypeFromName, AddressTypeModel } from '@src/view_models/AddressTypeModel';
import { Donation } from '@src/view_models/Donation';
Expand All @@ -76,6 +89,14 @@ import FormSummary from '@src/components/shared/FormSummary.vue';
import TextField from '@src/components/shared/form_fields/TextField.vue';
import CheckboxField from '@src/components/shared/form_fields/CheckboxField.vue';
import { CommentResource } from '@src/api/CommentResource';
import ErrorSummary from '@src/components/shared/validation_summary/ErrorSummary.vue';
import { Validity } from '@src/view_models/Validity';
import ScrollTarget from '@src/components/shared/ScrollTarget.vue';
enum CommentErrorTypes {
Empty,
Server
}
interface Props {
donation: Donation;
Expand All @@ -91,13 +112,21 @@ const comment = ref<string>( '' );
const commentIsPublic = ref<boolean>( false );
const commentHasPublicAuthorName = ref<boolean>( false );
const commentErrored = ref<boolean>( false );
const commentErrorType = ref<CommentErrorTypes>( CommentErrorTypes.Empty );
const commentHasBeenSubmitted = ref<boolean>( false );
const serverResponse = ref<string>( '' );
const showPublishAuthor = computed<boolean>( () => addressTypeFromName( props.addressType ) !== AddressTypeModel.ANON );
const postComment = async (): Promise<void> => {
trackFormSubmission( commentForm.value );
if ( comment.value === '' ) {
commentErrorType.value = CommentErrorTypes.Empty;
commentErrored.value = true;
return;
}
try {
const message = await commentResource.post( {
donationId: props.donation.id,
Expand All @@ -111,12 +140,26 @@ const postComment = async (): Promise<void> => {
serverResponse.value = message;
emit( 'disable-comment-link' );
} catch ( e ) {

Check failure on line 142 in src/components/pages/donation_confirmation/DonationCommentPopUp.vue

View workflow job for this annotation

GitHub Actions / build (18.x)

'e' is defined but never used
commentErrorType.value = CommentErrorTypes.Server;
commentErrored.value = true;
}
};
onMounted( trackDynamicForm );
const commentError = computed<string>( () => {
if ( commentErrorType.value === CommentErrorTypes.Empty ) {
return 'donation_comment_popup_empty_error';
}
return 'donation_comment_popup_error';
} );
watch( comment, ( newComment: string ) => {
if ( commentErrored.value && newComment !== '' ) {
commentErrored.value = false;
}
} );
</script>

<style lang="scss">
Expand Down
46 changes: 41 additions & 5 deletions tests/unit/components/DonationCommentPopUp.spec.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,7 @@
import { mount } from '@vue/test-utils';
import DonationCommentPopUp from '@src/components/pages/donation_confirmation/DonationCommentPopUp.vue';
import { AddressTypeModel, addressTypeName } from '@src/view_models/AddressTypeModel';
import {
FakeFailingCommentResource,
FakeSucceedingCommentResource,
successMessage
} from '@test/unit/TestDoubles/FakeCommentResource';
import { FakeFailingCommentResource, FakeSucceedingCommentResource, successMessage } from '@test/unit/TestDoubles/FakeCommentResource';

describe( 'DonationCommentPopUp.vue', () => {
function getDefaultConfirmationData( isAnonymous: boolean ): any {
Expand Down Expand Up @@ -62,6 +58,44 @@ describe( 'DonationCommentPopUp.vue', () => {
expect( wrapper.find( '#withName' ).exists() ).toBeFalsy();
} );

it( 'shows error when comment is empty', async () => {
const wrapper = mount( DonationCommentPopUp, {
props: getDefaultConfirmationData( true ),
global: {
provide: {
commentResource: new FakeSucceedingCommentResource(),
},
},
} );

await wrapper.trigger( 'submit' );

expect( wrapper.find( '#comment-error' ).exists() ).toBeTruthy();
expect( wrapper.find( '.error-summary' ).exists() ).toBeTruthy();
expect( wrapper.find( '#comment-error' ).text() ).toStrictEqual( 'donation_comment_popup_empty_error' );
} );

it( 'resets error when comment text is entered', async () => {
const wrapper = mount( DonationCommentPopUp, {
props: getDefaultConfirmationData( true ),
global: {
provide: {
commentResource: new FakeSucceedingCommentResource(),
},
},
} );

await wrapper.trigger( 'submit' );

expect( wrapper.find( '#comment-error' ).exists() ).toBeTruthy();
expect( wrapper.find( '.error-summary' ).exists() ).toBeTruthy();

await wrapper.find( '#comment' ).setValue( 'My super great comment' );

expect( wrapper.find( '#comment-error' ).exists() ).toBeFalsy();
expect( wrapper.find( '.error-summary' ).exists() ).toBeFalsy();
} );

it( 'shows error when API response is rejected', async () => {
const wrapper = mount( DonationCommentPopUp, {
props: getDefaultConfirmationData( true ),
Expand All @@ -72,9 +106,11 @@ describe( 'DonationCommentPopUp.vue', () => {
},
} );

await wrapper.find( '#comment' ).setValue( 'My super great comment' );
await wrapper.trigger( 'submit' );

expect( wrapper.find( '#comment-error' ).exists() ).toBeTruthy();
expect( wrapper.find( '.error-summary' ).exists() ).toBeTruthy();
expect( wrapper.find( '#comment-error' ).text() ).toStrictEqual( 'donation_comment_popup_error' );
} );

Expand Down

0 comments on commit 0253598

Please sign in to comment.