src/applications/appeals/shared/components/IssueCard.jsx
import React from 'react';
import PropTypes from 'prop-types';
import { isValid, format } from 'date-fns';
import { Link } from 'react-router';
import { replaceDescriptionContent } from '../utils/replace';
import {
FORMAT_YMD_DATE_FNS,
FORMAT_READABLE_DATE_FNS,
SELECTED,
} from '../constants';
import { parseDateToDateObj } from '../utils/dates';
import '../definitions';
// It would be better to validate the date based on the app's requirements
// import { isValidDate } from '../validations/date';
/**
* IssueCardContent
* @param {String} id - unique ID
* @param {String} description - contestable issue description
* @param {String} ratingIssuePercentNumber - rating %, number with no %
* @param {String} approxDecisionDate - contestable issue date formatted as
* "YYYY-MM-DD"
* @param {String} decisionDate - additional issue date formatted as
* "YYYY-MM-DD"
* @return {JSX.Element}
*/
export const IssueCardContent = ({
id,
description,
ratingIssuePercentNumber,
approxDecisionDate,
decisionDate,
}) => {
// May need to throw an error to DataDog if any of these don't exist
// A valid rated disability *can* have a rating percentage of 0%
const showPercentNumber = (ratingIssuePercentNumber || '') !== '';
const date = parseDateToDateObj(
approxDecisionDate || decisionDate || null,
FORMAT_YMD_DATE_FNS,
);
// const dateMessage = isValidDate(date) ? (
const dateMessage = isValid(date) ? (
<strong
className="dd-privacy-hidden"
data-dd-action-name="rated issue decision date"
>
{format(date, FORMAT_READABLE_DATE_FNS)}
</strong>
) : (
<span className="usa-input-error-message vads-u-display--inline">
Invalid decision date
</span>
);
return (
<div id={id} className="widget-content-wrap">
{description && (
<p
className="vads-u-margin-bottom--0 dd-privacy-hidden"
data-dd-action-name="rated issue description"
>
{replaceDescriptionContent(description)}
</p>
)}
{showPercentNumber && (
<p className="vads-u-margin-bottom--0">
Current rating:{' '}
<strong
className="dd-privacy-hidden"
data-dd-action-name="rated issue percentage"
>
{`${ratingIssuePercentNumber}%`}
</strong>
</p>
)}
<p>Decision date: {dateMessage}</p>
</div>
);
};
IssueCardContent.propTypes = {
approxDecisionDate: PropTypes.string,
decisionDate: PropTypes.string,
description: PropTypes.string,
id: PropTypes.string,
ratingIssuePercentNumber: PropTypes.string,
};
/**
* IssueCard
* @param {String} id - ID base for form elements
* @param {Number} index - index of item in list
* @param {ContestableIssueItem|AdditionalIssueItem} item - issue values
* @param {Object} options - ui:options
* @param {func} onChange - onChange callback
* @param {Boolean} showCheckbox - don't show checkbox on review & submit
* @param {func} onRemove - remove issue callback
* page when not in edit mode
* @param {Boolean} onReviewPage - When true, list is rendered on review page
* @return {JSX.Element}
*/
export const IssueCard = ({
id,
index,
item = {},
options = {},
onChange,
showCheckbox,
onRemove,
onReviewPage,
}) => {
// On the review & submit page, there may be more than one
// of these components in edit mode with the same content, e.g. 526
// ratedDisabilities & unemployabilityDisabilities causing
// duplicate input ids/names... an `appendId` value is added to the
// ui:options
const appendId = options.appendId ? `_${options.appendId}` : '';
const elementId = `${id}_${index}${appendId}`;
const itemIsSelected = item[SELECTED];
const isEditable = !!item.issue;
const issueName = item.issue || item.ratingIssueSubjectText;
const wrapperClass = [
'widget-wrapper',
isEditable ? 'additional-issue' : '',
showCheckbox ? '' : 'checkbox-hidden',
showCheckbox ? 'vads-u-padding-top--3' : '',
'vads-u-padding-right--3',
'vads-u-margin-bottom--0',
'vads-u-border-bottom--1px',
'vads-u-border-color--gray-light',
].join(' ');
const titleClass = [
'widget-title',
'dd-privacy-hidden',
'vads-u-font-size--h4',
'vads-u-margin--0',
'capitalize',
'overflow-wrap-word',
].join(' ');
const removeButtonClass = [
'remove-issue',
'vads-u-width--auto',
'vads-u-margin-left--2',
'vads-u-margin-top--0',
].join(' ');
const handlers = {
onRemove: event => {
event.preventDefault();
onRemove(index);
},
onChange: event => onChange(index, event),
};
const editControls =
showCheckbox && isEditable ? (
<div>
<Link
to={{
pathname: '/add-issue',
search: `?index=${index}`,
}}
className="edit-issue-link"
aria-label={`Edit ${issueName}`}
>
Edit
</Link>
<va-button
secondary
class={removeButtonClass}
label={`remove ${issueName}`}
onClick={handlers.onRemove}
text="Remove"
uswds
/>
</div>
) : null;
// Issues h4 disappears in edit mode, so we need to match the page header
// level
const Header = onReviewPage ? 'h5' : 'h4';
return (
<li id={`issue-${index}`} name={`issue-${index}`} key={index}>
<div className={wrapperClass}>
{showCheckbox ? (
<div
className="widget-checkbox-wrap"
data-dd-action-name="Issue name"
>
<input
type="checkbox"
id={elementId}
name={elementId}
checked={itemIsSelected}
onChange={handlers.onChange}
aria-describedby={`issue-${index}-description`}
aria-labelledby={`issue-${index}-title`}
data-dd-action-name="Issue Name"
/>
<label
className="schemaform-label"
htmlFor={elementId}
data-dd-action-name="Contestable Issue Name"
>
{' '}
</label>
</div>
) : null}
<div
className={`widget-content ${
editControls ? 'widget-editable vads-u-padding-bottom--2' : ''
}`}
data-index={index}
>
<Header
id={`issue-${index}-title`}
className={titleClass}
data-dd-action-name="contestable issue name"
>
{issueName}
</Header>
<IssueCardContent id={`issue-${index}-description`} {...item} />
{editControls}
</div>
</div>
</li>
);
};
IssueCard.propTypes = {
id: PropTypes.string,
index: PropTypes.number,
item: PropTypes.shape({
issue: PropTypes.string,
decisionDate: PropTypes.string,
ratingIssueSubjectText: PropTypes.string,
description: PropTypes.string,
ratingIssuePercentNumber: PropTypes.string,
approxDecisionDate: PropTypes.string,
[SELECTED]: PropTypes.bool,
}),
options: PropTypes.shape({
appendId: PropTypes.string,
}),
showCheckbox: PropTypes.bool,
onChange: PropTypes.func,
onRemove: PropTypes.func,
onReviewPage: PropTypes.bool,
};