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();