// Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ import { createMessageName } from '@formkit/validation' import { i18n } from '#shared/i18n.ts' import { commaSeparatedList, order } from '#shared/utils/formatter.ts' import type { FormKitLocale } from '@formkit/i18n' import type { FormKitValidationMessages } from '@formkit/validation' import type { ComputedRef } from 'vue' interface FormKitLocaleExtended extends FormKitLocale { validation: FormKitValidationMessages } // TODO: Use translateLabel for all validation messages if we stay with the labels inside of the messages. It's a open // question if we want use them inside of the messages. const loadLocales = (): FormKitLocaleExtended => { return { ui: { /** * Shown on a button for adding additional items. */ add: () => i18n.t('Add'), /** * Shown when a button to remove items is visible. */ remove: () => i18n.t('Remove'), /** * Shown when there are multiple items to remove at the same time. */ removeAll: () => i18n.t('Remove all'), /** * Shown when all fields are not filled out correctly. */ incomplete: () => i18n.t('Sorry, not all fields are filled out correctly.'), /** * Shown in a button inside a form to submit the form. */ submit: () => i18n.t('Submit'), /** * Shown when no files are selected. */ noFiles: () => i18n.t('No file chosen.'), /** * Shown on buttons that move fields up in a list. */ moveUp: () => i18n.t('Move up'), /** * Shown on buttons that move fields down in a list. */ moveDown: () => i18n.t('Move down'), /** * Shown when something is actively loading. */ isLoading: () => i18n.t('Loading…'), /** * Shown when there is more to load. */ loadMore: () => i18n.t('Load more'), /** * Show on buttons that navigate state forward */ next: () => i18n.t('Next'), /** * Show on buttons that navigate state backward */ prev: () => i18n.t('Previous'), /** * Shown when adding all values. */ addAllValues: () => i18n.t('Add all values'), /** * Shown when adding selected values. */ addSelectedValues: () => i18n.t('Add selected values'), /** * Shown when removing all values. */ removeAllValues: () => i18n.t('Remove all values'), /** * Shown when removing selected values. */ removeSelectedValues: () => i18n.t('Remove selected values'), /** * Shown when there is a date to choose. */ chooseDate: () => i18n.t('Choose date'), /** * Shown when there is a date to change. */ changeDate: () => i18n.t('Change date'), /** * Shown above error summaries when someone attempts to submit a form with * errors and the developer has implemented ``. */ summaryHeader: () => i18n.t('There were errors in your form.'), /* * Shown when there is something to close */ close: () => i18n.t('Close'), /** * Shown when there is something to open. */ open: () => i18n.t('Open'), }, validation: { /** * The value is not an accepted value. * @see {@link https://docs.formkit.com/essentials/validation#accepted} */ accepted({ name }) { /* */ return i18n.t('Please accept the %s.', name) /* */ }, /** * The value is not letter and/or spaces * @see {@link https://docs.formkit.com/essentials/validation#alpha-spaces} */ alpha_spaces() { /* */ return i18n.t('This field can only contain letters and spaces.') /* */ }, /** * The date is not after * @see {@link https://docs.formkit.com/essentials/validation#date-after} */ date_after({ args }) { if (Array.isArray(args) && args.length) { /* */ return i18n.t( 'This field must have a value that is after %s.', i18n.date(args[0]), ) /* */ } /* */ return i18n.t('This field must have a value that is in the future.') /* */ }, /** * The value is not a letter. * @see {@link https://docs.formkit.com/essentials/validation#alpha} */ alpha() { /* */ return i18n.t('This field can only contain alphabetical characters.') /* */ }, /** * The value is not alphanumeric * @see {@link https://docs.formkit.com/essentials/validation#alphanumeric} */ alphanumeric() { /* */ return i18n.t('This field can only contain letters and numbers.') /* */ }, /** * The value have no letter. * @see {@link https://formkit.com/essentials/validation#contains_alpha} */ contains_alpha() { /* */ return i18n.t('This field must contain alphabetical characters.') /* */ }, /** * The value have no alphanumeric * @see {@link https://formkit.com/essentials/validation#contains_alphanumeric} */ contains_alphanumeric() { /* */ return i18n.t('This field must contain letters or numbers.') /* */ }, /** * The value have no letter and/or spaces * @see {@link https://formkit.com/essentials/validation#contains_alpha-spaces} */ contains_alpha_spaces() { /* */ return i18n.t('This field must contain letters or spaces.') /* */ }, /** * The value have no symbol * @see {@link https://formkit.com/essentials/validation#contains_symbol} */ contains_symbol() { /* */ return i18n.t('This field must contain a symbol.') /* */ }, /** * The value have no uppercase * @see {@link https://formkit.com/essentials/validation#contains_uppercase} */ contains_uppercase() { /* */ return i18n.t('This field must contain an uppercase letter.') /* */ }, /** * The value have no lowercase * @see {@link https://formkit.com/essentials/validation#contains_lowercase} */ contains_lowercase() { /* */ return i18n.t('This field must contain a lowercase letter.') /* */ }, /** * The value have no numeric * @see {@link https://formkit.com/essentials/validation#contains_numeric} */ contains_numeric() { /* */ return i18n.t('This field must contain numbers.') /* */ }, /** * The value is not symbol * @see {@link https://formkit.com/essentials/validation#symbol} */ symbol() { /* */ return i18n.t('This field must be a symbol.') /* */ }, /** * The value is not uppercase * @see {@link https://formkit.com/essentials/validation#uppercase} */ uppercase() { /* */ return i18n.t('This field can only contain uppercase letters.') /* */ }, /** * The value is not lowercase * @see {@link https://formkit.com/essentials/validation#lowercase} */ lowercase() { /* */ return i18n.t('This field can only contain lowercase letters.') /* */ }, /** * The date is not before * @see {@link https://docs.formkit.com/essentials/validation#date-before} */ date_before({ args }) { if (Array.isArray(args) && args.length) { /* */ return i18n.t( 'This field must have a value that is before %s.', i18n.date(args[0]), ) /* */ } /* */ return i18n.t('This field must have a value that is in the past.') /* */ }, /** * The value is not between two numbers * @see {@link https://docs.formkit.com/essentials/validation#between} */ between({ args }) { if (Number.isNaN(args[0]) || Number.isNaN(args[1])) { /* */ return i18n.t( "This field was configured incorrectly and can't be submitted.", ) /* */ } const [first, second] = order(args[0], args[1]) /* */ return i18n.t( 'This field must have a value that is between %s and %s.', first, second, ) /* */ }, /** * The confirmation field does not match * @see {@link https://docs.formkit.com/essentials/validation#confirm} */ confirm() { // TODO: Check if message is in a good shape (e.g. for the usage for password + confirm password). /* */ return i18n.t("This field doesn't correspond to the expected value.") /* */ }, /** * The value is not a valid date * @see {@link https://docs.formkit.com/essentials/validation#date-format} */ date_format({ args }) { if (Array.isArray(args) && args.length) { /* */ return i18n.t( 'This field isn\'t a valid date, please use the format "%s".', args[0], ) /* */ } /* */ return i18n.t( "This field was configured incorrectly and can't be submitted.", ) /* */ }, /** * Is not within expected date range * @see {@link https://docs.formkit.com/essentials/validation#date-between} */ date_between({ args }) { /* */ return i18n.t( 'This field must have a value that is between %s and %s.', i18n.date(args[0]), i18n.date(args[1]), ) /* */ }, /** * Shown when the user-provided value is not a valid email address. * @see {@link https://docs.formkit.com/essentials/validation#email} */ email: i18n.t('Please enter a valid email address.'), /** * Does not end with the specified value * @see {@link https://docs.formkit.com/essentials/validation#ends-with} */ ends_with({ args }) { /* */ return i18n.t( 'This field doesn\'t end with "%s".', commaSeparatedList(args), ) /* */ }, /** * Is not an allowed value * @see {@link https://docs.formkit.com/essentials/validation#is} */ is() { /* */ return i18n.t("This field doesn't contain an allowed value.") /* */ }, /** * Does not match specified length * @see {@link https://docs.formkit.com/essentials/validation#length} */ length({ args: [first = 0, second = Infinity] }) { const min = Number(first) <= Number(second) ? first : second const max = Number(second) >= Number(first) ? second : first if (min === 1 && max === Infinity) { /* */ return i18n.t('This field must contain at least one character.') /* */ } if (min === 0 && max) { /* */ return i18n.t( 'This field must not contain more than %s characters.', max, ) /* */ } if (min && max === Infinity) { /* */ return i18n.t('This field must contain at least %s characters.', min) /* */ } /* */ return i18n.t( 'This field must contain between %s and %s characters.', min, max, ) /* */ }, /** * Value is not a match * @see {@link https://docs.formkit.com/essentials/validation#matches} */ matches() { /* */ return i18n.t("This field doesn't contain an allowed value.") /* */ }, /** * Exceeds maximum allowed value * @see {@link https://docs.formkit.com/essentials/validation#max} */ max({ node: { value }, args }) { if (Array.isArray(value)) { /* */ return i18n.t("This field can't have more than %s entries.", args[0]) /* */ } /* */ return i18n.t( 'This field must have a value that is at most %s.', args[0], ) /* */ }, /** * The (field-level) value does not match specified mime type * @see {@link https://docs.formkit.com/essentials/validation#mime} */ mime({ args }) { if (!args[0]) { /* */ return i18n.t('No file formats allowed.') /* */ } /* */ return i18n.t('This field must be of the type "%s".', args[0]) /* */ }, /** * Does not fulfill minimum allowed value * @see {@link https://docs.formkit.com/essentials/validation#min} */ min({ node: { value }, args }) { if (Array.isArray(value)) { /* */ return i18n.t("This field can't have less than %s entries.", args[0]) /* */ } /* */ return i18n.t( 'This field must have a value that is at least %s.', args[0], ) /* */ }, /** * Is not an allowed value * @see {@link https://docs.formkit.com/essentials/validation#not} */ not({ node: { value } }) { /* */ return i18n.t( 'This field can\'t contain the value "%s".', value as string, ) /* */ }, /** * Is not a number * @see {@link https://docs.formkit.com/essentials/validation#number} */ number() { /* */ return i18n.t('This field must contain a number.') /* */ }, /** * Required field. * @see {@link https://docs.formkit.com/essentials/validation#required} */ required() { /* */ return i18n.t('This field is required.') /* */ }, /** * Require one field. * @see {@link https://formkit.com/essentials/validation#require-one} */ require_one: ({ name, node, args: inputNames }) => { const labels = inputNames .map((name) => { const dependentNode = node.at(name) if (dependentNode) { return createMessageName(dependentNode) } return false }) .filter((name) => !!name) as unknown as ComputedRef[] labels.unshift(name as unknown as ComputedRef) /* */ // return `${labels.join(' or ')} is required.` const translatedSeparator = i18n.t('or') return i18n.t( '%s is required.', labels .map((label: ComputedRef) => label.value) .join(` ${translatedSeparator} `), ) /* */ }, /** * Does not start with specified value * @see {@link https://docs.formkit.com/essentials/validation#starts-with} */ starts_with({ args }) { /* */ return i18n.t( 'This field doesn\'t start with "%s".', commaSeparatedList(args), ) /* */ }, /** * Is not a url * @see {@link https://docs.formkit.com/essentials/validation#url} */ url() { /* */ return i18n.t('Please include a valid url.') /* */ }, /** * Shown when the date is invalid. */ invalidDate: () => i18n.t('The selected date is invalid.'), }, } } export default loadLocales