Skip to content

Commit

Permalink
fix(ui-date-time-input,ui-form-field): make DateTimeInput compatible …
Browse files Browse the repository at this point in the history
…with the new error format

Backport of INSTUI-4405 from the v10 branch Github PR #1826
TEST PLAN:
- check the datetimeinput examples in the error forms guide
- setting the isRequired prop should display
two separate asterisks for each field
- when required and displaying an error, the asterisks should
turn red
- the errors should be displayed under the input field like it
was a single field and not a group
  • Loading branch information
matyasf committed Dec 19, 2024
1 parent 7252438 commit f8c0fa6
Show file tree
Hide file tree
Showing 5 changed files with 49 additions and 6 deletions.
28 changes: 28 additions & 0 deletions docs/guides/form-errors.md
Original file line number Diff line number Diff line change
Expand Up @@ -135,6 +135,34 @@ const Example = () => {
renderMessages={() => messages}
/>

<DateTimeInput
description={`DateTimeInput (layout="column")`}
datePlaceholder="Choose a date"
dateRenderLabel="Date"
timeRenderLabel="Time"
invalidDateTimeMessage="Invalid date!"
prevMonthLabel="Previous month"
nextMonthLabel="Next month"
defaultValue="2018-01-18T13:30"
layout="columns"
isRequired={isRequired}
messages={messages}
/>

<DateTimeInput
description={`DateTimeInput (layout="stacked")`}
datePlaceholder="Choose a date"
dateRenderLabel="Date"
timeRenderLabel="Time"
invalidDateTimeMessage="Invalid date!"
prevMonthLabel="Previous month"
nextMonthLabel="Next month"
defaultValue="2018-01-18T13:30"
layout="stacked"
isRequired={isRequired}
messages={messages}
/>

</div>
</div>
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -173,7 +173,7 @@ describe('<DateTimeInput />', () => {
invalidDateTimeMessage={invalidDateTimeMessage}
/>
)
const timeInput = screen.getByLabelText('time-input')
const timeInput = screen.getByLabelText('time-input *')

await userEvent.type(timeInput, '1:00 PM')
fireEvent.blur(timeInput)
Expand Down
19 changes: 17 additions & 2 deletions packages/ui-date-time-input/src/DateTimeInput/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -165,7 +165,7 @@ class DateTimeInput extends Component<DateTimeInputProps, DateTimeInputState> {
? this.props.invalidDateTimeMessage(parsed.toISOString(true))
: this.props.invalidDateTimeMessage
}
errorMsg = text ? { text, type: 'error' } : undefined
errorMsg = text ? { text, type: 'newError' } : undefined
return {
iso: parsed.clone(),
calendarSelectedDate: parsed.clone(),
Expand Down Expand Up @@ -348,7 +348,7 @@ class DateTimeInput extends Component<DateTimeInputProps, DateTimeInputState> {
? this.props.invalidDateTimeMessage(dateStr ? dateStr : '')
: this.props.invalidDateTimeMessage
// eslint-disable-next-line no-param-reassign
newState.message = { text: text, type: 'error' }
newState.message = { text: text, type: 'newError' }
}
if (this.areDifferentDates(this.state.iso, newState.iso)) {
this.props.onChange?.(e, newState.iso?.toISOString())
Expand Down Expand Up @@ -547,6 +547,17 @@ class DateTimeInput extends Component<DateTimeInputProps, DateTimeInputState> {
allowNonStepInput
} = this.props

const allMessages = [
...(showMessages && this.state.message ? [this.state.message] : []),
...(messages || [])
]

const hasError = allMessages.find((m) => m.type === 'newError')
// if the component is in error state, create an empty error message to pass down to the subcomponents (DateInput and TimeInput) so they get a red outline and red required asterisk
const subComponentMessages: FormMessage[] = hasError
? [{ type: 'newError', text: '' }]
: []

return (
<FormFieldGroup
description={description}
Expand All @@ -555,6 +566,7 @@ class DateTimeInput extends Component<DateTimeInputProps, DateTimeInputState> {
colSpacing={colSpacing}
vAlign="top"
elementRef={this.handleRef}
isGroup={false}
messages={[
...(showMessages && this.state.message ? [this.state.message] : []),
...(messages || [])
Expand All @@ -581,6 +593,7 @@ class DateTimeInput extends Component<DateTimeInputProps, DateTimeInputState> {
onRequestRenderNextMonth={this.handleRenderNextMonth}
onRequestRenderPrevMonth={this.handleRenderPrevMonth}
isRequired={isRequired}
messages={subComponentMessages}
interaction={interaction}
renderNavigationLabel={
<span>
Expand All @@ -604,6 +617,8 @@ class DateTimeInput extends Component<DateTimeInputProps, DateTimeInputState> {
inputRef={timeInputRef}
interaction={interaction}
allowNonStepInput={allowNonStepInput}
isRequired={isRequired}
messages={subComponentMessages}
/>
</FormFieldGroup>
)
Expand Down
4 changes: 2 additions & 2 deletions packages/ui-form-field/src/FormFieldGroup/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -133,7 +133,7 @@ class FormFieldGroup extends Component<FormFieldGroupProps> {
}

render() {
const { styles, makeStyles, ...props } = this.props
const { styles, makeStyles, isGroup, ...props } = this.props

return (
<FormFieldLayout
Expand All @@ -145,7 +145,7 @@ class FormFieldGroup extends Component<FormFieldGroupProps> {
aria-disabled={props.disabled ? 'true' : undefined}
aria-invalid={this.invalid ? 'true' : undefined}
elementRef={this.handleRef}
isGroup
isGroup={isGroup}
>
{this.renderFields()}
</FormFieldLayout>
Expand Down
2 changes: 1 addition & 1 deletion packages/ui-form-field/src/FormFieldMessage/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,7 @@ class FormFieldMessage extends Component<FormFieldMessageProps> {

return this.props.variant !== 'screenreader-only' ? (
<span css={{ display: 'flex' }}>
{this.props.variant === 'newError' && (
{this.props.variant === 'newError' && this.props.children && (
<span css={styles?.errorIcon}>
<IconWarningSolid color="error" />
</span>
Expand Down

0 comments on commit f8c0fa6

Please sign in to comment.