diff --git a/src/applications/travel-pay/components/submit-flow/pages/AddressPage.jsx b/src/applications/travel-pay/components/submit-flow/pages/AddressPage.jsx index 325678f4b3a6..fa304470b837 100644 --- a/src/applications/travel-pay/components/submit-flow/pages/AddressPage.jsx +++ b/src/applications/travel-pay/components/submit-flow/pages/AddressPage.jsx @@ -1,28 +1,175 @@ -import React from 'react'; +import React, { useState, useEffect } from 'react'; import PropTypes from 'prop-types'; -import { VaButtonPair } from '@department-of-veterans-affairs/component-library/dist/react-bindings'; +import { + VaButtonPair, + VaRadio, +} from '@department-of-veterans-affairs/component-library/dist/react-bindings'; +import { focusElement, scrollToTop } from 'platform/utilities/ui'; + +import { HelpTextGeneral, HelpTextModalities } from '../../HelpText'; +import { BTSSS_PORTAL_URL } from '../../../constants'; + +const AddressPage = ({ + address, + pageIndex, + setPageIndex, + yesNo, + setYesNo, + setIsUnsupportedClaimType, +}) => { + useEffect( + () => { + scrollToTop('topScrollElement'); + if (!address) { + focusElement('h1'); + } else { + focusElement('h1', {}, 'va-radio'); + } + }, + [address], + ); + + const [requiredAlert, setRequiredAlert] = useState(false); + + const handlers = { + onNext: () => { + if (!yesNo.address) { + setRequiredAlert(true); + } else if (yesNo.address !== 'yes') { + setIsUnsupportedClaimType(true); + } else { + setIsUnsupportedClaimType(false); + setPageIndex(pageIndex + 1); + } + }, + onBack: () => { + setPageIndex(pageIndex - 1); + }, + }; + + if (!address) { + return ( + <> +

+ Did you travel from your home address? +

+ +

You don’t have an address on file

+

+ We’re sorry, we don’t have an address on file for you and can’t file + a claim in this tool right now. +

+
+ + + + + ); + } -const AddressPage = ({ handlers }) => { return (
-

Address page

+ { + setYesNo({ ...yesNo, address: e.detail.value }); + }} + value={yesNo.address} + data-testid="address-test-id" + error={requiredAlert ? 'You must make a selection to continue.' : null} + header-aria-describedby={null} + hint="" + label="" + label-header-level="" + > +
+

+ Answer “Yes” if you traveled from the address listed here and you + confirm that it’s not a Post Office box. +

+
+

+ Home address +
+ {address.addressLine1} +
+ {address.addressLine2 && ( + <> + {address.addressLine2} +
+ + )} + {address.addressLine3 && ( + <> + {address.addressLine3} +
+ + )} + {`${address.city}, ${address.stateCode} ${address.zipCode}`} +
+

+
+
+ + +
+ +

+ + If you traveled from a different address, you can’t file a claim in + this tool right now. + {' '} + But you can file your claim online, within 30 days, through the + + . Or you can use VA Form 10-3542 to submit a claim by mail or in + person. +

+
handlers.onNext(e)} - onSecondaryClick={e => handlers.onBack(e)} + onPrimaryClick={handlers.onNext} + onSecondaryClick={handlers.onBack} />
); }; AddressPage.propTypes = { - handlers: PropTypes.shape({ - onBack: PropTypes.func, - onNext: PropTypes.func, - }), + address: PropTypes.object, + pageIndex: PropTypes.number, + setIsUnsupportedClaimType: PropTypes.func, + setPageIndex: PropTypes.func, + setYesNo: PropTypes.func, + yesNo: PropTypes.object, }; export default AddressPage; diff --git a/src/applications/travel-pay/components/submit-flow/pages/MileagePage.jsx b/src/applications/travel-pay/components/submit-flow/pages/MileagePage.jsx index bf35c01b0c99..aeb1c425db23 100644 --- a/src/applications/travel-pay/components/submit-flow/pages/MileagePage.jsx +++ b/src/applications/travel-pay/components/submit-flow/pages/MileagePage.jsx @@ -1,28 +1,150 @@ -import React from 'react'; +import React, { useState, useEffect } from 'react'; import PropTypes from 'prop-types'; -import { VaButtonPair } from '@department-of-veterans-affairs/component-library/dist/react-bindings'; +import { + VaButtonPair, + VaRadio, +} from '@department-of-veterans-affairs/component-library/dist/react-bindings'; +import { focusElement, scrollToTop } from 'platform/utilities/ui'; -const MileagePage = ({ handlers }) => { +import { formatDateTime } from '../../../util/dates'; +import { BTSSS_PORTAL_URL } from '../../../constants'; + +const MileagePage = ({ + appointment, + pageIndex, + setPageIndex, + yesNo, + setYesNo, + setIsUnsupportedClaimType, +}) => { + useEffect(() => { + focusElement('h1', {}, 'va-radio'); + scrollToTop('topScrollElement'); + }, []); + + const [formattedDate, formattedTime] = formatDateTime( + appointment.vaos.apiData.start, + ); + + const [requiredAlert, setRequiredAlert] = useState(false); + + const handlers = { + onNext: () => { + if (!yesNo.mileage) { + setRequiredAlert(true); + } else if (yesNo.mileage !== 'yes') { + setIsUnsupportedClaimType(true); + } else { + setIsUnsupportedClaimType(false); + setPageIndex(pageIndex + 1); + } + }, + onBack: () => { + setPageIndex(pageIndex - 1); + }, + }; return (
-

Mileage page

+ { + setYesNo({ ...yesNo, mileage: e.detail.value }); + }} + value={yesNo.mileage} + data-testid="mileage-test-id" + error={requiredAlert ? 'You must make a selection to continue.' : null} + header-aria-describedby={null} + hint="" + label="" + label-header-level="" + > +
+
+

+ {' '} + + {formattedDate} {formattedTime} at{' '} + {appointment.vaos.apiData.location.attributes.name} + +

+

{appointment.vaos.apiData.reasonForAppointment}

+
+
+ + +
+ + +
    +
  • We pay round-trip mileage for your scheduled appointments.
  • +
  • + We may only pay return mileage for unscheduled appointments, like + walk-ins and labs. +
  • +
+ +
+ + +

+ + If you need to submit receipts for other expenses like tolls, meals, + or lodging, you can’t file a claim in this tool right now. + {' '} + But you can file your claim online, within 30 days, through the{' '} + + . Or you can use VA Form 10-3542 to submit a claim by mail or in + person. +

+
handlers.onNext(e)} - onSecondaryClick={e => handlers.onBack(e)} + onPrimaryClick={handlers.onNext} + onSecondaryClick={handlers.onBack} />
); }; MileagePage.propTypes = { - handlers: PropTypes.shape({ - onBack: PropTypes.func, - onNext: PropTypes.func, - }), + appointment: PropTypes.object, + pageIndex: PropTypes.number, + setIsUnsupportedClaimType: PropTypes.func, + setPageIndex: PropTypes.func, + setYesNo: PropTypes.func, + yesNo: PropTypes.object, }; export default MileagePage; diff --git a/src/applications/travel-pay/components/submit-flow/pages/UnsupportedClaimTypePage.jsx b/src/applications/travel-pay/components/submit-flow/pages/UnsupportedClaimTypePage.jsx index 5ff42a7a90b0..4824b3b92bad 100644 --- a/src/applications/travel-pay/components/submit-flow/pages/UnsupportedClaimTypePage.jsx +++ b/src/applications/travel-pay/components/submit-flow/pages/UnsupportedClaimTypePage.jsx @@ -6,8 +6,7 @@ const UnsupportedClaimTypePage = ({ setIsUnsupportedClaimType, setPageIndex, }) => { - const onBack = e => { - e.preventDefault(); + const onBack = () => { setIsUnsupportedClaimType(false); setPageIndex(pageIndex); }; @@ -17,11 +16,7 @@ const UnsupportedClaimTypePage = ({

We can’t file this type of travel reimbursement claim

- onBack(e)} - /> + ); }; diff --git a/src/applications/travel-pay/components/submit-flow/pages/VehiclePage.jsx b/src/applications/travel-pay/components/submit-flow/pages/VehiclePage.jsx index 4d343d901ccf..1a851f635d90 100644 --- a/src/applications/travel-pay/components/submit-flow/pages/VehiclePage.jsx +++ b/src/applications/travel-pay/components/submit-flow/pages/VehiclePage.jsx @@ -1,12 +1,94 @@ -import React from 'react'; +import React, { useState, useEffect } from 'react'; import PropTypes from 'prop-types'; -import { VaButtonPair } from '@department-of-veterans-affairs/component-library/dist/react-bindings'; +import { + VaButtonPair, + VaRadio, +} from '@department-of-veterans-affairs/component-library/dist/react-bindings'; +import { focusElement, scrollToTop } from 'platform/utilities/ui'; +import { BTSSS_PORTAL_URL } from '../../../constants'; + +const VehiclePage = ({ + pageIndex, + setPageIndex, + yesNo, + setYesNo, + setIsUnsupportedClaimType, +}) => { + useEffect(() => { + focusElement('h1', {}, 'va-radio'); + scrollToTop('topScrollElement'); + }, []); + + const [requiredAlert, setRequiredAlert] = useState(false); + + const handlers = { + onNext: () => { + if (!yesNo.vehicle) { + setRequiredAlert(true); + } else if (yesNo.vehicle !== 'yes') { + setIsUnsupportedClaimType(true); + } else { + setIsUnsupportedClaimType(false); + setPageIndex(pageIndex + 1); + } + }, + onBack: () => { + setPageIndex(pageIndex - 1); + }, + }; -const VehiclePage = ({ handlers }) => { return (
-

Vehicle page

+ setYesNo({ ...yesNo, vehicle: e.detail.value })} + value={yesNo.vehicle} + data-testid="vehicle-test-id" + error={requiredAlert ? 'You must make a selection to continue.' : null} + header-aria-describedby={null} + hint="" + label="" + label-header-level="" + > + + + + + +

+ + If you traveled by bus, train, taxi, or other authorized public + transportation, you can’t file a claim in this tool right now. + {' '} + But you can file your claim online, within 30 days, through the{' '} + + . Or you can use VA Form 10-3542 to submit a claim by mail or in + person. +

+
{ }; VehiclePage.propTypes = { - handlers: PropTypes.shape({ - onBack: PropTypes.func, - onNext: PropTypes.func, - }), + pageIndex: PropTypes.number, + setIsUnsupportedClaimType: PropTypes.func, + setPageIndex: PropTypes.func, + setYesNo: PropTypes.func, + yesNo: PropTypes.object, }; export default VehiclePage; diff --git a/src/applications/travel-pay/containers/SubmitFlowWrapper.jsx b/src/applications/travel-pay/containers/SubmitFlowWrapper.jsx index 6acd19411afb..3be61f1aa0f0 100644 --- a/src/applications/travel-pay/containers/SubmitFlowWrapper.jsx +++ b/src/applications/travel-pay/containers/SubmitFlowWrapper.jsx @@ -1,7 +1,15 @@ import React, { useState } from 'react'; +import PropTypes from 'prop-types'; +import { connect, useSelector } from 'react-redux'; import { Element } from 'platform/utilities/scroll'; import { useFeatureToggle } from 'platform/utilities/feature-toggles/useFeatureToggle'; +import { + selectVAPMailingAddress, + selectVAPResidentialAddress, + isProfileLoading, + isLoggedIn, +} from 'platform/user/selectors'; import IntroductionPage from '../components/submit-flow/pages/IntroductionPage'; import MileagePage from '../components/submit-flow/pages/MileagePage'; @@ -15,13 +23,19 @@ import UnsupportedClaimTypePage from '../components/submit-flow/pages/Unsupporte import SubmissionErrorPage from '../components/submit-flow/pages/SubmissionErrorPage'; import { appointment1 } from '../services/mocks/appointments'; -const SubmitFlowWrapper = () => { +const SubmitFlowWrapper = ({ homeAddress, mailingAddress }) => { // TODO: Placeholders until backend integration is complete // API call based on the URL Params, but for now is hard coded const appointment = appointment1; // This will actually be handled by the redux action, but for now it lives here const [isSubmissionError, setIsSubmissionError] = useState(false); + const [yesNo, setYesNo] = useState({ + mileage: '', + vehicle: '', + address: '', + }); + const [pageIndex, setPageIndex] = useState(0); const [isUnsupportedClaimType, setIsUnsupportedClaimType] = useState(false); @@ -61,15 +75,41 @@ const SubmitFlowWrapper = () => { }, { page: 'mileage', - component: , + component: ( + + ), }, { page: 'vehicle', - component: , + component: ( + + ), }, { page: 'address', - component: , + component: ( + + ), }, { page: 'review', @@ -81,6 +121,9 @@ const SubmitFlowWrapper = () => { }, ]; + const profileLoading = useSelector(state => isProfileLoading(state)); + const userLoggedIn = useSelector(state => isLoggedIn(state)); + const { useToggleValue, useToggleLoadingValue, @@ -92,7 +135,7 @@ const SubmitFlowWrapper = () => { TOGGLE_NAMES.travelPaySubmitMileageExpense, ); - if (toggleIsLoading) { + if ((profileLoading && !userLoggedIn) || toggleIsLoading) { return (
{ ); }; -export default SubmitFlowWrapper; +SubmitFlowWrapper.propTypes = { + homeAddress: PropTypes.object, + mailingAddress: PropTypes.object, +}; + +function mapStateToProps(state) { + return { + homeAddress: selectVAPResidentialAddress(state), + mailingAddress: selectVAPMailingAddress(state), + }; +} + +export default connect(mapStateToProps)(SubmitFlowWrapper); diff --git a/src/applications/travel-pay/tests/components/submit-flow/pages/AddressPage.unit.spec.jsx b/src/applications/travel-pay/tests/components/submit-flow/pages/AddressPage.unit.spec.jsx index 4248b256763b..1c74cd17861c 100644 --- a/src/applications/travel-pay/tests/components/submit-flow/pages/AddressPage.unit.spec.jsx +++ b/src/applications/travel-pay/tests/components/submit-flow/pages/AddressPage.unit.spec.jsx @@ -1,19 +1,104 @@ import React from 'react'; import { expect } from 'chai'; -import { render } from '@testing-library/react'; +import { render, fireEvent, waitFor } from '@testing-library/react'; + import { $ } from 'platform/forms-system/src/js/utilities/ui'; import AddressPage from '../../../../components/submit-flow/pages/AddressPage'; -it('should render with back and continue buttons', () => { +const homeAddress = { + addressLine1: '345 Home Address St.', + addressLine2: null, + addressLine3: null, + addressPou: 'RESIDENCE/CHOICE', + addressType: 'DOMESTIC', + city: 'San Francisco', + countryName: 'United States', + countryCodeIso2: 'US', + countryCodeIso3: 'USA', + countryCodeFips: null, + countyCode: null, + countyName: null, + internationalPostalCode: null, + sourceSystemUser: null, + stateCode: 'CA', + zipCode: '94118', +}; + +const mailingAddress = { + addressLine1: '123 Mailing Address St.', + addressLine2: null, + addressLine3: null, + addressPou: 'CORRESPONDENCE', + addressType: 'DOMESTIC', + city: 'San Francisco', + countryName: 'United States', + countryCodeIso2: 'US', + countryCodeIso3: 'USA', + countryCodeFips: null, + countyCode: null, + countyName: null, + internationalPostalCode: null, + sourceSystemUser: null, + stateCode: 'CA', + zipCode: '94118', +}; + +describe('Address page', () => { const props = { - handlers: { - onBack: () => {}, - onNext: () => {}, + pageIndex: 3, + setPageIndex: () => {}, + yesNo: { + mileage: 'yes', + vehicle: 'yes', + address: '', }, + setYesNo: () => {}, + setIsUnsupportedClaimType: () => {}, }; - const screen = render(); - expect(screen.getByText('Address page')).to.exist; - expect($('va-button-pair')).to.exist; + it('should render with user home address', async () => { + const screen = render(); + + expect(screen.getByTestId('address-test-id')).to.exist; + expect(screen.findByText('345 Home Address St')).to.exist; + expect($('va-button-pair')).to.exist; + + fireEvent.click( + $( + `va-additional-info[trigger="If you didn't travel from your home address"]`, + ), + ); + await waitFor(() => { + expect(screen.findByText(/If you traveled from a different address/i)).to + .exist; + }); + }); + + it('should render with mail address if no home address', () => { + const screen = render(); + + expect(screen.getByTestId('address-test-id')).to.exist; + expect(screen.findByText('123 Mailing Address St')).to.exist; + expect($('va-button-pair')).to.exist; + }); + + it('should show an alert if no address', () => { + const screen = render(); + + expect(screen.queryByTestId('address-test-id')).to.not.exist; + expect($('va-button-pair')).to.not.exist; + expect($('va-alert')).to.exist; + expect(screen.findByText(/You don't have an address on file/i)).to.exist; + }); + + it('should render an error if no selection made', async () => { + const screen = render(); + + expect(screen.getByTestId('address-test-id')).to.exist; + $('va-button-pair').__events.primaryClick(); // continue + await waitFor(() => { + expect(screen.findByText(/You must make a selection/i)).to.exist; + }); + }); }); diff --git a/src/applications/travel-pay/tests/components/submit-flow/pages/MileagePage.unit.spec.jsx b/src/applications/travel-pay/tests/components/submit-flow/pages/MileagePage.unit.spec.jsx index 9edccecc810c..0555eecd0ca9 100644 --- a/src/applications/travel-pay/tests/components/submit-flow/pages/MileagePage.unit.spec.jsx +++ b/src/applications/travel-pay/tests/components/submit-flow/pages/MileagePage.unit.spec.jsx @@ -1,19 +1,245 @@ import React from 'react'; import { expect } from 'chai'; -import { render } from '@testing-library/react'; +import { render, fireEvent, waitFor } from '@testing-library/react'; import { $ } from 'platform/forms-system/src/js/utilities/ui'; import MileagePage from '../../../../components/submit-flow/pages/MileagePage'; -it('should render with back and continue buttons', () => { +const appointment = { + resourceType: 'Appointment', + id: 'aa6bb54b5f8ba22a82720a30abdfa3efe0805cc0dc1b6b248815e942ad61847e', + status: 'booked', + cancelationReason: null, + start: '2024-11-20T10:30:00-05:00', + patientComments: null, + reasonForAppointment: 'Medication concern', + timezone: 'Asia/Manila', + description: 'VAOS_UNKNOWN', + minutesDuration: 60, + practitioners: [ + { + identifier: [ + { + system: 'https://veteran.apps.va.gov/terminology/fhir/sid/secid', + value: null, + }, + ], + name: { + family: 'BERNARDO', + given: ['KENNETH J'], + }, + }, + ], + location: { + vistaId: '983', + clinicId: '945', + stationId: '983GC', + clinicName: 'C&P BEV AUDIO FTC', + clinicPhysicalLocation: 'FORT COLLINS AUDIO', + clinicPhone: null, + clinicPhoneExtension: null, + }, + videoData: { + isVideo: false, + }, + communityCareProvider: null, + preferredProviderName: null, + vaos: { + isPendingAppointment: false, + isUpcomingAppointment: false, + isVideo: false, + isPastAppointment: true, + isCompAndPenAppointment: true, + isCancellable: false, + appointmentType: 'vaAppointment', + isCommunityCare: false, + isExpressCare: false, + isPhoneAppointment: false, + isCOVIDVaccine: false, + apiData: { + id: '167322', + identifier: [ + { + system: 'Appointment/', + value: '4139383339353233', + }, + ], + kind: 'clinic', + status: 'booked', + serviceType: 'audiology', + serviceTypes: [ + { + coding: [ + { + system: + 'http://veteran.apps.va.gov/terminologies/fhir/CodeSystem/vats-service-type', + code: 'audiology', + }, + ], + }, + ], + serviceCategory: [ + { + coding: [ + { + system: 'http://www.va.gov/Terminology/VistADefinedTerms/409_1', + code: 'COMPENSATION & PENSION', + display: 'COMPENSATION & PENSION', + }, + ], + text: 'COMPENSATION & PENSION', + }, + ], + patientIcn: '1013120826V646496', + locationId: '983GC', + localStartTime: '2024-11-20T10:30:00.000+08:00', + clinic: '945', + start: '2024-11-20T09:30:00Z', + end: '2024-11-20T10:00:00Z', + created: '2024-03-17T00:00:00Z', + cancellable: false, + extension: { + ccLocation: { + address: {}, + }, + vistaStatus: ['NO ACTION TAKEN'], + }, + serviceName: 'FTC C&P AUDIO BEV', + physicalLocation: 'FORT COLLINS AUDIO', + friendlyName: 'C&P BEV AUDIO FTC', + practitioners: [ + { + identifier: [ + { + system: 'https://veteran.apps.va.gov/terminology/fhir/sid/secid', + value: null, + }, + ], + name: { + family: 'BERNARDO', + given: ['KENNETH J'], + }, + }, + ], + location: { + id: '983GC', + type: 'appointments', + attributes: { + id: '983GC', + vistaSite: '983', + vastParent: '983', + type: 'va_facilities', + name: 'Fort Collins VA Clinic', + classification: 'Multi-Specialty CBOC', + timezone: { + timeZoneId: 'Asia/Manila', + }, + lat: 40.553875, + long: -105.08795, + website: + 'https://www.cheyenne.va.gov/locations/Fort_Collins_VA_CBOC.asp', + phone: { + main: '970-224-1550', + }, + physicalAddress: { + line: ['2509 Research Boulevard'], + city: 'Fort Collins', + state: 'CO', + postalCode: '80526-8108', + }, + healthService: [ + 'Audiology', + 'EmergencyCare', + 'MentalHealthCare', + 'PrimaryCare', + 'SpecialtyCare', + ], + }, + }, + claim: { + message: 'No claim for this appointment', + }, + }, + timeZone: 'Asia/Manila', + facilityData: { + resourceType: 'Location', + id: '983GC', + vistaId: '983', + name: 'Fort Collins VA Clinic', + identifier: [ + { + system: 'http://med.va.gov/fhir/urn', + value: 'urn:va:division:983:983GC', + }, + { + system: 'urn:oid:2.16.840.1.113883.6.233', + value: '983GC', + }, + ], + telecom: [ + { + system: 'phone', + value: '970-224-1550', + }, + ], + position: { + longitude: -105.08795, + latitude: 40.553875, + }, + address: { + line: ['2509 Research Boulevard'], + city: 'Fort Collins', + state: 'CO', + postalCode: '80526-8108', + }, + }, + }, + version: 2, +}; + +describe('Mileage page', () => { const props = { - handlers: { - onBack: () => {}, - onNext: () => {}, + appointment, + pageIndex: 1, + setPageIndex: () => {}, + yesNo: { + mileage: '', + vehicle: '', + address: '', }, + setYesNo: () => {}, + setIsUnsupportedClaimType: () => {}, }; - const screen = render(); - expect(screen.getByText('Mileage page')).to.exist; - expect($('va-button-pair')).to.exist; + it('should render correctly', async () => { + const screen = render(); + + expect(screen.getByTestId('mileage-test-id')).to.exist; + expect(screen.findByText('Fort Collins VA Clinic')).to.exist; + expect($('va-button-pair')).to.exist; + + fireEvent.click( + $(`va-additional-info[trigger="How do we calculate mileage"]`), + ); + await waitFor(() => { + expect(screen.findByText(/We pay round-trip mileage/i)).to.exist; + }); + + fireEvent.click( + $(`va-additional-info[trigger="If you have other expenses to claim"]`), + ); + await waitFor(() => { + expect(screen.findByText(/submit receipts for other expenses/i)).to.exist; + }); + }); + + it('should render an error if no selection made', async () => { + const screen = render(); + + expect(screen.getByTestId('mileage-test-id')).to.exist; + $('va-button-pair').__events.primaryClick(); // continue + await waitFor(() => { + expect(screen.findByText(/You must make a selection/i)).to.exist; + }); + }); }); diff --git a/src/applications/travel-pay/tests/components/submit-flow/pages/VehiclePage.unit.spec.jsx b/src/applications/travel-pay/tests/components/submit-flow/pages/VehiclePage.unit.spec.jsx index 45c4fbabd4b5..14442098fff1 100644 --- a/src/applications/travel-pay/tests/components/submit-flow/pages/VehiclePage.unit.spec.jsx +++ b/src/applications/travel-pay/tests/components/submit-flow/pages/VehiclePage.unit.spec.jsx @@ -1,19 +1,51 @@ import React from 'react'; import { expect } from 'chai'; -import { render } from '@testing-library/react'; +import { render, fireEvent, waitFor } from '@testing-library/react'; import { $ } from 'platform/forms-system/src/js/utilities/ui'; import VehiclePage from '../../../../components/submit-flow/pages/VehiclePage'; -it('should render with back and continue buttons', () => { +describe('Vehicle page', () => { const props = { - handlers: { - onBack: () => {}, - onNext: () => {}, + pageIndex: 2, + setPageIndex: () => {}, + yesNo: { + mileage: 'yes', + vehicle: '', + address: '', }, + setYesNo: () => {}, + setIsUnsupportedClaimType: () => {}, }; - const screen = render(); - expect(screen.getByText('Vehicle page')).to.exist; - expect($('va-button-pair')).to.exist; + it('should render correctly', async () => { + const screen = render(); + + expect(screen.getByTestId('vehicle-test-id')).to.exist; + expect(screen.findByText('Did you travel in your own vehicle?')).to.exist; + expect($('va-button-pair')).to.exist; + + fireEvent.click( + $( + `va-additional-info[trigger="If you didn't travel in your own vehicle"]`, + ), + ); + await waitFor(() => { + expect( + screen.findByText( + / bus, train, taxi, or other authorized public transportation/i, + ), + ).to.exist; + }); + }); + + it('should render an error if no selection made', async () => { + const screen = render(); + + expect(screen.getByTestId('vehicle-test-id')).to.exist; + $('va-button-pair').__events.primaryClick(); // continue + await waitFor(() => { + expect(screen.findByText(/You must make a selection/i)).to.exist; + }); + }); }); diff --git a/src/applications/travel-pay/tests/containers/TravelPayStatusApp.unit.spec.jsx b/src/applications/travel-pay/tests/containers/TravelPayStatusApp.unit.spec.jsx index 51e266a0e049..d1e7eb4df30b 100644 --- a/src/applications/travel-pay/tests/containers/TravelPayStatusApp.unit.spec.jsx +++ b/src/applications/travel-pay/tests/containers/TravelPayStatusApp.unit.spec.jsx @@ -159,20 +159,6 @@ describe('App', () => { .exist; }); - it('shows the login modal when clicking the login prompt', async () => { - const { container } = renderWithStoreAndRouter(, { - initialState: getData({ - areFeatureTogglesLoading: false, - hasFeatureFlag: true, - isLoggedIn: true, - }), - path: `/claims/`, - reducers: reducer, - }); - - expect($('va-loading-indicator', container)).to.exist; - }); - it('handles a failed fetch of claims', async () => { global.fetch.restore(); mockApiRequest({ errors: [{ title: 'Bad Request', status: 400 }] }, false); diff --git a/src/applications/travel-pay/tests/e2e/submit-claim.cypress.spec.js b/src/applications/travel-pay/tests/e2e/submit-claim.cypress.spec.js index be74185704eb..f0f97610bd85 100644 --- a/src/applications/travel-pay/tests/e2e/submit-claim.cypress.spec.js +++ b/src/applications/travel-pay/tests/e2e/submit-claim.cypress.spec.js @@ -16,33 +16,72 @@ describe('Submit Mileage Only Claims', () => { cy.get('h1').should('include.text', 'File a travel reimbursement claim'); }); - it('should navigate through the flow', () => { + it('should handle validation and answering "No" and navigate through the flow', () => { cy.get('va-link-action[text="File a mileage only claim"]') .first() .click(); - // Mileage question should be first - cy.get('h1').should('include.text', 'Mileage page'); + // Test that a No answer sends user to "Can't file this type..." page + cy.get('h1').should('include.text', 'Are you claiming only mileage?'); + + // Answer "No" and continue + cy.get('va-radio-option[label="No"]') + .first() + .click(); + + cy.selectVaButtonPairPrimary(); + + cy.get('h1').should( + 'include.text', + `We can’t file this type of travel reimbursement claim`, + ); + + cy.get('va-button[text="Back"]') + .first() + .click(); + + cy.get('h1').should('include.text', 'Are you claiming only mileage?'); + + // Answer "Yes" and continue through the rest of the flow + cy.get('va-radio-option[label="Yes"]') + .first() + .click(); - // Click the "Continue" button cy.selectVaButtonPairPrimary(); - // Then navigate to the Vehicle question - cy.get('h1').should('include.text', 'Vehicle page'); + // Test that not selecting any answer triggers an error + cy.get('h1').should('include.text', 'Did you travel in your own vehicle?'); - // Click the "Continue" button cy.selectVaButtonPairPrimary(); - // Then navigate to the Address question - cy.get('h1').should('include.text', 'Address page'); + cy.get('.usa-error-message').should( + 'include.text', + 'You must make a selection to continue.', + ); + + // Answer "Yes" and continue through the rest of the flow + cy.get('va-radio-option[label="Yes"]') + .first() + .click(); + + cy.selectVaButtonPairPrimary(); + + // Address question + cy.get('h1').should( + 'include.text', + 'Did you travel from your home address?', + ); + + // Answer "yes" and continue + cy.get('va-radio-option[label="Yes"]') + .first() + .click(); - // Click the "Continue" button cy.selectVaButtonPairPrimary(); - // Then navigate to the Review page + // Review page cy.get('h1').should('include.text', 'Review your travel claim'); - // Click the "Submit" button cy.get('va-button[text="Submit"]') .first() .click();