Skip to content

Commit 563dceb

Browse files
authored
Release v1.32.1 (#263)
1 parent 51d895b commit 563dceb

File tree

7 files changed

+184
-10
lines changed

7 files changed

+184
-10
lines changed

CHANGELOG.md

+9-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,12 @@ to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
77

88
## [Unreleased]
99

10+
## [1.32.1] - 2025-02-06
11+
12+
### Fixed
13+
14+
- Fix birthday field value format
15+
1016
## [1.32.0] - 2025-02-05
1117

1218
### Added
@@ -560,7 +566,9 @@ The eye icon is now correctly displayed in the Auth widget.
560566

561567
First version of the SDK Web UI.
562568

563-
[Unreleased]: https://github.com/ReachFive/identity-web-ui-sdk/compare/v1.32.0...HEAD
569+
[Unreleased]: https://github.com/ReachFive/identity-web-ui-sdk/compare/1.32.1...HEAD
570+
571+
[1.32.1]: https://github.com/ReachFive/identity-web-ui-sdk/compare/v1.32.0...v1.32.1
564572

565573
[1.32.0]: https://github.com/ReachFive/identity-web-ui-sdk/compare/v1.31.3...v1.32.0
566574

package-lock.json

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@reachfive/identity-ui",
3-
"version": "1.32.0",
3+
"version": "1.32.1",
44
"description": "ReachFive Identity Web UI SDK",
55
"author": "ReachFive",
66
"repository": {

src/components/form/fields/birthdayField.tsx

+15-1
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,9 @@
1-
import { differenceInYears } from "date-fns"
1+
import { differenceInYears, formatISO, isValid, parseISO } from "date-fns"
22

33
import dateField from './dateField'
44
import { Validator } from '../../../core/validation';
55
import { Config } from '../../../types';
6+
import { isRichFormValue } from "../../../helpers/utils";
67

78
export const ageLimitValidator = (min = 6, max = 129) => new Validator<Date>({
89
rule: (value) => {
@@ -20,6 +21,19 @@ export default function birthdateField(
2021
return dateField({
2122
...props,
2223
label: label,
24+
format: {
25+
bind: (value) => {
26+
const dt = value ? parseISO(value) : undefined
27+
return dt && isValid(dt) ? { raw: dt } : undefined
28+
},
29+
unbind: (value) => {
30+
return isRichFormValue(value, 'raw')
31+
? formatISO(value.raw, { representation: 'date' })
32+
: value
33+
? formatISO(value, { representation: 'date' })
34+
: null
35+
}
36+
},
2337
validator: ageLimitValidator(min, max)
2438
}, config)
2539
}

src/components/form/fields/dateField.tsx

+5-3
Original file line numberDiff line numberDiff line change
@@ -200,10 +200,12 @@ export const datetimeValidator = (locale: string) => new Validator<Date>({
200200

201201
export default function dateField(
202202
{
203+
format,
203204
key = 'date',
204205
label = 'date',
205-
yearDebounce,
206206
locale,
207+
validator,
208+
yearDebounce,
207209
...props
208210
}: Optional<FieldDefinition<string, Date>, 'key' | 'label'> & Optional<ExtraParams, 'locale'>,
209211
config: Config
@@ -212,7 +214,7 @@ export default function dateField(
212214
key,
213215
label,
214216
...props,
215-
format: {
217+
format: format ?? {
216218
bind: (value) => {
217219
const dt = value ? parseISO(value) : undefined
218220
return dt && isValid(dt) ? { raw: dt } : undefined
@@ -225,7 +227,7 @@ export default function dateField(
225227
: null
226228
}
227229
},
228-
validator: props.validator ? datetimeValidator(config.language).and(props.validator) : datetimeValidator(config.language),
230+
validator: validator ? datetimeValidator(config.language).and(validator) : datetimeValidator(config.language),
229231
component: DateField,
230232
extendedParams: {
231233
locale: locale ?? config.language,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,150 @@
1+
/**
2+
* @jest-environment jsdom
3+
*/
4+
5+
import React from 'react'
6+
import { afterEach, beforeEach, describe, expect, jest, test } from '@jest/globals';
7+
import { render, screen, waitFor } from '@testing-library/react';
8+
import userEvent from '@testing-library/user-event'
9+
import '@testing-library/jest-dom/jest-globals'
10+
import 'jest-styled-components';
11+
import { formatISO, getDate, getMonth, getYear, startOfDay, subYears } from 'date-fns';
12+
13+
import type { Config } from '../../../../src/types';
14+
15+
import { createForm } from '../../../../src/components/form/formComponent'
16+
import birthdayField from '../../../../src/components/form/fields/birthdayField'
17+
import { I18nMessages } from '../../../../src/core/i18n';
18+
import { WidgetContext } from '../WidgetContext';
19+
20+
const defaultConfig: Config = {
21+
clientId: 'local',
22+
domain: 'local.reach5.net',
23+
sso: false,
24+
sms: false,
25+
webAuthn: false,
26+
language: 'fr',
27+
pkceEnforced: false,
28+
isPublic: true,
29+
socialProviders: ['facebook', 'google'],
30+
customFields: [],
31+
resourceBaseUrl: 'http://localhost',
32+
mfaSmsEnabled: false,
33+
mfaEmailEnabled: false,
34+
rbaEnabled: false,
35+
consentsVersions: {},
36+
passwordPolicy: {
37+
minLength: 8,
38+
minStrength: 2,
39+
allowUpdateWithAccessTokenOnly: true,
40+
}
41+
};
42+
43+
const defaultI18n: I18nMessages = {
44+
date: 'Date',
45+
year: 'Année',
46+
month: 'Mois',
47+
day: 'Jour',
48+
}
49+
50+
type Model = { date: string }
51+
52+
describe('DOM testing', () => {
53+
54+
beforeEach(() => {
55+
jest.useFakeTimers();
56+
})
57+
58+
afterEach(() => {
59+
jest.runOnlyPendingTimers()
60+
jest.useRealTimers();
61+
});
62+
63+
test('default settings', async () => {
64+
const user = userEvent.setup({ advanceTimers: jest.advanceTimersByTime })
65+
66+
const key = 'birthday'
67+
const label = 'birthday'
68+
69+
const onFieldChange = jest.fn()
70+
const onSubmit = jest.fn<(data: Model) => Promise<Model>>(data => Promise.resolve(data))
71+
72+
const Form = createForm<Model>({
73+
fields: [
74+
birthdayField({ key, label }, defaultConfig)
75+
],
76+
})
77+
78+
await waitFor(async () => {
79+
return render(
80+
<WidgetContext
81+
config={defaultConfig}
82+
defaultMessages={defaultI18n}
83+
>
84+
<Form
85+
fieldValidationDebounce={0} // trigger validation instantly
86+
onFieldChange={onFieldChange}
87+
handler={onSubmit}
88+
/>
89+
</WidgetContext>
90+
)
91+
})
92+
93+
const yearInput = screen.getByTestId('birthday.year')
94+
const monthInput = screen.getByTestId('birthday.month')
95+
const dayInput = screen.getByTestId('birthday.day')
96+
97+
const fiveYearsOld = subYears(new Date(), 5)
98+
await user.clear(yearInput)
99+
await user.type(yearInput, String(getYear(fiveYearsOld)))
100+
101+
// Fast-forward until all timers have been executed (handle year debounced value)
102+
await jest.runOnlyPendingTimersAsync();
103+
104+
await user.selectOptions(monthInput, String(getMonth(fiveYearsOld)))
105+
await user.selectOptions(dayInput, String(getDate(fiveYearsOld)))
106+
107+
await waitFor(() => expect(onFieldChange).toHaveBeenLastCalledWith(
108+
expect.objectContaining({
109+
birthday: expect.objectContaining({
110+
isDirty: true,
111+
value: startOfDay(fiveYearsOld),
112+
validation: {
113+
valid: false,
114+
error: "validation.birthdate.yearLimit"
115+
}
116+
})
117+
})
118+
))
119+
120+
const eighteenYearsOld = subYears(new Date(), 18)
121+
await user.clear(yearInput)
122+
await user.type(yearInput, String(getYear(eighteenYearsOld)))
123+
124+
// Fast-forward until all timers have been executed (handle year debounced value)
125+
await jest.runOnlyPendingTimersAsync();
126+
127+
await user.selectOptions(monthInput, String(getMonth(eighteenYearsOld)))
128+
await user.selectOptions(dayInput, String(getDate(eighteenYearsOld)))
129+
130+
await waitFor(() => expect(onFieldChange).toHaveBeenLastCalledWith(
131+
expect.objectContaining({
132+
birthday: expect.objectContaining({
133+
isDirty: true,
134+
value: startOfDay(eighteenYearsOld),
135+
})
136+
})
137+
))
138+
139+
const submitBtn = screen.getByRole('button')
140+
await user.click(submitBtn)
141+
142+
await waitFor(() => expect(onSubmit).toHaveBeenCalled())
143+
144+
expect(onSubmit).toBeCalledWith(
145+
expect.objectContaining({
146+
birthday: formatISO(eighteenYearsOld, { representation: 'date' }) // value is formatted in handler data
147+
})
148+
)
149+
})
150+
})

types/identity-ui.d.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/**
22
* @reachfive/identity-ui - v1.32.0
3-
* Compiled Wed, 05 Feb 2025 14:52:54 UTC
3+
* Compiled Thu, 06 Feb 2025 14:40:46 UTC
44
*
55
* Copyright (c) ReachFive.
66
*
@@ -390,7 +390,7 @@ type ExtraParams$2 = {
390390
};
391391
interface DateFieldProps extends FieldComponentProps<Date, ExtraParams$2> {
392392
}
393-
declare function dateField({ key, label, yearDebounce, locale, ...props }: Optional<FieldDefinition<string, Date>, 'key' | 'label'> & Optional<ExtraParams$2, 'locale'>, config: Config): FieldCreator<Date, DateFieldProps, ExtraParams$2>;
393+
declare function dateField({ format, key, label, locale, validator, yearDebounce, ...props }: Optional<FieldDefinition<string, Date>, 'key' | 'label'> & Optional<ExtraParams$2, 'locale'>, config: Config): FieldCreator<Date, DateFieldProps, ExtraParams$2>;
394394

395395
interface Option {
396396
label: string;

0 commit comments

Comments
 (0)