From 7a6f5621c8ab5c7a1c8e9c10ef7abe7ae08e8f28 Mon Sep 17 00:00:00 2001 From: Gao Sun Date: Tue, 5 Sep 2023 00:04:55 +0800 Subject: [PATCH] feat(ui): show password policy requirements and errors --- .../tabs/PasswordPolicy/index.tsx | 2 +- .../phrases-ui/src/locales/de/description.ts | 3 + .../locales/de/{error.ts => error/index.ts} | 3 + .../src/locales/de/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/de/index.ts | 4 +- packages/phrases-ui/src/locales/de/list.ts | 7 ++ .../phrases-ui/src/locales/en/description.ts | 3 + .../locales/en/{error.ts => error/index.ts} | 3 + .../src/locales/en/error/password-rejected.ts | 19 +++++ packages/phrases-ui/src/locales/en/index.ts | 4 +- packages/phrases-ui/src/locales/en/list.ts | 7 ++ .../phrases-ui/src/locales/es/description.ts | 3 + .../locales/es/{error.ts => error/index.ts} | 3 + .../src/locales/es/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/es/index.ts | 4 +- packages/phrases-ui/src/locales/es/list.ts | 7 ++ .../phrases-ui/src/locales/fr/description.ts | 3 + .../locales/fr/{error.ts => error/index.ts} | 3 + .../src/locales/fr/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/fr/index.ts | 4 +- packages/phrases-ui/src/locales/fr/list.ts | 7 ++ .../phrases-ui/src/locales/it/description.ts | 3 + .../locales/it/{error.ts => error/index.ts} | 3 + .../src/locales/it/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/it/index.ts | 4 +- packages/phrases-ui/src/locales/it/list.ts | 7 ++ .../phrases-ui/src/locales/ja/description.ts | 3 + .../locales/ja/{error.ts => error/index.ts} | 3 + .../src/locales/ja/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/ja/index.ts | 4 +- packages/phrases-ui/src/locales/ja/list.ts | 7 ++ .../phrases-ui/src/locales/ko/description.ts | 3 + .../locales/ko/{error.ts => error/index.ts} | 3 + .../src/locales/ko/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/ko/index.ts | 4 +- packages/phrases-ui/src/locales/ko/list.ts | 7 ++ .../src/locales/pl-pl/description.ts | 3 + .../pl-pl/{error.ts => error/index.ts} | 3 + .../locales/pl-pl/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/pl-pl/index.ts | 4 +- packages/phrases-ui/src/locales/pl-pl/list.ts | 7 ++ .../src/locales/pt-br/description.ts | 3 + .../pt-br/{error.ts => error/index.ts} | 3 + .../locales/pt-br/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/pt-br/index.ts | 4 +- packages/phrases-ui/src/locales/pt-br/list.ts | 7 ++ .../src/locales/pt-pt/description.ts | 3 + .../pt-pt/{error.ts => error/index.ts} | 3 + .../locales/pt-pt/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/pt-pt/index.ts | 4 +- packages/phrases-ui/src/locales/pt-pt/list.ts | 7 ++ .../phrases-ui/src/locales/ru/description.ts | 3 + .../locales/ru/{error.ts => error/index.ts} | 3 + .../src/locales/ru/error/password-rejected.ts | 14 ++++ packages/phrases-ui/src/locales/ru/index.ts | 4 +- packages/phrases-ui/src/locales/ru/list.ts | 7 ++ .../src/locales/tr-tr/description.ts | 3 + .../tr-tr/{error.ts => error/index.ts} | 3 + .../locales/tr-tr/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/tr-tr/index.ts | 4 +- packages/phrases-ui/src/locales/tr-tr/list.ts | 7 ++ .../src/locales/zh-cn/description.ts | 3 + .../zh-cn/{error.ts => error/index.ts} | 3 + .../locales/zh-cn/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/zh-cn/index.ts | 4 +- packages/phrases-ui/src/locales/zh-cn/list.ts | 7 ++ .../src/locales/zh-hk/description.ts | 3 + .../zh-hk/{error.ts => error/index.ts} | 3 + .../locales/zh-hk/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/zh-hk/index.ts | 4 +- packages/phrases-ui/src/locales/zh-hk/list.ts | 7 ++ .../src/locales/zh-tw/description.ts | 3 + .../zh-tw/{error.ts => error/index.ts} | 3 + .../locales/zh-tw/error/password-rejected.ts | 14 ++++ .../phrases-ui/src/locales/zh-tw/index.ts | 4 +- packages/phrases-ui/src/locales/zh-tw/list.ts | 7 ++ packages/schemas/src/types/interactions.ts | 4 +- .../core-kit/src/password-policy.test.ts | 10 +-- .../toolkit/core-kit/src/password-policy.ts | 64 +++++++-------- .../components/ErrorMessage/index.module.scss | 4 + .../ui/src/components/ErrorMessage/index.tsx | 9 ++- .../InputFields/InputField/index.tsx | 55 ++++++++----- .../src/containers/SetPassword/Lite.test.tsx | 8 +- .../ui/src/containers/SetPassword/Lite.tsx | 13 +-- .../SetPassword/SetPassword.test.tsx | 14 +++- .../containers/SetPassword/SetPassword.tsx | 12 --- .../ui/src/hooks/use-list-translation.test.ts | 57 +++++++++++++ packages/ui/src/hooks/use-list-translation.ts | 76 ++++++++++++++++++ packages/ui/src/hooks/use-password-action.ts | 80 +++++++++++++++++++ .../src/hooks/use-password-error-message.ts | 78 ++++++++++++++++++ packages/ui/src/hooks/use-sie.ts | 19 ++++- .../src/pages/Continue/SetPassword/index.tsx | 63 +++++++++++++-- .../Continue/SetPassword/use-set-password.ts | 53 ------------ .../ui/src/pages/RegisterPassword/index.tsx | 58 +++++++++++--- .../use-username-password-register.ts | 44 ---------- packages/ui/src/pages/ResetPassword/index.tsx | 64 +++++++++++++-- .../pages/ResetPassword/use-reset-password.ts | 65 --------------- 97 files changed, 1027 insertions(+), 295 deletions(-) rename packages/phrases-ui/src/locales/de/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/de/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/de/list.ts rename packages/phrases-ui/src/locales/en/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/en/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/en/list.ts rename packages/phrases-ui/src/locales/es/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/es/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/es/list.ts rename packages/phrases-ui/src/locales/fr/{error.ts => error/index.ts} (95%) create mode 100644 packages/phrases-ui/src/locales/fr/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/fr/list.ts rename packages/phrases-ui/src/locales/it/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/it/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/it/list.ts rename packages/phrases-ui/src/locales/ja/{error.ts => error/index.ts} (95%) create mode 100644 packages/phrases-ui/src/locales/ja/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/ja/list.ts rename packages/phrases-ui/src/locales/ko/{error.ts => error/index.ts} (95%) create mode 100644 packages/phrases-ui/src/locales/ko/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/ko/list.ts rename packages/phrases-ui/src/locales/pl-pl/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/pl-pl/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/pl-pl/list.ts rename packages/phrases-ui/src/locales/pt-br/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/pt-br/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/pt-br/list.ts rename packages/phrases-ui/src/locales/pt-pt/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/pt-pt/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/pt-pt/list.ts rename packages/phrases-ui/src/locales/ru/{error.ts => error/index.ts} (96%) create mode 100644 packages/phrases-ui/src/locales/ru/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/ru/list.ts rename packages/phrases-ui/src/locales/tr-tr/{error.ts => error/index.ts} (94%) create mode 100644 packages/phrases-ui/src/locales/tr-tr/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/tr-tr/list.ts rename packages/phrases-ui/src/locales/zh-cn/{error.ts => error/index.ts} (93%) create mode 100644 packages/phrases-ui/src/locales/zh-cn/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/zh-cn/list.ts rename packages/phrases-ui/src/locales/zh-hk/{error.ts => error/index.ts} (93%) create mode 100644 packages/phrases-ui/src/locales/zh-hk/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/zh-hk/list.ts rename packages/phrases-ui/src/locales/zh-tw/{error.ts => error/index.ts} (93%) create mode 100644 packages/phrases-ui/src/locales/zh-tw/error/password-rejected.ts create mode 100644 packages/phrases-ui/src/locales/zh-tw/list.ts create mode 100644 packages/ui/src/hooks/use-list-translation.test.ts create mode 100644 packages/ui/src/hooks/use-list-translation.ts create mode 100644 packages/ui/src/hooks/use-password-action.ts create mode 100644 packages/ui/src/hooks/use-password-error-message.ts delete mode 100644 packages/ui/src/pages/Continue/SetPassword/use-set-password.ts delete mode 100644 packages/ui/src/pages/RegisterPassword/use-username-password-register.ts delete mode 100644 packages/ui/src/pages/ResetPassword/use-reset-password.ts diff --git a/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx b/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx index 9cf208955..eb735d80b 100644 --- a/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx +++ b/packages/console/src/pages/SignInExperience/tabs/PasswordPolicy/index.tsx @@ -62,7 +62,7 @@ function PasswordPolicy({ isActive }: Props) { getValues, formState: { errors }, } = useFormContext(); - const { max } = getValues('passwordPolicy.length'); + const max = getValues('passwordPolicy.length.max'); const { t } = useTranslation(undefined, { keyPrefix: 'admin_console.sign_in_exp.password_policy', }); diff --git a/packages/phrases-ui/src/locales/de/description.ts b/packages/phrases-ui/src/locales/de/description.ts index ca43f9be4..63e209190 100644 --- a/packages/phrases-ui/src/locales/de/description.ts +++ b/packages/phrases-ui/src/locales/de/description.ts @@ -62,6 +62,9 @@ const description = { no_region_code_found: 'Kein Regionencode gefunden', verify_email: 'Bestätige deine E-Mail-Adresse', verify_phone: 'Bestätige deine Telefonnummer', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/de/error.ts b/packages/phrases-ui/src/locales/de/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/de/error.ts rename to packages/phrases-ui/src/locales/de/error/index.ts index e4239166d..d2657a2c6 100644 --- a/packages/phrases-ui/src/locales/de/error.ts +++ b/packages/phrases-ui/src/locales/de/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}} ist erforderlich', general_invalid: 'Die {{types, list(type: disjunction;)}} is ungültig', @@ -18,6 +20,7 @@ const error = { unknown: 'Unbekannter Fehler. Versuche es später noch einmal.', invalid_session: 'Die Sitzung ist ungültig. Bitte melde dich erneut an.', timeout: 'Zeitüberschreitung. Bitte melde dich erneut an.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/de/error/password-rejected.ts b/packages/phrases-ui/src/locales/de/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/de/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/de/index.ts b/packages/phrases-ui/src/locales/de/index.ts index 35e57ef19..876e28f70 100644 --- a/packages/phrases-ui/src/locales/de/index.ts +++ b/packages/phrases-ui/src/locales/de/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const de = { @@ -15,6 +16,7 @@ const de = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/de/list.ts b/packages/phrases-ui/src/locales/de/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/de/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/en/description.ts b/packages/phrases-ui/src/locales/en/description.ts index 876cea86c..324e5cacb 100644 --- a/packages/phrases-ui/src/locales/en/description.ts +++ b/packages/phrases-ui/src/locales/en/description.ts @@ -59,6 +59,9 @@ const description = { no_region_code_found: 'No region code found', verify_email: 'Verify your email', verify_phone: 'Verify your phone number', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/en/error.ts b/packages/phrases-ui/src/locales/en/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/en/error.ts rename to packages/phrases-ui/src/locales/en/error/index.ts index b8c8c8336..9ba12f196 100644 --- a/packages/phrases-ui/src/locales/en/error.ts +++ b/packages/phrases-ui/src/locales/en/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: `{{types, list(type: disjunction;)}} is required`, general_invalid: `The {{types, list(type: disjunction;)}} is invalid`, @@ -18,6 +20,7 @@ const error = { unknown: 'Unknown error. Please try again later.', invalid_session: 'Session not found. Please go back and sign in again.', timeout: 'Request timeout. Please try again later.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/en/error/password-rejected.ts b/packages/phrases-ui/src/locales/en/error/password-rejected.ts new file mode 100644 index 000000000..d7311965a --- /dev/null +++ b/packages/phrases-ui/src/locales/en/error/password-rejected.ts @@ -0,0 +1,19 @@ +import { type PasswordRejectionCode } from '@logto/core-kit'; + +const password_rejected = { + too_short: 'Minimum length is {{min}}.', + too_long: 'Maximum length is {{max}}.', + character_types: 'At least {{min}} types of characters are required.', + unsupported_characters: 'Unsupported character found.', + pwned: 'Avoid using simple passwords that are easy to guess.', + restricted_found: 'Avoid overusing {{list}}.', + 'restricted.repetition': 'repeated characters', + 'restricted.sequence': 'sequential characters', + 'restricted.personal_info': 'your personal information', + 'restricted.words': 'product context', +} satisfies Record & { + // Use for displaying a list of restricted issues + restricted_found: string; +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/en/index.ts b/packages/phrases-ui/src/locales/en/index.ts index 4396b682c..d3a51215f 100644 --- a/packages/phrases-ui/src/locales/en/index.ts +++ b/packages/phrases-ui/src/locales/en/index.ts @@ -1,8 +1,9 @@ import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const en = { @@ -13,6 +14,7 @@ const en = { description, error, demo_app, + list, }, }; diff --git a/packages/phrases-ui/src/locales/en/list.ts b/packages/phrases-ui/src/locales/en/list.ts new file mode 100644 index 000000000..880b2a789 --- /dev/null +++ b/packages/phrases-ui/src/locales/en/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', + and: ' and ', + separator: ', ', +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/es/description.ts b/packages/phrases-ui/src/locales/es/description.ts index 8ece54254..69177d0e6 100644 --- a/packages/phrases-ui/src/locales/es/description.ts +++ b/packages/phrases-ui/src/locales/es/description.ts @@ -60,6 +60,9 @@ const description = { no_region_code_found: 'No se encontró código de región', verify_email: 'Verificar su correo electrónico', verify_phone: 'Verificar su número de teléfono', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/es/error.ts b/packages/phrases-ui/src/locales/es/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/es/error.ts rename to packages/phrases-ui/src/locales/es/error/index.ts index 0e5ecb751..fc2b32d64 100644 --- a/packages/phrases-ui/src/locales/es/error.ts +++ b/packages/phrases-ui/src/locales/es/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: `{{types, list(type: disjunction;)}} es requerido`, general_invalid: `El/La {{types, list(type: disjunction;)}} no es válido`, @@ -19,6 +21,7 @@ const error = { unknown: 'Error desconocido. Por favor intente de nuevo más tarde.', invalid_session: 'No se encontró la sesión. Por favor regrese e inicie sesión nuevamente.', timeout: 'Tiempo de espera de solicitud agotado. Por favor intente de nuevo más tarde.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/es/error/password-rejected.ts b/packages/phrases-ui/src/locales/es/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/es/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/es/index.ts b/packages/phrases-ui/src/locales/es/index.ts index da23da3ac..a04897c9b 100644 --- a/packages/phrases-ui/src/locales/es/index.ts +++ b/packages/phrases-ui/src/locales/es/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const es = { @@ -15,6 +16,7 @@ const es = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/es/list.ts b/packages/phrases-ui/src/locales/es/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/es/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/fr/description.ts b/packages/phrases-ui/src/locales/fr/description.ts index d6bc014a5..e633e5459 100644 --- a/packages/phrases-ui/src/locales/fr/description.ts +++ b/packages/phrases-ui/src/locales/fr/description.ts @@ -62,6 +62,9 @@ const description = { no_region_code_found: 'Aucun code de région trouvé', verify_email: 'Vérifiez votre e-mail', verify_phone: 'Vérifiez votre numéro de téléphone', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/fr/error.ts b/packages/phrases-ui/src/locales/fr/error/index.ts similarity index 95% rename from packages/phrases-ui/src/locales/fr/error.ts rename to packages/phrases-ui/src/locales/fr/error/index.ts index 60aad4581..574d73889 100644 --- a/packages/phrases-ui/src/locales/fr/error.ts +++ b/packages/phrases-ui/src/locales/fr/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: 'Le {{types, list(type: disjunction;)}} est requis', general_invalid: "Le {{types, list(type: disjunction;)}} n'est pas valide", @@ -20,6 +22,7 @@ const error = { unknown: 'Erreur inconnue. Veuillez réessayer plus tard.', invalid_session: 'Session non trouvée. Veuillez revenir en arrière et vous connecter à nouveau.', timeout: "Délai d'attente de la requête dépassé. Veuillez réessayer plus tard.", + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/fr/error/password-rejected.ts b/packages/phrases-ui/src/locales/fr/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/fr/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/fr/index.ts b/packages/phrases-ui/src/locales/fr/index.ts index 0a3583322..686330b66 100644 --- a/packages/phrases-ui/src/locales/fr/index.ts +++ b/packages/phrases-ui/src/locales/fr/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const fr = { @@ -15,6 +16,7 @@ const fr = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/fr/list.ts b/packages/phrases-ui/src/locales/fr/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/fr/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/it/description.ts b/packages/phrases-ui/src/locales/it/description.ts index c9163c772..000f481b2 100644 --- a/packages/phrases-ui/src/locales/it/description.ts +++ b/packages/phrases-ui/src/locales/it/description.ts @@ -58,6 +58,9 @@ const description = { no_region_code_found: 'Nessun codice di regione trovato', verify_email: 'Verifica la tua email', verify_phone: 'Verifica il tuo numero di telefono', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/it/error.ts b/packages/phrases-ui/src/locales/it/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/it/error.ts rename to packages/phrases-ui/src/locales/it/error/index.ts index 9b2bf1f21..3d91c976f 100644 --- a/packages/phrases-ui/src/locales/it/error.ts +++ b/packages/phrases-ui/src/locales/it/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: `{{types, list(type: disjunction;)}} è richiesto`, general_invalid: `Il {{types, list(type: disjunction;)}} non è valido`, @@ -18,6 +20,7 @@ const error = { unknown: 'Errore sconosciuto. Si prega di riprovare più tardi.', invalid_session: 'Sessione non trovata. Si prega di tornare indietro e accedere di nuovo.', timeout: 'Timeout della richiesta. Si prega di riprovare più tardi.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/it/error/password-rejected.ts b/packages/phrases-ui/src/locales/it/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/it/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/it/index.ts b/packages/phrases-ui/src/locales/it/index.ts index 505ce7328..fa37f79f2 100644 --- a/packages/phrases-ui/src/locales/it/index.ts +++ b/packages/phrases-ui/src/locales/it/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const it = { @@ -15,6 +16,7 @@ const it = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/it/list.ts b/packages/phrases-ui/src/locales/it/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/it/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/ja/description.ts b/packages/phrases-ui/src/locales/ja/description.ts index 26d22d1b3..b0892fc49 100644 --- a/packages/phrases-ui/src/locales/ja/description.ts +++ b/packages/phrases-ui/src/locales/ja/description.ts @@ -60,6 +60,9 @@ const description = { no_region_code_found: '地域コードが見つかりません', verify_email: 'Eメールを確認する', verify_phone: '電話番号を確認する', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/ja/error.ts b/packages/phrases-ui/src/locales/ja/error/index.ts similarity index 95% rename from packages/phrases-ui/src/locales/ja/error.ts rename to packages/phrases-ui/src/locales/ja/error/index.ts index 7f752e2c0..f9445b267 100644 --- a/packages/phrases-ui/src/locales/ja/error.ts +++ b/packages/phrases-ui/src/locales/ja/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: `{{types, list(type: disjunction;)}}が必要です`, general_invalid: `{{types, list(type: disjunction;)}}が無効です`, @@ -19,6 +21,7 @@ const error = { unknown: '不明なエラーが発生しました。後でもう一度お試しください。', invalid_session: 'セッションが見つかりません。もう一度サインインしてください。', timeout: 'リクエストタイムアウト。後でもう一度お試しください。', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/ja/error/password-rejected.ts b/packages/phrases-ui/src/locales/ja/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/ja/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/ja/index.ts b/packages/phrases-ui/src/locales/ja/index.ts index 3b00c0b1f..e00afece6 100644 --- a/packages/phrases-ui/src/locales/ja/index.ts +++ b/packages/phrases-ui/src/locales/ja/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const ja = { @@ -15,6 +16,7 @@ const ja = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/ja/list.ts b/packages/phrases-ui/src/locales/ja/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/ja/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/ko/description.ts b/packages/phrases-ui/src/locales/ko/description.ts index 4231c65cf..49368bce3 100644 --- a/packages/phrases-ui/src/locales/ko/description.ts +++ b/packages/phrases-ui/src/locales/ko/description.ts @@ -55,6 +55,9 @@ const description = { no_region_code_found: '지역 코드를 찾을 수 없습니다.', verify_email: '이메일 인증', verify_phone: '휴대전화번호 인증', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/ko/error.ts b/packages/phrases-ui/src/locales/ko/error/index.ts similarity index 95% rename from packages/phrases-ui/src/locales/ko/error.ts rename to packages/phrases-ui/src/locales/ko/error/index.ts index 4e3485ed4..72b123477 100644 --- a/packages/phrases-ui/src/locales/ko/error.ts +++ b/packages/phrases-ui/src/locales/ko/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}}은/는 필수예요.', general_invalid: '{{types, list(type: disjunction;)}}은/는 유효하지 않아요.', @@ -17,6 +19,7 @@ const error = { unknown: '알 수 없는 오류가 발생했어요. 잠시 후에 시도해 주세요.', invalid_session: '세션을 찾을 수 없어요. 다시 로그인해 주세요.', timeout: '요청 시간이 초과되었어요. 잠시 후에 다시 시도해 주세요.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/ko/error/password-rejected.ts b/packages/phrases-ui/src/locales/ko/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/ko/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/ko/index.ts b/packages/phrases-ui/src/locales/ko/index.ts index 0c4a5fb5b..10f510b5c 100644 --- a/packages/phrases-ui/src/locales/ko/index.ts +++ b/packages/phrases-ui/src/locales/ko/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const ko = { @@ -15,6 +16,7 @@ const ko = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/ko/list.ts b/packages/phrases-ui/src/locales/ko/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/ko/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/pl-pl/description.ts b/packages/phrases-ui/src/locales/pl-pl/description.ts index 95aae97ac..6f5dc5f42 100644 --- a/packages/phrases-ui/src/locales/pl-pl/description.ts +++ b/packages/phrases-ui/src/locales/pl-pl/description.ts @@ -58,6 +58,9 @@ const description = { no_region_code_found: 'Nie znaleziono kodu regionu', verify_email: 'Potwierdź swój email', verify_phone: 'Potwierdź swój numer telefonu', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/pl-pl/error.ts b/packages/phrases-ui/src/locales/pl-pl/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/pl-pl/error.ts rename to packages/phrases-ui/src/locales/pl-pl/error/index.ts index 7c986ae2c..fe7dc0f12 100644 --- a/packages/phrases-ui/src/locales/pl-pl/error.ts +++ b/packages/phrases-ui/src/locales/pl-pl/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: `{{types, list(type: disjunction;)}} jest wymagany`, general_invalid: `{{types, list(type: disjunction;)}} jest nieprawidłowe`, @@ -19,6 +21,7 @@ const error = { unknown: 'Nieznany błąd. Proszę spróbuj ponownie później.', invalid_session: 'Sesja nie znaleziona. Proszę wróć i zaloguj się ponownie.', timeout: 'Czas żądania upłynął. Proszę spróbuj ponownie później.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/pl-pl/error/password-rejected.ts b/packages/phrases-ui/src/locales/pl-pl/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/pl-pl/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/pl-pl/index.ts b/packages/phrases-ui/src/locales/pl-pl/index.ts index 8e847eeee..e44fa5faa 100644 --- a/packages/phrases-ui/src/locales/pl-pl/index.ts +++ b/packages/phrases-ui/src/locales/pl-pl/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const pl_pl = { @@ -15,6 +16,7 @@ const pl_pl = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/pl-pl/list.ts b/packages/phrases-ui/src/locales/pl-pl/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/pl-pl/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/pt-br/description.ts b/packages/phrases-ui/src/locales/pt-br/description.ts index 39d5f9d50..c3fe11e6e 100644 --- a/packages/phrases-ui/src/locales/pt-br/description.ts +++ b/packages/phrases-ui/src/locales/pt-br/description.ts @@ -57,6 +57,9 @@ const description = { no_region_code_found: 'Não foi possível encontrar o código de região do seu telefone.', verify_email: 'Verificar e-mail', verify_phone: 'Verificar número de telefone', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/pt-br/error.ts b/packages/phrases-ui/src/locales/pt-br/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/pt-br/error.ts rename to packages/phrases-ui/src/locales/pt-br/error/index.ts index 8d503dea4..b389b997e 100644 --- a/packages/phrases-ui/src/locales/pt-br/error.ts +++ b/packages/phrases-ui/src/locales/pt-br/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}} é obrigatório', general_invalid: 'O {{types, list(type: disjunction;)}} é inválido', @@ -18,6 +20,7 @@ const error = { unknown: 'Erro desconhecido. Por favor, tente novamente mais tarde.', invalid_session: 'Sessão não encontrada. Volte e faça login novamente.', timeout: 'Tempo limite excedido. Por favor, tente novamente mais tarde.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/pt-br/error/password-rejected.ts b/packages/phrases-ui/src/locales/pt-br/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/pt-br/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/pt-br/index.ts b/packages/phrases-ui/src/locales/pt-br/index.ts index 2bddcefbd..acdfee044 100644 --- a/packages/phrases-ui/src/locales/pt-br/index.ts +++ b/packages/phrases-ui/src/locales/pt-br/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const pt_br = { @@ -15,6 +16,7 @@ const pt_br = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/pt-br/list.ts b/packages/phrases-ui/src/locales/pt-br/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/pt-br/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/pt-pt/description.ts b/packages/phrases-ui/src/locales/pt-pt/description.ts index 3b9e8ecf5..5f6499e5c 100644 --- a/packages/phrases-ui/src/locales/pt-pt/description.ts +++ b/packages/phrases-ui/src/locales/pt-pt/description.ts @@ -57,6 +57,9 @@ const description = { no_region_code_found: 'Não foi possível encontrar o código de região do seu telefone.', verify_email: 'Verifique o seu email', verify_phone: 'Verifique o seu número de telefone', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/pt-pt/error.ts b/packages/phrases-ui/src/locales/pt-pt/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/pt-pt/error.ts rename to packages/phrases-ui/src/locales/pt-pt/error/index.ts index f0ed7c931..decbf3ea2 100644 --- a/packages/phrases-ui/src/locales/pt-pt/error.ts +++ b/packages/phrases-ui/src/locales/pt-pt/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}} is necessário', general_invalid: 'O {{types, list(type: disjunction;)}} é inválido', @@ -19,6 +21,7 @@ const error = { unknown: 'Erro desconhecido. Por favor, tente novamente mais tarde.', invalid_session: 'Sessão não encontrada. Volte e faça login novamente.', timeout: 'Tempo limite de sessão. Volte e faça login novamente.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/pt-pt/error/password-rejected.ts b/packages/phrases-ui/src/locales/pt-pt/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/pt-pt/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/pt-pt/index.ts b/packages/phrases-ui/src/locales/pt-pt/index.ts index b053cb10f..21a98a2c3 100644 --- a/packages/phrases-ui/src/locales/pt-pt/index.ts +++ b/packages/phrases-ui/src/locales/pt-pt/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const pt_pt = { @@ -15,6 +16,7 @@ const pt_pt = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/pt-pt/list.ts b/packages/phrases-ui/src/locales/pt-pt/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/pt-pt/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/ru/description.ts b/packages/phrases-ui/src/locales/ru/description.ts index 218bea0cb..172a6797e 100644 --- a/packages/phrases-ui/src/locales/ru/description.ts +++ b/packages/phrases-ui/src/locales/ru/description.ts @@ -61,6 +61,9 @@ const description = { no_region_code_found: 'Не удалось определить код региона', verify_email: 'Подтвердите Ваш электронный адрес', verify_phone: 'Подтвердите свой номер телефона', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/ru/error.ts b/packages/phrases-ui/src/locales/ru/error/index.ts similarity index 96% rename from packages/phrases-ui/src/locales/ru/error.ts rename to packages/phrases-ui/src/locales/ru/error/index.ts index 6a2215ae5..71f594da2 100644 --- a/packages/phrases-ui/src/locales/ru/error.ts +++ b/packages/phrases-ui/src/locales/ru/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: 'Введите {{types, list(type: disjunction;)}}', general_invalid: 'Проверьте {{types, list(type: disjunction;)}}', @@ -19,6 +21,7 @@ const error = { unknown: 'Неизвестная ошибка. Пожалуйста, повторите попытку позднее.', invalid_session: 'Сессия не найдена. Пожалуйста, войдите снова.', timeout: 'Время ожидания истекло. Пожалуйста, повторите попытку позднее.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/ru/error/password-rejected.ts b/packages/phrases-ui/src/locales/ru/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/ru/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/ru/index.ts b/packages/phrases-ui/src/locales/ru/index.ts index 6bc2ec80c..3ec9038ed 100644 --- a/packages/phrases-ui/src/locales/ru/index.ts +++ b/packages/phrases-ui/src/locales/ru/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const ru = { @@ -15,6 +16,7 @@ const ru = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/ru/list.ts b/packages/phrases-ui/src/locales/ru/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/ru/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/tr-tr/description.ts b/packages/phrases-ui/src/locales/tr-tr/description.ts index 25d2e1b2c..db80d9176 100644 --- a/packages/phrases-ui/src/locales/tr-tr/description.ts +++ b/packages/phrases-ui/src/locales/tr-tr/description.ts @@ -58,6 +58,9 @@ const description = { no_region_code_found: 'Bölge kodu bulunamadı', verify_email: 'E-postanızın doğrulanması', verify_phone: 'Telefon numaranızın doğrulanması', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/tr-tr/error.ts b/packages/phrases-ui/src/locales/tr-tr/error/index.ts similarity index 94% rename from packages/phrases-ui/src/locales/tr-tr/error.ts rename to packages/phrases-ui/src/locales/tr-tr/error/index.ts index c754c368d..4342b1143 100644 --- a/packages/phrases-ui/src/locales/tr-tr/error.ts +++ b/packages/phrases-ui/src/locales/tr-tr/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}} gerekli', general_invalid: '{{types, list(type: disjunction;)}} geçersiz', @@ -18,6 +20,7 @@ const error = { unknown: 'Bilinmeyen hata. Lütfen daha sonra tekrar deneyiniz.', invalid_session: 'Oturum bulunamadı. Lütfen geri dönüp tekrar giriş yapınız.', timeout: 'Oturum zaman aşımına uğradı. Lütfen geri dönüp tekrar giriş yapınız.', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/tr-tr/error/password-rejected.ts b/packages/phrases-ui/src/locales/tr-tr/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/tr-tr/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/tr-tr/index.ts b/packages/phrases-ui/src/locales/tr-tr/index.ts index 06c28cbfa..de55482e1 100644 --- a/packages/phrases-ui/src/locales/tr-tr/index.ts +++ b/packages/phrases-ui/src/locales/tr-tr/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const tr_tr = { @@ -15,6 +16,7 @@ const tr_tr = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/tr-tr/list.ts b/packages/phrases-ui/src/locales/tr-tr/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/tr-tr/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/zh-cn/description.ts b/packages/phrases-ui/src/locales/zh-cn/description.ts index 6bdf92313..e064f9da5 100644 --- a/packages/phrases-ui/src/locales/zh-cn/description.ts +++ b/packages/phrases-ui/src/locales/zh-cn/description.ts @@ -51,6 +51,9 @@ const description = { no_region_code_found: '没有找到区域码', verify_email: '验证你的邮箱', verify_phone: '验证你的手机号', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/zh-cn/error.ts b/packages/phrases-ui/src/locales/zh-cn/error/index.ts similarity index 93% rename from packages/phrases-ui/src/locales/zh-cn/error.ts rename to packages/phrases-ui/src/locales/zh-cn/error/index.ts index 67435d059..345caab5b 100644 --- a/packages/phrases-ui/src/locales/zh-cn/error.ts +++ b/packages/phrases-ui/src/locales/zh-cn/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}}必填', general_invalid: '无效的{{types, list(type: disjunction;)}}', @@ -17,6 +19,7 @@ const error = { unknown: '未知错误,请稍后重试。', invalid_session: '未找到会话,请返回并重新登录。', timeout: '请求超时,请稍后重试。', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/zh-cn/error/password-rejected.ts b/packages/phrases-ui/src/locales/zh-cn/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/zh-cn/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/zh-cn/index.ts b/packages/phrases-ui/src/locales/zh-cn/index.ts index 42fefc867..0da5ea6f2 100644 --- a/packages/phrases-ui/src/locales/zh-cn/index.ts +++ b/packages/phrases-ui/src/locales/zh-cn/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const zh_cn = { @@ -15,6 +16,7 @@ const zh_cn = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/zh-cn/list.ts b/packages/phrases-ui/src/locales/zh-cn/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/zh-cn/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/zh-hk/description.ts b/packages/phrases-ui/src/locales/zh-hk/description.ts index 23a5b7c5a..908e61d0a 100644 --- a/packages/phrases-ui/src/locales/zh-hk/description.ts +++ b/packages/phrases-ui/src/locales/zh-hk/description.ts @@ -51,6 +51,9 @@ const description = { no_region_code_found: '沒有找到區域碼', verify_email: '驗證你的郵箱', verify_phone: '驗證你的手機號', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/zh-hk/error.ts b/packages/phrases-ui/src/locales/zh-hk/error/index.ts similarity index 93% rename from packages/phrases-ui/src/locales/zh-hk/error.ts rename to packages/phrases-ui/src/locales/zh-hk/error/index.ts index 1681854d3..aa1ebee7e 100644 --- a/packages/phrases-ui/src/locales/zh-hk/error.ts +++ b/packages/phrases-ui/src/locales/zh-hk/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}}必填', general_invalid: '無效的{{types, list(type: disjunction;)}}', @@ -17,6 +19,7 @@ const error = { unknown: '未知錯誤,請稍後重試。', invalid_session: '未找到會話,請返回並重新登錄。', timeout: '請求超時,請稍後重試。', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/zh-hk/error/password-rejected.ts b/packages/phrases-ui/src/locales/zh-hk/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/zh-hk/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/zh-hk/index.ts b/packages/phrases-ui/src/locales/zh-hk/index.ts index 285adeb34..6036f0f02 100644 --- a/packages/phrases-ui/src/locales/zh-hk/index.ts +++ b/packages/phrases-ui/src/locales/zh-hk/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const zh_hk = { @@ -15,6 +16,7 @@ const zh_hk = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/zh-hk/list.ts b/packages/phrases-ui/src/locales/zh-hk/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/zh-hk/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/phrases-ui/src/locales/zh-tw/description.ts b/packages/phrases-ui/src/locales/zh-tw/description.ts index 4d5edd7f0..1ea2c0392 100644 --- a/packages/phrases-ui/src/locales/zh-tw/description.ts +++ b/packages/phrases-ui/src/locales/zh-tw/description.ts @@ -51,6 +51,9 @@ const description = { no_region_code_found: '沒有找到區域碼', verify_email: '驗證你的郵箱', verify_phone: '驗證你的手機號碼', + password_requirements_with_type_one: 'Password requires a minimum of {{min}} characters.', // UNTRANSLATED + password_requirements_with_type_other: + 'Password requires a minimum of {{min}} characters, and contains {{count}} of the following: uppercase letters (A-Z), lowercase letters (a-z), digits (0-9), and symbols.', // UNTRANSLATED }; export default Object.freeze(description); diff --git a/packages/phrases-ui/src/locales/zh-tw/error.ts b/packages/phrases-ui/src/locales/zh-tw/error/index.ts similarity index 93% rename from packages/phrases-ui/src/locales/zh-tw/error.ts rename to packages/phrases-ui/src/locales/zh-tw/error/index.ts index c285902e4..3583f10e5 100644 --- a/packages/phrases-ui/src/locales/zh-tw/error.ts +++ b/packages/phrases-ui/src/locales/zh-tw/error/index.ts @@ -1,3 +1,5 @@ +import password_rejected from './password-rejected.js'; + const error = { general_required: '{{types, list(type: disjunction;)}}必填', general_invalid: '無效的{{types, list(type: disjunction;)}}', @@ -17,6 +19,7 @@ const error = { unknown: '未知錯誤,請稍後重試。', invalid_session: '未找到會話,請返回並重新登錄。', timeout: '請求超時,請稍後重試。', + password_rejected, }; export default Object.freeze(error); diff --git a/packages/phrases-ui/src/locales/zh-tw/error/password-rejected.ts b/packages/phrases-ui/src/locales/zh-tw/error/password-rejected.ts new file mode 100644 index 000000000..fad0b338a --- /dev/null +++ b/packages/phrases-ui/src/locales/zh-tw/error/password-rejected.ts @@ -0,0 +1,14 @@ +const password_rejected = { + too_short: 'Minimum length is {{min}}.', // UNTRANSLATED + too_long: 'Maximum length is {{max}}.', // UNTRANSLATED + character_types: 'At least {{min}} types of characters are required.', // UNTRANSLATED + unsupported_characters: 'Unsupported character found.', // UNTRANSLATED + pwned: 'Avoid using simple passwords that are easy to guess.', // UNTRANSLATED + restricted_found: 'Avoid overusing {{list}}.', // UNTRANSLATED + 'restricted.repetition': 'repeated characters', // UNTRANSLATED + 'restricted.sequence': 'sequential characters', // UNTRANSLATED + 'restricted.personal_info': 'your personal information', // UNTRANSLATED + 'restricted.words': 'product context', // UNTRANSLATED +}; + +export default Object.freeze(password_rejected); diff --git a/packages/phrases-ui/src/locales/zh-tw/index.ts b/packages/phrases-ui/src/locales/zh-tw/index.ts index eaf0c5f13..113ee1928 100644 --- a/packages/phrases-ui/src/locales/zh-tw/index.ts +++ b/packages/phrases-ui/src/locales/zh-tw/index.ts @@ -3,8 +3,9 @@ import type { LocalePhrase } from '../../types.js'; import action from './action.js'; import demo_app from './demo-app.js'; import description from './description.js'; -import error from './error.js'; +import error from './error/index.js'; import input from './input.js'; +import list from './list.js'; import secondary from './secondary.js'; const zh_tw = { @@ -15,6 +16,7 @@ const zh_tw = { description, error, demo_app, + list, }, } satisfies LocalePhrase; diff --git a/packages/phrases-ui/src/locales/zh-tw/list.ts b/packages/phrases-ui/src/locales/zh-tw/list.ts new file mode 100644 index 000000000..f595571dd --- /dev/null +++ b/packages/phrases-ui/src/locales/zh-tw/list.ts @@ -0,0 +1,7 @@ +const list = { + or: ' or ', // UNTRANSLATED + and: ' and ', // UNTRANSLATED + separator: ', ', // UNTRANSLATED +}; + +export default Object.freeze(list); diff --git a/packages/schemas/src/types/interactions.ts b/packages/schemas/src/types/interactions.ts index c5e356f9c..bce1a5c45 100644 --- a/packages/schemas/src/types/interactions.ts +++ b/packages/schemas/src/types/interactions.ts @@ -1,4 +1,4 @@ -import { emailRegEx, phoneRegEx, usernameRegEx, passwordRegEx } from '@logto/core-kit'; +import { emailRegEx, phoneRegEx, usernameRegEx } from '@logto/core-kit'; import { z } from 'zod'; import { jsonObjectGuard } from '../foundations/index.js'; @@ -91,7 +91,7 @@ export const profileGuard = z.object({ email: z.string().regex(emailRegEx).optional(), phone: z.string().regex(phoneRegEx).optional(), connectorId: z.string().optional(), - password: z.string().regex(passwordRegEx).optional(), + password: z.string().optional(), }); export type Profile = z.infer; diff --git a/packages/toolkit/core-kit/src/password-policy.test.ts b/packages/toolkit/core-kit/src/password-policy.test.ts index 041062266..f4c88aaf7 100644 --- a/packages/toolkit/core-kit/src/password-policy.test.ts +++ b/packages/toolkit/core-kit/src/password-policy.test.ts @@ -52,23 +52,23 @@ describe('PasswordPolicyChecker -> check()', () => { expect(await checker.check('aaa', {})).toEqual([ { code: 'password_rejected.too_short', interpolation: { min: 7 } }, { code: 'password_rejected.character_types', interpolation: { min: 2 } }, - { code: 'password_rejected.repetition' }, + { code: 'password_rejected.restricted.repetition' }, ]); expect(await checker.check('123456', { phoneNumber: '12345' })).toEqual([ { code: 'password_rejected.too_short', interpolation: { min: 7 } }, { code: 'password_rejected.character_types', interpolation: { min: 2 } }, - { code: 'password_rejected.sequence' }, { code: 'password_rejected.pwned' }, - { code: 'password_rejected.personal_info' }, + { code: 'password_rejected.restricted.sequence' }, + { code: 'password_rejected.restricted.personal_info' }, ]); expect(await checker.check('aaaaaatest😀', {})).toEqual([ { code: 'password_rejected.too_long', interpolation: { max: 8 } }, { code: 'password_rejected.unsupported_characters' }, - { code: 'password_rejected.repetition' }, + { code: 'password_rejected.restricted.repetition' }, { - code: 'password_rejected.restricted_words', + code: 'password_rejected.restricted.words', interpolation: { words: 'test\naaaa', count: 2 }, }, ]); diff --git a/packages/toolkit/core-kit/src/password-policy.ts b/packages/toolkit/core-kit/src/password-policy.ts index e1f17560f..cb0a6dcc4 100644 --- a/packages/toolkit/core-kit/src/password-policy.ts +++ b/packages/toolkit/core-kit/src/password-policy.ts @@ -65,15 +65,15 @@ export type PasswordRejectionCode = | 'character_types' | 'unsupported_characters' | 'pwned' - | 'repetition' - | 'sequence' - | 'personal_info' - | 'restricted_words'; + | 'restricted.repetition' + | 'restricted.sequence' + | 'restricted.personal_info' + | 'restricted.words'; /** A password issue that does not meet the policy. */ -export type PasswordIssue = { +export type PasswordIssue = { /** Issue code. */ - code: `password_rejected.${PasswordRejectionCode}`; + code: `password_rejected.${Code}`; /** Interpolation data for the issue message. */ interpolation?: Record; }; @@ -104,7 +104,7 @@ export type PersonalInfo = Partial<{ * // { code: 'password_rejected.too_short' }, * // { code: 'password_rejected.character_types', interpolation: { min: 2 } }, * // { code: 'password_rejected.pwned' }, - * // { code: 'password_rejected.sequence' }, + * // { code: 'password_rejected.restricted.sequence' }, * // ] * ``` */ @@ -142,6 +142,29 @@ export class PasswordPolicyChecker { }); } + if (this.policy.rejects.repetitionAndSequence) { + if (this.hasRepetition(password)) { + issues.push({ + code: 'password_rejected.restricted.repetition', + }); + } + + if (this.hasSequentialChars(password)) { + issues.push({ + code: 'password_rejected.restricted.sequence', + }); + } + } + + const words = this.hasWords(password); + + if (words.length > 0) { + issues.push({ + code: 'password_rejected.restricted.words', + interpolation: { words: words.join('\n'), count: words.length }, + }); + } + if (this.policy.rejects.personalInfo) { if (!personalInfo) { throw new TypeError('Personal information is required to check personal information.'); @@ -149,7 +172,7 @@ export class PasswordPolicyChecker { if (this.hasPersonalInfo(password, personalInfo)) { issues.push({ - code: 'password_rejected.personal_info', + code: 'password_rejected.restricted.personal_info', }); } } @@ -159,7 +182,7 @@ export class PasswordPolicyChecker { /** * Perform a fast check to see if the password passes the basic requirements. - * No pwned password and personal information check will be performed. + * Only the length and character types will be checked. * * This method is used for frontend validation. * @@ -193,29 +216,6 @@ export class PasswordPolicyChecker { }); } - if (this.policy.rejects.repetitionAndSequence) { - if (this.hasRepetition(password)) { - issues.push({ - code: 'password_rejected.repetition', - }); - } - - if (this.hasSequentialChars(password)) { - issues.push({ - code: 'password_rejected.sequence', - }); - } - } - - const words = this.hasWords(password); - - if (words.length > 0) { - issues.push({ - code: 'password_rejected.restricted_words', - interpolation: { words: words.join('\n'), count: words.length }, - }); - } - return issues; } /* eslint-enable @silverhand/fp/no-mutating-methods */ diff --git a/packages/ui/src/components/ErrorMessage/index.module.scss b/packages/ui/src/components/ErrorMessage/index.module.scss index fa6829667..fe8b54d0a 100644 --- a/packages/ui/src/components/ErrorMessage/index.module.scss +++ b/packages/ui/src/components/ErrorMessage/index.module.scss @@ -3,4 +3,8 @@ .error { font: var(--font-body-2); color: var(--color-danger-default); + + ul { + padding-inline-start: 1rem; + } } diff --git a/packages/ui/src/components/ErrorMessage/index.tsx b/packages/ui/src/components/ErrorMessage/index.tsx index 3ee16a6c4..e777e3e5f 100644 --- a/packages/ui/src/components/ErrorMessage/index.tsx +++ b/packages/ui/src/components/ErrorMessage/index.tsx @@ -5,7 +5,14 @@ import { useTranslation } from 'react-i18next'; import * as styles from './index.module.scss'; -type ErrorCode = TFuncKey<'translation', 'error'>; +type RemovePrefix = T extends `${Prefix}${string}` ? never : T; + +/** + * All error codes that can be passed to ErrorMessage. + * Nested keys are removed since they will result in a non-string return value. They + * can be processed manually. + */ +type ErrorCode = RemovePrefix, 'password_rejected'>; export type ErrorType = ErrorCode | { code: ErrorCode; data?: Record }; diff --git a/packages/ui/src/components/InputFields/InputField/index.tsx b/packages/ui/src/components/InputFields/InputField/index.tsx index 295cf5305..b26b9e15d 100644 --- a/packages/ui/src/components/InputFields/InputField/index.tsx +++ b/packages/ui/src/components/InputFields/InputField/index.tsx @@ -28,25 +28,40 @@ const InputField = ( ...props }: Props, reference: ForwardedRef -) => ( -
-
- {prefix} - - {suffix && - cloneElement(suffix, { - className: classNames([suffix.props.className, styles.suffix]), - })} -
- {errorMessage && {errorMessage}} -
-); +) => { + const errorMessages = errorMessage?.split('\n'); + return ( +
+
+ {prefix} + + {suffix && + cloneElement(suffix, { + className: classNames([suffix.props.className, styles.suffix]), + })} +
+ {errorMessages && ( + + {errorMessages.length > 1 ? ( +
    + {errorMessages.map((message) => ( +
  • {message}
  • + ))} +
+ ) : ( + errorMessages[0] + )} +
+ )} +
+ ); +}; export default forwardRef(InputField); diff --git a/packages/ui/src/containers/SetPassword/Lite.test.tsx b/packages/ui/src/containers/SetPassword/Lite.test.tsx index 266309d99..cfa0715b2 100644 --- a/packages/ui/src/containers/SetPassword/Lite.test.tsx +++ b/packages/ui/src/containers/SetPassword/Lite.test.tsx @@ -51,7 +51,9 @@ describe('', () => { }); await waitFor(() => { - expect(queryByText('error.password_min_length')).not.toBeNull(); + expect(queryByText('error.password_rejected.too_short')).not.toBeNull(); + expect(queryByText('error.password_rejected.character_types')).not.toBeNull(); + expect(queryByText('error.password_rejected.sequence')).not.toBeNull(); }); expect(submit).not.toBeCalled(); @@ -65,7 +67,9 @@ describe('', () => { }); await waitFor(() => { - expect(queryByText('error.password_min_length')).toBeNull(); + expect(queryByText('error.password_rejected.too_short')).toBeNull(); + expect(queryByText('error.password_rejected.character_types')).toBeNull(); + expect(queryByText('error.password_rejected.sequence')).toBeNull(); }); }); diff --git a/packages/ui/src/containers/SetPassword/Lite.tsx b/packages/ui/src/containers/SetPassword/Lite.tsx index 4cafbfbb5..b7dfba50e 100644 --- a/packages/ui/src/containers/SetPassword/Lite.tsx +++ b/packages/ui/src/containers/SetPassword/Lite.tsx @@ -6,7 +6,6 @@ import { useTranslation } from 'react-i18next'; import Button from '@/components/Button'; import ErrorMessage from '@/components/ErrorMessage'; import { PasswordInputField } from '@/components/InputFields'; -import { validatePassword } from '@/utils/form'; import * as styles from './index.module.scss'; @@ -37,6 +36,7 @@ const Lite = ({ className, autoFocus, onSubmit, errorMessage, clearErrorMessage useEffect(() => { if (!isValid) { + console.log('!isValid'); clearErrorMessage?.(); } }, [clearErrorMessage, isValid]); @@ -64,17 +64,6 @@ const Lite = ({ className, autoFocus, onSubmit, errorMessage, clearErrorMessage aria-invalid={!!errors.newPassword} {...register('newPassword', { required: t('error.password_required'), - validate: (password) => { - const errorMessage = validatePassword(password); - - if (errorMessage) { - return typeof errorMessage === 'string' - ? t(`error.${errorMessage}`) - : t(`error.${errorMessage.code}`, errorMessage.data ?? {}); - } - - return true; - }, })} /> diff --git a/packages/ui/src/containers/SetPassword/SetPassword.test.tsx b/packages/ui/src/containers/SetPassword/SetPassword.test.tsx index 0bbc2632a..d47855752 100644 --- a/packages/ui/src/containers/SetPassword/SetPassword.test.tsx +++ b/packages/ui/src/containers/SetPassword/SetPassword.test.tsx @@ -54,7 +54,9 @@ describe('', () => { }); await waitFor(() => { - expect(queryByText('error.password_min_length')).not.toBeNull(); + expect(queryByText('error.password_rejected.too_short')).not.toBeNull(); + expect(queryByText('error.password_rejected.character_types')).not.toBeNull(); + expect(queryByText('error.password_rejected.sequence')).not.toBeNull(); }); expect(submit).not.toBeCalled(); @@ -68,7 +70,9 @@ describe('', () => { }); await waitFor(() => { - expect(queryByText('error.password_min_length')).toBeNull(); + expect(queryByText('error.password_rejected.too_short')).toBeNull(); + expect(queryByText('error.password_rejected.character_types')).toBeNull(); + expect(queryByText('error.password_rejected.sequence')).toBeNull(); }); }); @@ -86,7 +90,8 @@ describe('', () => { }); await waitFor(() => { - expect(queryByText('error.invalid_password')).not.toBeNull(); + expect(queryByText('error.password_rejected.character_types')).not.toBeNull(); + expect(queryByText('error.password_rejected.sequence')).not.toBeNull(); }); act(() => { @@ -98,7 +103,8 @@ describe('', () => { }); await waitFor(() => { - expect(queryByText('error.invalid_password')).toBeNull(); + expect(queryByText('error.password_rejected.character_types')).toBeNull(); + expect(queryByText('error.password_rejected.sequence')).toBeNull(); }); }); diff --git a/packages/ui/src/containers/SetPassword/SetPassword.tsx b/packages/ui/src/containers/SetPassword/SetPassword.tsx index 341c259e4..47e59f86c 100644 --- a/packages/ui/src/containers/SetPassword/SetPassword.tsx +++ b/packages/ui/src/containers/SetPassword/SetPassword.tsx @@ -8,7 +8,6 @@ import Button from '@/components/Button'; import IconButton from '@/components/Button/IconButton'; import ErrorMessage from '@/components/ErrorMessage'; import { InputField } from '@/components/InputFields'; -import { validatePassword } from '@/utils/form'; import TogglePassword from './TogglePassword'; import * as styles from './index.module.scss'; @@ -79,17 +78,6 @@ const SetPassword = ({ aria-invalid={!!errors.newPassword} {...register('newPassword', { required: t('error.password_required'), - validate: (password) => { - const errorMessage = validatePassword(password); - - if (errorMessage) { - return typeof errorMessage === 'string' - ? t(`error.${errorMessage}`) - : t(`error.${errorMessage.code}`, errorMessage.data ?? {}); - } - - return true; - }, })} isSuffixFocusVisible={!!watch('newPassword')} suffix={ diff --git a/packages/ui/src/hooks/use-list-translation.test.ts b/packages/ui/src/hooks/use-list-translation.test.ts new file mode 100644 index 000000000..f3a2a0c09 --- /dev/null +++ b/packages/ui/src/hooks/use-list-translation.test.ts @@ -0,0 +1,57 @@ +import useListTranslation from './use-list-translation'; + +describe('useListTranslation (en)', () => { + const translateList = useListTranslation(); + + it('returns undefined for an empty list', () => { + expect(translateList([])).toBeUndefined(); + }); + + it('returns the first item for a list of one item', () => { + expect(translateList(['a'])).toBe('a'); + }); + + it('returns the list with "or" for a list of two items', () => { + expect(translateList(['a', 'b'])).toBe('a or b'); + }); + + it('returns the list with the Oxford comma for a list of three items', () => { + expect(translateList(['a', 'b', 'c'])).toBe('a, b, or c'); + }); + + it('returns the list with the specified joint', () => { + expect(translateList(['a', 'b', 'c'], 'and')).toBe('a, b, and c'); + }); +}); + +describe('useListTranslation (zh)', () => { + const translateList = useListTranslation(); + + it('returns undefined for an empty list', () => { + expect(translateList([])).toBeUndefined(); + }); + + it('returns the first item for a list of one item', () => { + expect(translateList(['苹果'])).toBe('苹果'); + }); + + it('returns the list with "或" for a list of two items', () => { + expect(translateList(['苹果', '橘子'])).toBe('苹果或橘子'); + }); + + it('returns the list with the AP style for a list of three items', () => { + expect(translateList(['苹果', '橘子', '香蕉'])).toBe('苹果、橘子或香蕉'); + }); + + it('returns the list with the specified joint', () => { + expect(translateList(['苹果', '橘子', '香蕉'], 'and')).toBe('苹果、橘子和香蕉'); + }); + + it('adds a space between CJK and non-CJK characters', () => { + expect(translateList(['苹果', '橘子', 'banana'])).toBe('苹果、橘子或 banana'); + expect(translateList(['苹果', '橘子', 'banana'], 'and')).toBe('苹果、橘子和 banana'); + expect(translateList(['banana', '苹果', '橘子'])).toBe('banana、苹果或橘子'); + expect(translateList(['苹果', 'banana', '橘子'])).toBe('苹果、banana 或橘子'); + expect(translateList(['苹果', 'banana', 'orange'])).toBe('苹果、banana 或 orange'); + }); +}); diff --git a/packages/ui/src/hooks/use-list-translation.ts b/packages/ui/src/hooks/use-list-translation.ts new file mode 100644 index 000000000..d9b8bcbeb --- /dev/null +++ b/packages/ui/src/hooks/use-list-translation.ts @@ -0,0 +1,76 @@ +import { conditionalArray } from '@silverhand/essentials'; +import { useCallback } from 'react'; +import { useTranslation } from 'react-i18next'; + +/** + * Returns whether the given character is a CJK character. + * + * @see https://stackoverflow.com/questions/43418812 + */ +const isCjk = (char?: string) => + Boolean( + char?.[0] && /[\u3040-\u30FF\u3400-\u4DBF\u4E00-\u9FFF\uF900-\uFAFF\uFF66-\uFF9F]/.test(char[0]) + ); + +/** + * Returns a function that translates a list of strings into a human-readable list. If the list + * is empty, `undefined` is returned. + * + * For non-CJK languages, the list is translated with the Oxford comma. For CJK languages, the + * list is translated with the AP style. + * + * CAUTION: This function may not be suitable for translating lists of non-English strings if the + * target language does not have the same rules for list translation as English. + * + * @example + * ```ts + * const translateList = useListTranslation(); + * + * // en + * translateList([]); // undefined + * translateList(['a']); // 'a' + * translateList(['a', 'b']); // 'a and b' + * translateList(['a', 'b', 'c']); // 'a, b, or c' + * translateList(['a', 'b', 'c'], 'and'); // 'a, b, and c' + * + * // zh + * translateList(['a', 'b']); // 'a 或 b' + * translateList(['苹果', '橘子', '香蕉']); // '苹果、橘子或香蕉' + * translateList(['苹果', '橘子', 'banana']); // '苹果、橘子或 banana' + * ``` + */ +const useListTranslation = () => { + const { t } = useTranslation(); + + return useCallback( + (list: string[], joint: 'or' | 'and' = 'or') => { + if (list.length === 0) { + return; + } + + if (list.length === 1) { + return list[0]; + } + + const prefix = list.slice(0, -1).join(t('list.separator')); + const suffix = list.at(-1)!; // eslint-disable-line @typescript-eslint/no-non-null-assertion -- `list` is not empty + const jointT = t(`list.${joint}`); + + if (!isCjk(jointT)) { + // Oxford comma + return `${prefix}${t(`list.separator`)}${jointT}${suffix}`; + } + + return conditionalArray( + prefix, + isCjk(prefix.at(-1)) && ' ', + jointT, + isCjk(suffix[0]) && ' ', + suffix + ).join(''); + }, + [t] + ); +}; + +export default useListTranslation; diff --git a/packages/ui/src/hooks/use-password-action.ts b/packages/ui/src/hooks/use-password-action.ts new file mode 100644 index 000000000..52c98b7df --- /dev/null +++ b/packages/ui/src/hooks/use-password-action.ts @@ -0,0 +1,80 @@ +import { type RequestErrorBody } from '@logto/schemas'; +import { useCallback } from 'react'; + +import useApi from '@/hooks/use-api'; + +import useErrorHandler, { type ErrorHandlers } from './use-error-handler'; +import usePasswordErrorMessage from './use-password-error-message'; +import { usePasswordPolicy } from './use-sie'; + +// eslint-disable-next-line @typescript-eslint/no-explicit-any -- we don't know the type of the api, use `any` to avoid type error +export type PasswordAction = (...args: any[]) => Promise; + +export type SuccessHandler = F extends PasswordAction + ? (result?: Response) => void + : never; + +type UsePasswordApiInit = { + api: PasswordAction; + setErrorMessage: (message?: string) => void; + errorHandlers: ErrorHandlers; + successHandler: SuccessHandler>; +}; + +const usePasswordAction = ({ + api, + errorHandlers, + setErrorMessage, + successHandler, +}: UsePasswordApiInit) => { + const asyncAction = useApi(api); + const handleError = useErrorHandler(); + const { getErrorMessage, getErrorMessageFromBody } = usePasswordErrorMessage(); + const { policyChecker } = usePasswordPolicy(); + const passwordRejectionHandler = useCallback( + (error: RequestErrorBody) => { + setErrorMessage(getErrorMessageFromBody(error)); + }, + [getErrorMessageFromBody, setErrorMessage] + ); + + const action = useCallback( + async (password: string) => { + // Perform fast check before sending request + const fastCheckErrorMessage = getErrorMessage(policyChecker.fastCheck(password)); + if (fastCheckErrorMessage) { + setErrorMessage(fastCheckErrorMessage); + return; + } + + const [error, result] = await asyncAction(password); + + if (error) { + await handleError(error, { + 'password.rejected': passwordRejectionHandler, + ...errorHandlers, + }); + + return; + } + + successHandler(result); + }, + [ + asyncAction, + errorHandlers, + getErrorMessage, + handleError, + passwordRejectionHandler, + policyChecker, + setErrorMessage, + successHandler, + ] + ); + + return { + action, + }; +}; + +export default usePasswordAction; diff --git a/packages/ui/src/hooks/use-password-error-message.ts b/packages/ui/src/hooks/use-password-error-message.ts new file mode 100644 index 000000000..6472dbb9d --- /dev/null +++ b/packages/ui/src/hooks/use-password-error-message.ts @@ -0,0 +1,78 @@ +import { type PasswordRejectionCode, type PasswordIssue } from '@logto/core-kit'; +import { type RequestErrorBody } from '@logto/schemas'; +import { useCallback } from 'react'; +import { useTranslation } from 'react-i18next'; + +import useListTranslation from '@/hooks/use-list-translation'; + +/** + * Return an object with two functions for getting the error message from an array of {@link PasswordIssue} or a {@link RequestErrorBody}. + */ +const usePasswordErrorMessage = () => { + const { t } = useTranslation(); + const translateList = useListTranslation(); + const getErrorMessage = useCallback( + (issues: PasswordIssue[]) => { + // Errors that should be displayed first and alone + const singleDisplayError = ( + [ + 'unsupported_characters', + 'too_short', + 'character_types', + 'too_long', + 'pwned', + ] satisfies PasswordRejectionCode[] + ) + .map((code) => issues.find((issue) => issue.code === `password_rejected.${code}`)) + .find(Boolean); + + if (singleDisplayError) { + return t(`error.${singleDisplayError.code}`, singleDisplayError.interpolation ?? {}); + } + + // The `restricted` errors should be displayed together + const restrictedErrors = issues + .filter((issue): issue is PasswordIssue<`restricted.${string}` & PasswordRejectionCode> => + issue.code.startsWith('password_rejected.restricted.') + ) + .map((issue) => t(`error.${issue.code}`)); + + if (restrictedErrors.length > 0) { + return t('error.password_rejected.restricted_found', { + list: translateList(restrictedErrors), + }); + } + }, + [translateList, t] + ); + + const getErrorMessageFromBody = useCallback( + (error: RequestErrorBody) => { + if (error.code === 'password.rejected') { + // eslint-disable-next-line no-restricted-syntax -- trust the type from the server if the code matches + return getErrorMessage(error.data as PasswordIssue[]) ?? error.message; + } + }, + [getErrorMessage] + ); + + return { + /** + * Get the error message from an array of {@link PasswordIssue}. + * If the array is empty or the error is not recognized, `undefined` is returned. + */ + getErrorMessage, + /** + * Get the error message from a {@link RequestErrorBody}. + * + * First, it will check if the error code is `password.rejected`: + * - If so, it will assume `error.data` is an array of {@link PasswordIssue}, then it will call + * {@link getErrorMessage} with the array; + * - If the result is `undefined`, it will directly return the error message from the body. + * - Otherwise, it will return `undefined`. + */ + getErrorMessageFromBody, + }; +}; + +export default usePasswordErrorMessage; diff --git a/packages/ui/src/hooks/use-sie.ts b/packages/ui/src/hooks/use-sie.ts index c4c3bb744..f4b7c4ceb 100644 --- a/packages/ui/src/hooks/use-sie.ts +++ b/packages/ui/src/hooks/use-sie.ts @@ -1,5 +1,7 @@ +import { PasswordPolicyChecker, passwordPolicyGuard } from '@logto/core-kit'; import { SignInIdentifier } from '@logto/schemas'; -import { useContext } from 'react'; +import { useContext, useMemo } from 'react'; +import { useTranslation } from 'react-i18next'; import PageContext from '@/Providers/PageContextProvider/PageContext'; import { type VerificationCodeIdentifier } from '@/types'; @@ -28,6 +30,21 @@ export const useSignInExperience = () => { return experienceSettings; }; +export const usePasswordPolicy = () => { + const { t } = useTranslation(); + const { experienceSettings } = useContext(PageContext); + const policy = useMemo( + () => passwordPolicyGuard.parse(experienceSettings?.passwordPolicy ?? {}), + [experienceSettings] + ); + const policyChecker = useMemo(() => new PasswordPolicyChecker(policy), [policy]); + + return { + policy, + policyChecker, + }; +}; + export const useForgotPasswordSettings = () => { const { experienceSettings } = useContext(PageContext); const { forgotPassword } = experienceSettings ?? {}; diff --git a/packages/ui/src/pages/Continue/SetPassword/index.tsx b/packages/ui/src/pages/Continue/SetPassword/index.tsx index f2ac7ed8b..299558062 100644 --- a/packages/ui/src/pages/Continue/SetPassword/index.tsx +++ b/packages/ui/src/pages/Continue/SetPassword/index.tsx @@ -1,19 +1,66 @@ -import SecondaryPageLayout from '@/Layout/SecondaryPageLayout'; -import SetPasswordForm from '@/containers/SetPassword'; -import { passwordMinLength } from '@/utils/form'; +import { useCallback, useMemo, useState } from 'react'; +import { useNavigate } from 'react-router-dom'; -import useSetPassword from './use-set-password'; +import SecondaryPageLayout from '@/Layout/SecondaryPageLayout'; +import { addProfile } from '@/apis/interaction'; +import SetPasswordForm from '@/containers/SetPassword'; +import { useConfirmModal } from '@/hooks/use-confirm-modal'; +import type { ErrorHandlers } from '@/hooks/use-error-handler'; +import usePasswordAction, { type SuccessHandler } from '@/hooks/use-password-action'; +import useRequiredProfileErrorHandler from '@/hooks/use-required-profile-error-handler'; +import { usePasswordPolicy } from '@/hooks/use-sie'; const SetPassword = () => { - const { setPassword } = useSetPassword(); + const [errorMessage, setErrorMessage] = useState(); + const clearErrorMessage = useCallback(() => { + setErrorMessage(undefined); + }, []); + + const navigate = useNavigate(); + const { show } = useConfirmModal(); + + const requiredProfileErrorHandler = useRequiredProfileErrorHandler(); + const errorHandlers: ErrorHandlers = useMemo( + () => ({ + 'user.password_exists_in_profile': async (error) => { + await show({ type: 'alert', ModalContent: error.message, cancelText: 'action.got_it' }); + navigate(-1); + }, + ...requiredProfileErrorHandler, + }), + [navigate, requiredProfileErrorHandler, show] + ); + const successHandler: SuccessHandler = useCallback((result) => { + if (result?.redirectTo) { + window.location.replace(result.redirectTo); + } + }, []); + const { action } = usePasswordAction({ + api: addProfile, + setErrorMessage, + errorHandlers, + successHandler, + }); + + const { + policy: { + length: { min }, + characterTypes: { min: count }, + }, + } = usePasswordPolicy(); return ( - + ); }; diff --git a/packages/ui/src/pages/Continue/SetPassword/use-set-password.ts b/packages/ui/src/pages/Continue/SetPassword/use-set-password.ts deleted file mode 100644 index 949706c30..000000000 --- a/packages/ui/src/pages/Continue/SetPassword/use-set-password.ts +++ /dev/null @@ -1,53 +0,0 @@ -import { useMemo, useCallback } from 'react'; -import { useNavigate } from 'react-router-dom'; - -import { addProfile } from '@/apis/interaction'; -import useApi from '@/hooks/use-api'; -import { useConfirmModal } from '@/hooks/use-confirm-modal'; -import useErrorHandler from '@/hooks/use-error-handler'; -import type { ErrorHandlers } from '@/hooks/use-error-handler'; -import useRequiredProfileErrorHandler from '@/hooks/use-required-profile-error-handler'; - -const useSetPassword = () => { - const navigate = useNavigate(); - const { show } = useConfirmModal(); - - const handleError = useErrorHandler(); - const asyncAddProfile = useApi(addProfile); - - const requiredProfileErrorHandler = useRequiredProfileErrorHandler(); - - const errorHandlers: ErrorHandlers = useMemo( - () => ({ - 'user.password_exists_in_profile': async (error) => { - await show({ type: 'alert', ModalContent: error.message, cancelText: 'action.got_it' }); - navigate(-1); - }, - ...requiredProfileErrorHandler, - }), - [navigate, requiredProfileErrorHandler, show] - ); - - const setPassword = useCallback( - async (password: string) => { - const [error, result] = await asyncAddProfile({ password }); - - if (error) { - await handleError(error, errorHandlers); - - return; - } - - if (result?.redirectTo) { - window.location.replace(result.redirectTo); - } - }, - [asyncAddProfile, errorHandlers, handleError] - ); - - return { - setPassword, - }; -}; - -export default useSetPassword; diff --git a/packages/ui/src/pages/RegisterPassword/index.tsx b/packages/ui/src/pages/RegisterPassword/index.tsx index 0cb187f31..9db7f7a84 100644 --- a/packages/ui/src/pages/RegisterPassword/index.tsx +++ b/packages/ui/src/pages/RegisterPassword/index.tsx @@ -1,17 +1,55 @@ import { SignInIdentifier } from '@logto/schemas'; +import { useCallback, useMemo, useState } from 'react'; +import { useNavigate } from 'react-router-dom'; import SecondaryPageLayout from '@/Layout/SecondaryPageLayout'; +import { setUserPassword } from '@/apis/interaction'; import SetPassword from '@/containers/SetPassword'; -import { useSieMethods } from '@/hooks/use-sie'; -import { passwordMinLength } from '@/utils/form'; +import { useConfirmModal } from '@/hooks/use-confirm-modal'; +import { type ErrorHandlers } from '@/hooks/use-error-handler'; +import usePasswordAction, { type SuccessHandler } from '@/hooks/use-password-action'; +import { usePasswordPolicy, useSieMethods } from '@/hooks/use-sie'; import ErrorPage from '../ErrorPage'; -import useUsernamePasswordRegister from './use-username-password-register'; - const RegisterPassword = () => { const { signUpMethods } = useSieMethods(); - const setPassword = useUsernamePasswordRegister(); + + const navigate = useNavigate(); + const { show } = useConfirmModal(); + const [errorMessage, setErrorMessage] = useState(); + const clearErrorMessage = useCallback(() => { + setErrorMessage(undefined); + }, []); + const errorHandlers: ErrorHandlers = useMemo( + () => ({ + // Incase previous page submitted username has been taken + 'user.username_already_in_use': async (error) => { + await show({ type: 'alert', ModalContent: error.message, cancelText: 'action.got_it' }); + navigate(-1); + }, + }), + [navigate, show] + ); + const successHandler: SuccessHandler = useCallback((result) => { + if (result && 'redirectTo' in result) { + window.location.replace(result.redirectTo); + } + }, []); + + const { action } = usePasswordAction({ + api: setUserPassword, + setErrorMessage, + errorHandlers, + successHandler, + }); + + const { + policy: { + length: { min }, + characterTypes: { min: count }, + }, + } = usePasswordPolicy(); if (!signUpMethods.includes(SignInIdentifier.Username)) { return ; @@ -20,14 +58,14 @@ const RegisterPassword = () => { return ( { - void setPassword(password); - }} + errorMessage={errorMessage} + clearErrorMessage={clearErrorMessage} + onSubmit={action} /> ); diff --git a/packages/ui/src/pages/RegisterPassword/use-username-password-register.ts b/packages/ui/src/pages/RegisterPassword/use-username-password-register.ts deleted file mode 100644 index 0de870f58..000000000 --- a/packages/ui/src/pages/RegisterPassword/use-username-password-register.ts +++ /dev/null @@ -1,44 +0,0 @@ -import { useMemo, useCallback } from 'react'; -import { useNavigate } from 'react-router-dom'; - -import { setUserPassword } from '@/apis/interaction'; -import useApi from '@/hooks/use-api'; -import { useConfirmModal } from '@/hooks/use-confirm-modal'; -import useErrorHandler from '@/hooks/use-error-handler'; -import type { ErrorHandlers } from '@/hooks/use-error-handler'; - -const useUsernamePasswordRegister = () => { - const navigate = useNavigate(); - const { show } = useConfirmModal(); - - const handleError = useErrorHandler(); - const asyncSetPassword = useApi(setUserPassword); - - const errorHandlers: ErrorHandlers = useMemo( - () => ({ - // Incase previous page submitted username has been taken - 'user.username_already_in_use': async (error) => { - await show({ type: 'alert', ModalContent: error.message, cancelText: 'action.got_it' }); - navigate(-1); - }, - }), - [navigate, show] - ); - - return useCallback( - async (password: string) => { - const [error, result] = await asyncSetPassword(password); - - if (error) { - await handleError(error, errorHandlers); - } - - if (result && 'redirectTo' in result) { - window.location.replace(result.redirectTo); - } - }, - [asyncSetPassword, errorHandlers, handleError] - ); -}; - -export default useUsernamePasswordRegister; diff --git a/packages/ui/src/pages/ResetPassword/index.tsx b/packages/ui/src/pages/ResetPassword/index.tsx index 08e853472..b55faea9a 100644 --- a/packages/ui/src/pages/ResetPassword/index.tsx +++ b/packages/ui/src/pages/ResetPassword/index.tsx @@ -1,23 +1,71 @@ -import SecondaryPageLayout from '@/Layout/SecondaryPageLayout'; -import SetPassword from '@/containers/SetPassword'; -import { passwordMinLength } from '@/utils/form'; +import { useCallback, useMemo, useState } from 'react'; +import { useTranslation } from 'react-i18next'; +import { useNavigate } from 'react-router-dom'; -import useResetPassword from './use-reset-password'; +import SecondaryPageLayout from '@/Layout/SecondaryPageLayout'; +import { setUserPassword } from '@/apis/interaction'; +import SetPassword from '@/containers/SetPassword'; +import { useConfirmModal } from '@/hooks/use-confirm-modal'; +import { type ErrorHandlers } from '@/hooks/use-error-handler'; +import usePasswordAction, { type SuccessHandler } from '@/hooks/use-password-action'; +import { usePasswordPolicy } from '@/hooks/use-sie'; +import useToast from '@/hooks/use-toast'; const ResetPassword = () => { - const { resetPassword, errorMessage, clearErrorMessage } = useResetPassword(); + const [errorMessage, setErrorMessage] = useState(); + const clearErrorMessage = useCallback(() => { + setErrorMessage(undefined); + }, []); + const { t } = useTranslation(); + const { setToast } = useToast(); + const navigate = useNavigate(); + const { show } = useConfirmModal(); + const errorHandlers: ErrorHandlers = useMemo( + () => ({ + 'session.verification_session_not_found': async (error) => { + await show({ type: 'alert', ModalContent: error.message, cancelText: 'action.got_it' }); + navigate(-2); + }, + 'user.same_password': (error) => { + setErrorMessage(error.message); + }, + }), + [navigate, setErrorMessage, show] + ); + const successHandler: SuccessHandler = useCallback( + (result) => { + if (result) { + setToast(t('description.password_changed')); + navigate('/sign-in', { replace: true }); + } + }, + [navigate, setToast, t] + ); + + const { action } = usePasswordAction({ + api: setUserPassword, + setErrorMessage, + errorHandlers, + successHandler, + }); + const { + policy: { + length: { min }, + characterTypes: { min: count }, + }, + } = usePasswordPolicy(); return ( ); diff --git a/packages/ui/src/pages/ResetPassword/use-reset-password.ts b/packages/ui/src/pages/ResetPassword/use-reset-password.ts deleted file mode 100644 index b9cf6bca7..000000000 --- a/packages/ui/src/pages/ResetPassword/use-reset-password.ts +++ /dev/null @@ -1,65 +0,0 @@ -import { useMemo, useState, useCallback } from 'react'; -import { useTranslation } from 'react-i18next'; -import { useNavigate } from 'react-router-dom'; - -import { setUserPassword } from '@/apis/interaction'; -import useApi from '@/hooks/use-api'; -import { useConfirmModal } from '@/hooks/use-confirm-modal'; -import useErrorHandler from '@/hooks/use-error-handler'; -import type { ErrorHandlers } from '@/hooks/use-error-handler'; -import useToast from '@/hooks/use-toast'; - -const useResetPassword = () => { - const { t } = useTranslation(); - const navigate = useNavigate(); - const { setToast } = useToast(); - const { show } = useConfirmModal(); - const [errorMessage, setErrorMessage] = useState(); - - const handleError = useErrorHandler(); - const asyncResetPassword = useApi(setUserPassword); - - const clearErrorMessage = useCallback(() => { - // eslint-disable-next-line unicorn/no-useless-undefined - setErrorMessage(undefined); - }, []); - - const resetPasswordErrorHandlers: ErrorHandlers = useMemo( - () => ({ - 'session.verification_session_not_found': async (error) => { - await show({ type: 'alert', ModalContent: error.message, cancelText: 'action.got_it' }); - navigate(-2); - }, - 'user.same_password': (error) => { - setErrorMessage(error.message); - }, - }), - [navigate, setErrorMessage, show] - ); - - const resetPassword = useCallback( - async (password: string) => { - const [error, result] = await asyncResetPassword(password); - - if (error) { - await handleError(error, resetPasswordErrorHandlers); - - return; - } - - if (result) { - setToast(t('description.password_changed')); - navigate('/sign-in', { replace: true }); - } - }, - [asyncResetPassword, handleError, navigate, resetPasswordErrorHandlers, setToast, t] - ); - - return { - resetPassword, - errorMessage, - clearErrorMessage, - }; -}; - -export default useResetPassword;