@commercetools-uikit/localized-text-input 16.12.1 → 17.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +2 -0
- package/dist/commercetools-uikit-localized-text-input.cjs.d.ts +1 -1
- package/dist/commercetools-uikit-localized-text-input.cjs.dev.js +15 -8
- package/dist/commercetools-uikit-localized-text-input.cjs.prod.js +9 -6
- package/dist/commercetools-uikit-localized-text-input.esm.js +16 -9
- package/dist/declarations/src/localized-text-input.d.ts +5 -1
- package/package.json +13 -13
- package/dist/commercetools-uikit-localized-text-input.cjs.d.ts.map +0 -1
package/README.md
CHANGED
|
@@ -68,7 +68,9 @@ export default Example;
|
|
|
68
68
|
| `placeholder` | `Record` | | | Placeholders for each language. Object of the same shape as `value`. |
|
|
69
69
|
| `horizontalConstraint` | `union`<br/>Possible values:<br/>`, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto'` | | `'scale'` | Horizontal size limit of the input fields. |
|
|
70
70
|
| `hasError` | `boolean` | | | Will apply the error state to each input without showing any error message. |
|
|
71
|
+
| `hasWarning` | `boolean` | | | Indicates the input field has a warning |
|
|
71
72
|
| `errors` | `Record` | | | Used to show errors underneath the inputs of specific currencies. Pass an object whose key is a currency and whose value is the error to show for that key. |
|
|
73
|
+
| `warnings` | `Record` | | | A map of warnings. |
|
|
72
74
|
|
|
73
75
|
## `data-*` props
|
|
74
76
|
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
export * from "./declarations/src/index";
|
|
2
2
|
export { default } from "./declarations/src/index";
|
|
3
|
-
//# sourceMappingURL=
|
|
3
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbWVyY2V0b29scy11aWtpdC1sb2NhbGl6ZWQtdGV4dC1pbnB1dC5janMuZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4vZGVjbGFyYXRpb25zL3NyYy9pbmRleC5kLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBIn0=
|
|
@@ -88,7 +88,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
88
88
|
} : {
|
|
89
89
|
name: "1ysn02y-LocalizedInput",
|
|
90
90
|
styles: "width:100%;position:relative;display:flex;label:LocalizedInput;",
|
|
91
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-text-input.tsx"],"names":[],"mappings":"AA4Lc","file":"localized-text-input.tsx","sourcesContent":["import {\n  type FocusEventHandler,\n  type ChangeEventHandler,\n  useCallback,\n} from 'react';\nimport { FormattedMessage } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport { ErrorMessage } from '@commercetools-uikit/messages';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  isTouched,\n  omitEmptyTranslations,\n  isEmpty,\n  createLocalizedString,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { createSequentialId, warning } from '@commercetools-uikit/utils';\nimport TextInput from '@commercetools-uikit/text-input';\nimport {\n  LocalizedInputToggle,\n  messagesLocalizedInput,\n} from '@commercetools-uikit/input-utils';\nimport {\n  getLocalizedInputStyles,\n  getLanguageLabelStyles,\n} from './localized-text-input.styles';\n\ninterface HTMLLocalizedInputElement extends HTMLInputElement {\n  language: string;\n}\n\nexport type TLocalizedTextInputProps = {\n  id?: string;\n  name?: string;\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   *   then input doesn't accept a \"languages\" prop, instead all possible\n  languages have to exist (with empty or filled strings) on the value:\n    { en: 'foo', de: '', es: '' }\n   */\n  value: Record<string, string>;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  /**\n   * Specifies which language will be shown in case the `LocalizedTextInput` is collapsed.\n   */\n  selectedLanguage: string;\n  /**\n   * Called when any field is blurred. Is called with the `event` of that field.\n   */\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Will hide the language expansion controls when set to `true`. All languages will be shown when set to `true`.\n   */\n  hideLanguageExpansionControls?: boolean;\n  /**\n   * Controls whether one or all languages are visible by default\n   */\n  defaultExpandLanguages?: boolean;\n  /**\n   * Focus the input field on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Disables all input fields.\n   */\n  isDisabled?: boolean;\n  /**\n   * Disables all input fields and shows them in read-only mode.\n   */\n  isReadOnly?: boolean;\n  /**\n   * Placeholders for each language. Object of the same shape as `value`.\n   */\n  placeholder?: Record<string, string>;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 1\n    | 2\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Will apply the error state to each input without showing any error message.\n   */\n  hasError?: boolean;\n  /**\n   * Used to show errors underneath the inputs of specific currencies. Pass an object whose key is a currency and whose value is the error to show for that key.\n   */\n  errors?: Record<string, string>;\n};\n\nexport type TLocalizedInputProps = {\n  /**\n   * Used as prefix of HTML `id` property. Each input field id will have the language as a suffix (`${idPrefix}.${lang}`), e.g. `foo.en`. You can use the static `LocalizedTextInput.getId(idPrefix, language)` to create this id string, e.g. for labels.\n   */\n  id?: string;\n  /**\n   * Used as HTML `name` property for each input field. Each input field name will have the language as a suffix (`${namePrefix}.${lang}`), e.g. `foo.en`\n   */\n  name?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  value: string;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  language: string;\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  isAutofocussed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasError?: boolean;\n  placeholder?: string;\n};\n\nconst sequentialId = createSequentialId('localized-text-input-');\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the language to the target.\n      // That way the users of LocalizedTextInput's onChange can read\n      // event.target.language and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the language from\n      // the input name anymore.\n      //\n      event.target.language = props.language;\n      onChange?.(event);\n    },\n    [props.language, onChange]\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  return (\n    <div\n      key={props.language}\n      css={css`\n        width: 100%;\n        position: relative;\n        display: flex;\n      `}\n    >\n      <label htmlFor={props.id} css={getLanguageLabelStyles(props)}>\n        {props.language.toUpperCase()}\n      </label>\n      <TextInput\n        {...props}\n        onChange={handleChange}\n        css={getLocalizedInputStyles}\n      />\n    </div>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst RequiredValueErrorMessage = () => (\n  <ErrorMessage>\n    <FormattedMessage {...messagesLocalizedInput.missingRequiredField} />\n  </ErrorMessage>\n);\n\nRequiredValueErrorMessage.displayName = 'RequiredValueErrorMessage';\n\nconst LocalizedTextInput = (props: TLocalizedTextInputProps) => {\n  const defaultExpansionState =\n    props.hideLanguageExpansionControls ||\n    props.defaultExpandLanguages || // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areLanguagesExpanded, toggleLanguages] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedInputToggle = useCallback(\n    () => toggleLanguages(),\n    [toggleLanguages]\n  );\n\n  const languages = sortLanguages(\n    props.selectedLanguage,\n    Object.keys(props.value)\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingLanguages =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);\n\n  if (hasErrorInRemainingLanguages) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (hasErrorInRemainingLanguages !== areLanguagesExpanded) {\n      toggleLanguages();\n    }\n  }\n\n  const shouldRenderLanguagesButton =\n    languages.length > 1 && !props.hideLanguageExpansionControls;\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  if (props.hideLanguageExpansionControls) {\n    warning(\n      typeof props.defaultExpandLanguages !== 'boolean',\n      'LocalizedTextInput: \"defaultExpandLanguages\" does not have any effect when \"hideLanguageExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack>\n          {languages.map((language, index) => {\n            const isFirstLanguage = index === 0;\n            if (!isFirstLanguage && !areLanguagesExpanded) return null;\n\n            return (\n              <div key={language}>\n                <Stack scale=\"xs\">\n                  <LocalizedInput\n                    autoComplete={props.autoComplete}\n                    id={LocalizedTextInput.getId(id, language)}\n                    name={LocalizedTextInput.getName(props.name, language)}\n                    value={props.value[language]}\n                    onChange={props.onChange}\n                    language={language}\n                    placeholder={\n                      props.placeholder\n                        ? props.placeholder[language]\n                        : undefined\n                    }\n                    onBlur={props.onBlur}\n                    onFocus={props.onFocus}\n                    isAutofocussed={index === 0 && props.isAutofocussed}\n                    isDisabled={props.isDisabled}\n                    isReadOnly={props.isReadOnly}\n                    hasError={Boolean(\n                      props.hasError || (props.errors && props.errors[language])\n                    )}\n                    {...createLocalizedDataAttributes(props, language)}\n                    /* ARIA */\n                    aria-invalid={props['aria-invalid']}\n                    aria-errormessage={props['aria-errormessage']}\n                  />\n                  {props.errors && props.errors[language]}\n                </Stack>\n              </div>\n            );\n          })}\n        </Stack>\n        {shouldRenderLanguagesButton && (\n          <LocalizedInputToggle\n            isOpen={areLanguagesExpanded}\n            onClick={onLocalizedInputToggle}\n            isDisabled={areLanguagesExpanded && hasErrorInRemainingLanguages}\n            remainingLocalizations={languages.length - 1}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedTextInput.displayName = 'LocalizedTextInput';\n\nLocalizedTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;\n\nLocalizedTextInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedTextInput.getId = getId;\n\nLocalizedTextInput.getName = getName;\n\nLocalizedTextInput.createLocalizedString = createLocalizedString;\n\nLocalizedTextInput.isEmpty = isEmpty;\n\nLocalizedTextInput.omitEmptyTranslations = omitEmptyTranslations;\n\nLocalizedTextInput.isTouched = isTouched;\n\nexport default LocalizedTextInput;\n"]} */",
|
|
91
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-text-input.tsx"],"names":[],"mappings":"AA8Mc","file":"localized-text-input.tsx","sourcesContent":["import {\n  type FocusEventHandler,\n  type ChangeEventHandler,\n  type ReactNode,\n  useCallback,\n} from 'react';\nimport { FormattedMessage } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport { ErrorMessage } from '@commercetools-uikit/messages';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  isTouched,\n  omitEmptyTranslations,\n  isEmpty,\n  createLocalizedString,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { createSequentialId, warning } from '@commercetools-uikit/utils';\nimport TextInput from '@commercetools-uikit/text-input';\nimport {\n  LocalizedInputToggle,\n  messagesLocalizedInput,\n} from '@commercetools-uikit/input-utils';\nimport {\n  getLocalizedInputStyles,\n  getLanguageLabelStyles,\n} from './localized-text-input.styles';\n\ninterface HTMLLocalizedInputElement extends HTMLInputElement {\n  language: string;\n}\n\nexport type TLocalizedTextInputProps = {\n  id?: string;\n  name?: string;\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   *   then input doesn't accept a \"languages\" prop, instead all possible\n  languages have to exist (with empty or filled strings) on the value:\n    { en: 'foo', de: '', es: '' }\n   */\n  value: Record<string, string>;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  /**\n   * Specifies which language will be shown in case the `LocalizedTextInput` is collapsed.\n   */\n  selectedLanguage: string;\n  /**\n   * Called when any field is blurred. Is called with the `event` of that field.\n   */\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Will hide the language expansion controls when set to `true`. All languages will be shown when set to `true`.\n   */\n  hideLanguageExpansionControls?: boolean;\n  /**\n   * Controls whether one or all languages are visible by default\n   */\n  defaultExpandLanguages?: boolean;\n  /**\n   * Focus the input field on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Disables all input fields.\n   */\n  isDisabled?: boolean;\n  /**\n   * Disables all input fields and shows them in read-only mode.\n   */\n  isReadOnly?: boolean;\n  /**\n   * Placeholders for each language. Object of the same shape as `value`.\n   */\n  placeholder?: Record<string, string>;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 1\n    | 2\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Will apply the error state to each input without showing any error message.\n   */\n  hasError?: boolean;\n  /**\n   * Indicates the input field has a warning\n   */\n  hasWarning?: boolean;\n  /**\n   * Used to show errors underneath the inputs of specific currencies. Pass an object whose key is a currency and whose value is the error to show for that key.\n   */\n  errors?: Record<string, string>;\n  /**\n   * A map of warnings.\n   */\n  warnings?: Record<string, ReactNode>;\n};\n\nexport type TLocalizedInputProps = {\n  /**\n   * Used as prefix of HTML `id` property. Each input field id will have the language as a suffix (`${idPrefix}.${lang}`), e.g. `foo.en`. You can use the static `LocalizedTextInput.getId(idPrefix, language)` to create this id string, e.g. for labels.\n   */\n  id?: string;\n  /**\n   * Used as HTML `name` property for each input field. Each input field name will have the language as a suffix (`${namePrefix}.${lang}`), e.g. `foo.en`\n   */\n  name?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  value: string;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  language: string;\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  isAutofocussed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasError?: boolean;\n  /**\n   * Indicates the input field has a warning\n   */\n  hasWarning?: boolean;\n  /**\n   * HTML node to display warning\n   */\n  warning?: ReactNode;\n  placeholder?: string;\n};\n\nconst sequentialId = createSequentialId('localized-text-input-');\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the language to the target.\n      // That way the users of LocalizedTextInput's onChange can read\n      // event.target.language and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the language from\n      // the input name anymore.\n      //\n      event.target.language = props.language;\n      onChange?.(event);\n    },\n    [props.language, onChange]\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  return (\n    <div\n      key={props.language}\n      css={css`\n        width: 100%;\n        position: relative;\n        display: flex;\n      `}\n    >\n      <label htmlFor={props.id} css={getLanguageLabelStyles(props)}>\n        {props.language.toUpperCase()}\n      </label>\n      <TextInput\n        {...props}\n        onChange={handleChange}\n        css={getLocalizedInputStyles}\n      />\n    </div>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst RequiredValueErrorMessage = () => (\n  <ErrorMessage>\n    <FormattedMessage {...messagesLocalizedInput.missingRequiredField} />\n  </ErrorMessage>\n);\n\nRequiredValueErrorMessage.displayName = 'RequiredValueErrorMessage';\n\nconst LocalizedTextInput = (props: TLocalizedTextInputProps) => {\n  const defaultExpansionState =\n    props.hideLanguageExpansionControls ||\n    props.defaultExpandLanguages || // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areLanguagesExpanded, toggleLanguages] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedInputToggle = useCallback(\n    () => toggleLanguages(),\n    [toggleLanguages]\n  );\n\n  const languages = sortLanguages(\n    props.selectedLanguage,\n    Object.keys(props.value)\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingLanguages =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);\n\n  const hasWarningInRemainingLanguages =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);\n\n  if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (!areLanguagesExpanded) {\n      toggleLanguages();\n    }\n  }\n\n  const shouldRenderLanguagesButton =\n    languages.length > 1 && !props.hideLanguageExpansionControls;\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  if (props.hideLanguageExpansionControls) {\n    warning(\n      typeof props.defaultExpandLanguages !== 'boolean',\n      'LocalizedTextInput: \"defaultExpandLanguages\" does not have any effect when \"hideLanguageExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack>\n          {languages.map((language, index) => {\n            const isFirstLanguage = index === 0;\n            if (!isFirstLanguage && !areLanguagesExpanded) return null;\n\n            return (\n              <div key={language}>\n                <Stack scale=\"xs\">\n                  <LocalizedInput\n                    autoComplete={props.autoComplete}\n                    id={LocalizedTextInput.getId(id, language)}\n                    name={LocalizedTextInput.getName(props.name, language)}\n                    value={props.value[language]}\n                    onChange={props.onChange}\n                    language={language}\n                    placeholder={\n                      props.placeholder\n                        ? props.placeholder[language]\n                        : undefined\n                    }\n                    onBlur={props.onBlur}\n                    onFocus={props.onFocus}\n                    isAutofocussed={index === 0 && props.isAutofocussed}\n                    isDisabled={props.isDisabled}\n                    isReadOnly={props.isReadOnly}\n                    hasError={Boolean(\n                      props.hasError || (props.errors && props.errors[language])\n                    )}\n                    hasWarning={Boolean(\n                      props.hasWarning ||\n                        (props.warnings && props.warnings[language])\n                    )}\n                    warning={props.warnings && props.warnings[language]}\n                    {...createLocalizedDataAttributes(props, language)}\n                    /* ARIA */\n                    aria-invalid={props['aria-invalid']}\n                    aria-errormessage={props['aria-errormessage']}\n                  />\n                  {props.errors && props.errors[language]}\n                  {props.warnings && props.warnings[language]}\n                </Stack>\n              </div>\n            );\n          })}\n        </Stack>\n        {shouldRenderLanguagesButton && (\n          <LocalizedInputToggle\n            isOpen={areLanguagesExpanded}\n            onClick={onLocalizedInputToggle}\n            isDisabled={\n              areLanguagesExpanded &&\n              Boolean(\n                hasErrorInRemainingLanguages || hasWarningInRemainingLanguages\n              )\n            }\n            remainingLocalizations={languages.length - 1}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedTextInput.displayName = 'LocalizedTextInput';\n\nLocalizedTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;\n\nLocalizedTextInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedTextInput.getId = getId;\n\nLocalizedTextInput.getName = getName;\n\nLocalizedTextInput.createLocalizedString = createLocalizedString;\n\nLocalizedTextInput.isEmpty = isEmpty;\n\nLocalizedTextInput.omitEmptyTranslations = omitEmptyTranslations;\n\nLocalizedTextInput.isTouched = isTouched;\n\nexport default LocalizedTextInput;\n"]} */",
|
|
92
92
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
93
93
|
};
|
|
94
94
|
const LocalizedInput = props => {
|
|
@@ -136,6 +136,8 @@ LocalizedInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
|
136
136
|
isDisabled: _pt__default["default"].bool,
|
|
137
137
|
isReadOnly: _pt__default["default"].bool,
|
|
138
138
|
hasError: _pt__default["default"].bool,
|
|
139
|
+
hasWarning: _pt__default["default"].bool,
|
|
140
|
+
warning: _pt__default["default"].node,
|
|
139
141
|
placeholder: _pt__default["default"].string
|
|
140
142
|
} : {};
|
|
141
143
|
LocalizedInput.displayName = 'LocalizedInput';
|
|
@@ -155,10 +157,11 @@ const LocalizedTextInput = props => {
|
|
|
155
157
|
const languages = localizedUtils.sortLanguages(props.selectedLanguage, _Object$keys__default["default"](props.value));
|
|
156
158
|
const id = hooks.useFieldId(props.id, sequentialId);
|
|
157
159
|
const hasErrorInRemainingLanguages = props.hasError || localizedUtils.getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);
|
|
158
|
-
|
|
160
|
+
const hasWarningInRemainingLanguages = props.hasWarning || localizedUtils.getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);
|
|
161
|
+
if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {
|
|
159
162
|
// this update within render replaces the old `getDerivedStateFromProps` functionality
|
|
160
163
|
// https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops
|
|
161
|
-
if (
|
|
164
|
+
if (!areLanguagesExpanded) {
|
|
162
165
|
toggleLanguages();
|
|
163
166
|
}
|
|
164
167
|
}
|
|
@@ -193,19 +196,21 @@ const LocalizedTextInput = props => {
|
|
|
193
196
|
isAutofocussed: index === 0 && props.isAutofocussed,
|
|
194
197
|
isDisabled: props.isDisabled,
|
|
195
198
|
isReadOnly: props.isReadOnly,
|
|
196
|
-
hasError: Boolean(props.hasError || props.errors && props.errors[language])
|
|
199
|
+
hasError: Boolean(props.hasError || props.errors && props.errors[language]),
|
|
200
|
+
hasWarning: Boolean(props.hasWarning || props.warnings && props.warnings[language]),
|
|
201
|
+
warning: props.warnings && props.warnings[language]
|
|
197
202
|
}, localizedUtils.createLocalizedDataAttributes(props, language)), {}, {
|
|
198
203
|
/* ARIA */
|
|
199
204
|
"aria-invalid": props['aria-invalid'],
|
|
200
205
|
"aria-errormessage": props['aria-errormessage']
|
|
201
|
-
})), props.errors && props.errors[language]]
|
|
206
|
+
})), props.errors && props.errors[language], props.warnings && props.warnings[language]]
|
|
202
207
|
})
|
|
203
208
|
}, language);
|
|
204
209
|
})
|
|
205
210
|
}), shouldRenderLanguagesButton && jsxRuntime.jsx(inputUtils.LocalizedInputToggle, {
|
|
206
211
|
isOpen: areLanguagesExpanded,
|
|
207
212
|
onClick: onLocalizedInputToggle,
|
|
208
|
-
isDisabled: areLanguagesExpanded && hasErrorInRemainingLanguages,
|
|
213
|
+
isDisabled: areLanguagesExpanded && Boolean(hasErrorInRemainingLanguages || hasWarningInRemainingLanguages),
|
|
209
214
|
remainingLocalizations: languages.length - 1
|
|
210
215
|
})]
|
|
211
216
|
})
|
|
@@ -230,7 +235,9 @@ LocalizedTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
|
230
235
|
placeholder: _pt__default["default"].objectOf(_pt__default["default"].string),
|
|
231
236
|
horizontalConstraint: _pt__default["default"].oneOf([1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
232
237
|
hasError: _pt__default["default"].bool,
|
|
233
|
-
|
|
238
|
+
hasWarning: _pt__default["default"].bool,
|
|
239
|
+
errors: _pt__default["default"].objectOf(_pt__default["default"].string),
|
|
240
|
+
warnings: _pt__default["default"].objectOf(_pt__default["default"].node)
|
|
234
241
|
} : {};
|
|
235
242
|
LocalizedTextInput.displayName = 'LocalizedTextInput';
|
|
236
243
|
LocalizedTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;
|
|
@@ -246,7 +253,7 @@ LocalizedTextInput.isTouched = localizedUtils.isTouched;
|
|
|
246
253
|
var LocalizedTextInput$1 = LocalizedTextInput;
|
|
247
254
|
|
|
248
255
|
// NOTE: This string will be replaced on build time with the package version.
|
|
249
|
-
var version = "
|
|
256
|
+
var version = "17.0.1";
|
|
250
257
|
|
|
251
258
|
exports["default"] = LocalizedTextInput$1;
|
|
252
259
|
exports.version = version;
|
|
@@ -126,10 +126,11 @@ const LocalizedTextInput = props => {
|
|
|
126
126
|
const languages = localizedUtils.sortLanguages(props.selectedLanguage, _Object$keys__default["default"](props.value));
|
|
127
127
|
const id = hooks.useFieldId(props.id, sequentialId);
|
|
128
128
|
const hasErrorInRemainingLanguages = props.hasError || localizedUtils.getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);
|
|
129
|
-
|
|
129
|
+
const hasWarningInRemainingLanguages = props.hasWarning || localizedUtils.getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);
|
|
130
|
+
if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {
|
|
130
131
|
// this update within render replaces the old `getDerivedStateFromProps` functionality
|
|
131
132
|
// https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops
|
|
132
|
-
if (
|
|
133
|
+
if (!areLanguagesExpanded) {
|
|
133
134
|
toggleLanguages();
|
|
134
135
|
}
|
|
135
136
|
}
|
|
@@ -160,19 +161,21 @@ const LocalizedTextInput = props => {
|
|
|
160
161
|
isAutofocussed: index === 0 && props.isAutofocussed,
|
|
161
162
|
isDisabled: props.isDisabled,
|
|
162
163
|
isReadOnly: props.isReadOnly,
|
|
163
|
-
hasError: Boolean(props.hasError || props.errors && props.errors[language])
|
|
164
|
+
hasError: Boolean(props.hasError || props.errors && props.errors[language]),
|
|
165
|
+
hasWarning: Boolean(props.hasWarning || props.warnings && props.warnings[language]),
|
|
166
|
+
warning: props.warnings && props.warnings[language]
|
|
164
167
|
}, localizedUtils.createLocalizedDataAttributes(props, language)), {}, {
|
|
165
168
|
/* ARIA */
|
|
166
169
|
"aria-invalid": props['aria-invalid'],
|
|
167
170
|
"aria-errormessage": props['aria-errormessage']
|
|
168
|
-
})), props.errors && props.errors[language]]
|
|
171
|
+
})), props.errors && props.errors[language], props.warnings && props.warnings[language]]
|
|
169
172
|
})
|
|
170
173
|
}, language);
|
|
171
174
|
})
|
|
172
175
|
}), shouldRenderLanguagesButton && jsxRuntime.jsx(inputUtils.LocalizedInputToggle, {
|
|
173
176
|
isOpen: areLanguagesExpanded,
|
|
174
177
|
onClick: onLocalizedInputToggle,
|
|
175
|
-
isDisabled: areLanguagesExpanded && hasErrorInRemainingLanguages,
|
|
178
|
+
isDisabled: areLanguagesExpanded && Boolean(hasErrorInRemainingLanguages || hasWarningInRemainingLanguages),
|
|
176
179
|
remainingLocalizations: languages.length - 1
|
|
177
180
|
})]
|
|
178
181
|
})
|
|
@@ -193,7 +196,7 @@ LocalizedTextInput.isTouched = localizedUtils.isTouched;
|
|
|
193
196
|
var LocalizedTextInput$1 = LocalizedTextInput;
|
|
194
197
|
|
|
195
198
|
// NOTE: This string will be replaced on build time with the package version.
|
|
196
|
-
var version = "
|
|
199
|
+
var version = "17.0.1";
|
|
197
200
|
|
|
198
201
|
exports["default"] = LocalizedTextInput$1;
|
|
199
202
|
exports.version = version;
|
|
@@ -17,7 +17,7 @@ import { useToggleState, useFieldId } from '@commercetools-uikit/hooks';
|
|
|
17
17
|
import { ErrorMessage } from '@commercetools-uikit/messages';
|
|
18
18
|
import Stack from '@commercetools-uikit/spacings-stack';
|
|
19
19
|
import Constraints from '@commercetools-uikit/constraints';
|
|
20
|
-
import { sortLanguages, getHasErrorOnRemainingLanguages, createLocalizedDataAttributes, getId, getName, createLocalizedString, isEmpty, omitEmptyTranslations, isTouched } from '@commercetools-uikit/localized-utils';
|
|
20
|
+
import { sortLanguages, getHasErrorOnRemainingLanguages, getHasWarningOnRemainingLanguages, createLocalizedDataAttributes, getId, getName, createLocalizedString, isEmpty, omitEmptyTranslations, isTouched } from '@commercetools-uikit/localized-utils';
|
|
21
21
|
import { createSequentialId, warning } from '@commercetools-uikit/utils';
|
|
22
22
|
import TextInput from '@commercetools-uikit/text-input';
|
|
23
23
|
import { LocalizedInputToggle, messagesLocalizedInput } from '@commercetools-uikit/input-utils';
|
|
@@ -68,7 +68,7 @@ var _ref = process.env.NODE_ENV === "production" ? {
|
|
|
68
68
|
} : {
|
|
69
69
|
name: "1ysn02y-LocalizedInput",
|
|
70
70
|
styles: "width:100%;position:relative;display:flex;label:LocalizedInput;",
|
|
71
|
-
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-text-input.tsx"],"names":[],"mappings":"AA4Lc","file":"localized-text-input.tsx","sourcesContent":["import {\n  type FocusEventHandler,\n  type ChangeEventHandler,\n  useCallback,\n} from 'react';\nimport { FormattedMessage } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport { ErrorMessage } from '@commercetools-uikit/messages';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  isTouched,\n  omitEmptyTranslations,\n  isEmpty,\n  createLocalizedString,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { createSequentialId, warning } from '@commercetools-uikit/utils';\nimport TextInput from '@commercetools-uikit/text-input';\nimport {\n  LocalizedInputToggle,\n  messagesLocalizedInput,\n} from '@commercetools-uikit/input-utils';\nimport {\n  getLocalizedInputStyles,\n  getLanguageLabelStyles,\n} from './localized-text-input.styles';\n\ninterface HTMLLocalizedInputElement extends HTMLInputElement {\n  language: string;\n}\n\nexport type TLocalizedTextInputProps = {\n  id?: string;\n  name?: string;\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   *   then input doesn't accept a \"languages\" prop, instead all possible\n  languages have to exist (with empty or filled strings) on the value:\n    { en: 'foo', de: '', es: '' }\n   */\n  value: Record<string, string>;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  /**\n   * Specifies which language will be shown in case the `LocalizedTextInput` is collapsed.\n   */\n  selectedLanguage: string;\n  /**\n   * Called when any field is blurred. Is called with the `event` of that field.\n   */\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Will hide the language expansion controls when set to `true`. All languages will be shown when set to `true`.\n   */\n  hideLanguageExpansionControls?: boolean;\n  /**\n   * Controls whether one or all languages are visible by default\n   */\n  defaultExpandLanguages?: boolean;\n  /**\n   * Focus the input field on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Disables all input fields.\n   */\n  isDisabled?: boolean;\n  /**\n   * Disables all input fields and shows them in read-only mode.\n   */\n  isReadOnly?: boolean;\n  /**\n   * Placeholders for each language. Object of the same shape as `value`.\n   */\n  placeholder?: Record<string, string>;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 1\n    | 2\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Will apply the error state to each input without showing any error message.\n   */\n  hasError?: boolean;\n  /**\n   * Used to show errors underneath the inputs of specific currencies. Pass an object whose key is a currency and whose value is the error to show for that key.\n   */\n  errors?: Record<string, string>;\n};\n\nexport type TLocalizedInputProps = {\n  /**\n   * Used as prefix of HTML `id` property. Each input field id will have the language as a suffix (`${idPrefix}.${lang}`), e.g. `foo.en`. You can use the static `LocalizedTextInput.getId(idPrefix, language)` to create this id string, e.g. for labels.\n   */\n  id?: string;\n  /**\n   * Used as HTML `name` property for each input field. Each input field name will have the language as a suffix (`${namePrefix}.${lang}`), e.g. `foo.en`\n   */\n  name?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  value: string;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  language: string;\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  isAutofocussed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasError?: boolean;\n  placeholder?: string;\n};\n\nconst sequentialId = createSequentialId('localized-text-input-');\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the language to the target.\n      // That way the users of LocalizedTextInput's onChange can read\n      // event.target.language and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the language from\n      // the input name anymore.\n      //\n      event.target.language = props.language;\n      onChange?.(event);\n    },\n    [props.language, onChange]\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  return (\n    <div\n      key={props.language}\n      css={css`\n        width: 100%;\n        position: relative;\n        display: flex;\n      `}\n    >\n      <label htmlFor={props.id} css={getLanguageLabelStyles(props)}>\n        {props.language.toUpperCase()}\n      </label>\n      <TextInput\n        {...props}\n        onChange={handleChange}\n        css={getLocalizedInputStyles}\n      />\n    </div>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst RequiredValueErrorMessage = () => (\n  <ErrorMessage>\n    <FormattedMessage {...messagesLocalizedInput.missingRequiredField} />\n  </ErrorMessage>\n);\n\nRequiredValueErrorMessage.displayName = 'RequiredValueErrorMessage';\n\nconst LocalizedTextInput = (props: TLocalizedTextInputProps) => {\n  const defaultExpansionState =\n    props.hideLanguageExpansionControls ||\n    props.defaultExpandLanguages || // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areLanguagesExpanded, toggleLanguages] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedInputToggle = useCallback(\n    () => toggleLanguages(),\n    [toggleLanguages]\n  );\n\n  const languages = sortLanguages(\n    props.selectedLanguage,\n    Object.keys(props.value)\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingLanguages =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);\n\n  if (hasErrorInRemainingLanguages) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (hasErrorInRemainingLanguages !== areLanguagesExpanded) {\n      toggleLanguages();\n    }\n  }\n\n  const shouldRenderLanguagesButton =\n    languages.length > 1 && !props.hideLanguageExpansionControls;\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  if (props.hideLanguageExpansionControls) {\n    warning(\n      typeof props.defaultExpandLanguages !== 'boolean',\n      'LocalizedTextInput: \"defaultExpandLanguages\" does not have any effect when \"hideLanguageExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack>\n          {languages.map((language, index) => {\n            const isFirstLanguage = index === 0;\n            if (!isFirstLanguage && !areLanguagesExpanded) return null;\n\n            return (\n              <div key={language}>\n                <Stack scale=\"xs\">\n                  <LocalizedInput\n                    autoComplete={props.autoComplete}\n                    id={LocalizedTextInput.getId(id, language)}\n                    name={LocalizedTextInput.getName(props.name, language)}\n                    value={props.value[language]}\n                    onChange={props.onChange}\n                    language={language}\n                    placeholder={\n                      props.placeholder\n                        ? props.placeholder[language]\n                        : undefined\n                    }\n                    onBlur={props.onBlur}\n                    onFocus={props.onFocus}\n                    isAutofocussed={index === 0 && props.isAutofocussed}\n                    isDisabled={props.isDisabled}\n                    isReadOnly={props.isReadOnly}\n                    hasError={Boolean(\n                      props.hasError || (props.errors && props.errors[language])\n                    )}\n                    {...createLocalizedDataAttributes(props, language)}\n                    /* ARIA */\n                    aria-invalid={props['aria-invalid']}\n                    aria-errormessage={props['aria-errormessage']}\n                  />\n                  {props.errors && props.errors[language]}\n                </Stack>\n              </div>\n            );\n          })}\n        </Stack>\n        {shouldRenderLanguagesButton && (\n          <LocalizedInputToggle\n            isOpen={areLanguagesExpanded}\n            onClick={onLocalizedInputToggle}\n            isDisabled={areLanguagesExpanded && hasErrorInRemainingLanguages}\n            remainingLocalizations={languages.length - 1}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedTextInput.displayName = 'LocalizedTextInput';\n\nLocalizedTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;\n\nLocalizedTextInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedTextInput.getId = getId;\n\nLocalizedTextInput.getName = getName;\n\nLocalizedTextInput.createLocalizedString = createLocalizedString;\n\nLocalizedTextInput.isEmpty = isEmpty;\n\nLocalizedTextInput.omitEmptyTranslations = omitEmptyTranslations;\n\nLocalizedTextInput.isTouched = isTouched;\n\nexport default LocalizedTextInput;\n"]} */",
|
|
71
|
+
map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["localized-text-input.tsx"],"names":[],"mappings":"AA8Mc","file":"localized-text-input.tsx","sourcesContent":["import {\n  type FocusEventHandler,\n  type ChangeEventHandler,\n  type ReactNode,\n  useCallback,\n} from 'react';\nimport { FormattedMessage } from 'react-intl';\nimport { css } from '@emotion/react';\nimport { useFieldId, useToggleState } from '@commercetools-uikit/hooks';\nimport { ErrorMessage } from '@commercetools-uikit/messages';\nimport Stack from '@commercetools-uikit/spacings-stack';\nimport Constraints from '@commercetools-uikit/constraints';\nimport {\n  sortLanguages,\n  createLocalizedDataAttributes,\n  getHasErrorOnRemainingLanguages,\n  getHasWarningOnRemainingLanguages,\n  isTouched,\n  omitEmptyTranslations,\n  isEmpty,\n  createLocalizedString,\n  getId,\n  getName,\n} from '@commercetools-uikit/localized-utils';\nimport { createSequentialId, warning } from '@commercetools-uikit/utils';\nimport TextInput from '@commercetools-uikit/text-input';\nimport {\n  LocalizedInputToggle,\n  messagesLocalizedInput,\n} from '@commercetools-uikit/input-utils';\nimport {\n  getLocalizedInputStyles,\n  getLanguageLabelStyles,\n} from './localized-text-input.styles';\n\ninterface HTMLLocalizedInputElement extends HTMLInputElement {\n  language: string;\n}\n\nexport type TLocalizedTextInputProps = {\n  id?: string;\n  name?: string;\n  autoComplete?: string;\n  /**\n   * Indicate if the value entered in the input is invalid.\n   */\n  'aria-invalid'?: boolean;\n  /**\n   * HTML ID of an element containing an error message related to the input.\n   */\n  'aria-errormessage'?: string;\n  /**\n   *   then input doesn't accept a \"languages\" prop, instead all possible\n  languages have to exist (with empty or filled strings) on the value:\n    { en: 'foo', de: '', es: '' }\n   */\n  value: Record<string, string>;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  /**\n   * Specifies which language will be shown in case the `LocalizedTextInput` is collapsed.\n   */\n  selectedLanguage: string;\n  /**\n   * Called when any field is blurred. Is called with the `event` of that field.\n   */\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Called when any field is focussed. Is called with the `event` of that field.\n   */\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  /**\n   * Will hide the language expansion controls when set to `true`. All languages will be shown when set to `true`.\n   */\n  hideLanguageExpansionControls?: boolean;\n  /**\n   * Controls whether one or all languages are visible by default\n   */\n  defaultExpandLanguages?: boolean;\n  /**\n   * Focus the input field on initial render\n   */\n  isAutofocussed?: boolean;\n  /**\n   * Disables all input fields.\n   */\n  isDisabled?: boolean;\n  /**\n   * Disables all input fields and shows them in read-only mode.\n   */\n  isReadOnly?: boolean;\n  /**\n   * Placeholders for each language. Object of the same shape as `value`.\n   */\n  placeholder?: Record<string, string>;\n  /**\n   * Horizontal size limit of the input fields.\n   */\n  horizontalConstraint?:\n    | 1\n    | 2\n    | 3\n    | 4\n    | 5\n    | 6\n    | 7\n    | 8\n    | 9\n    | 10\n    | 11\n    | 12\n    | 13\n    | 14\n    | 15\n    | 16\n    | 'scale'\n    | 'auto';\n  /**\n   * Will apply the error state to each input without showing any error message.\n   */\n  hasError?: boolean;\n  /**\n   * Indicates the input field has a warning\n   */\n  hasWarning?: boolean;\n  /**\n   * Used to show errors underneath the inputs of specific currencies. Pass an object whose key is a currency and whose value is the error to show for that key.\n   */\n  errors?: Record<string, string>;\n  /**\n   * A map of warnings.\n   */\n  warnings?: Record<string, ReactNode>;\n};\n\nexport type TLocalizedInputProps = {\n  /**\n   * Used as prefix of HTML `id` property. Each input field id will have the language as a suffix (`${idPrefix}.${lang}`), e.g. `foo.en`. You can use the static `LocalizedTextInput.getId(idPrefix, language)` to create this id string, e.g. for labels.\n   */\n  id?: string;\n  /**\n   * Used as HTML `name` property for each input field. Each input field name will have the language as a suffix (`${namePrefix}.${lang}`), e.g. `foo.en`\n   */\n  name?: string;\n  /**\n   * Used as HTML `autocomplete` property\n   */\n  autoComplete?: string;\n  value: string;\n  /**\n   * Gets called when any input is changed. Is called with the change event of the changed input.\n   */\n  onChange?: ChangeEventHandler<HTMLLocalizedInputElement>;\n  language: string;\n  onBlur?: FocusEventHandler<HTMLInputElement>;\n  onFocus?: FocusEventHandler<HTMLInputElement>;\n  isAutofocussed?: boolean;\n  isDisabled?: boolean;\n  isReadOnly?: boolean;\n  hasError?: boolean;\n  /**\n   * Indicates the input field has a warning\n   */\n  hasWarning?: boolean;\n  /**\n   * HTML node to display warning\n   */\n  warning?: ReactNode;\n  placeholder?: string;\n};\n\nconst sequentialId = createSequentialId('localized-text-input-');\n\nconst LocalizedInput = (props: TLocalizedInputProps) => {\n  const { onChange } = props;\n  const handleChange = useCallback(\n    (event) => {\n      // We manipulate the event to add the language to the target.\n      // That way the users of LocalizedTextInput's onChange can read\n      // event.target.language and event.target.value to determine the next value.\n      //\n      // We only need this information for the story, the MC application code will\n      // never need to access the information in such an inconvenient way, as\n      // Formik can deal with a name like \"foo.en\" and sets the value correctly.\n      // We can't use this as we aren't guaranteed a name in the story as the user\n      // might clear it using the knob, and then we can't parse the language from\n      // the input name anymore.\n      //\n      event.target.language = props.language;\n      onChange?.(event);\n    },\n    [props.language, onChange]\n  );\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  return (\n    <div\n      key={props.language}\n      css={css`\n        width: 100%;\n        position: relative;\n        display: flex;\n      `}\n    >\n      <label htmlFor={props.id} css={getLanguageLabelStyles(props)}>\n        {props.language.toUpperCase()}\n      </label>\n      <TextInput\n        {...props}\n        onChange={handleChange}\n        css={getLocalizedInputStyles}\n      />\n    </div>\n  );\n};\n\nLocalizedInput.displayName = 'LocalizedInput';\n\nconst RequiredValueErrorMessage = () => (\n  <ErrorMessage>\n    <FormattedMessage {...messagesLocalizedInput.missingRequiredField} />\n  </ErrorMessage>\n);\n\nRequiredValueErrorMessage.displayName = 'RequiredValueErrorMessage';\n\nconst LocalizedTextInput = (props: TLocalizedTextInputProps) => {\n  const defaultExpansionState =\n    props.hideLanguageExpansionControls ||\n    props.defaultExpandLanguages || // default to `false`, because useToggleState defaults to `true`\n    false;\n\n  const [areLanguagesExpanded, toggleLanguages] = useToggleState(\n    defaultExpansionState\n  );\n\n  const onLocalizedInputToggle = useCallback(\n    () => toggleLanguages(),\n    [toggleLanguages]\n  );\n\n  const languages = sortLanguages(\n    props.selectedLanguage,\n    Object.keys(props.value)\n  );\n\n  const id = useFieldId(props.id, sequentialId);\n\n  const hasErrorInRemainingLanguages =\n    props.hasError ||\n    getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);\n\n  const hasWarningInRemainingLanguages =\n    props.hasWarning ||\n    getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);\n\n  if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {\n    // this update within render replaces the old `getDerivedStateFromProps` functionality\n    // https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops\n    if (!areLanguagesExpanded) {\n      toggleLanguages();\n    }\n  }\n\n  const shouldRenderLanguagesButton =\n    languages.length > 1 && !props.hideLanguageExpansionControls;\n\n  if (!props.isReadOnly) {\n    warning(\n      typeof props.onChange === 'function',\n      'LocalizedTextInput: \"onChange\" is required when isReadOnly is not true'\n    );\n  }\n\n  if (props.hideLanguageExpansionControls) {\n    warning(\n      typeof props.defaultExpandLanguages !== 'boolean',\n      'LocalizedTextInput: \"defaultExpandLanguages\" does not have any effect when \"hideLanguageExpansionControls\" is set.'\n    );\n  }\n\n  return (\n    <Constraints.Horizontal max={props.horizontalConstraint}>\n      <Stack scale=\"xs\">\n        <Stack>\n          {languages.map((language, index) => {\n            const isFirstLanguage = index === 0;\n            if (!isFirstLanguage && !areLanguagesExpanded) return null;\n\n            return (\n              <div key={language}>\n                <Stack scale=\"xs\">\n                  <LocalizedInput\n                    autoComplete={props.autoComplete}\n                    id={LocalizedTextInput.getId(id, language)}\n                    name={LocalizedTextInput.getName(props.name, language)}\n                    value={props.value[language]}\n                    onChange={props.onChange}\n                    language={language}\n                    placeholder={\n                      props.placeholder\n                        ? props.placeholder[language]\n                        : undefined\n                    }\n                    onBlur={props.onBlur}\n                    onFocus={props.onFocus}\n                    isAutofocussed={index === 0 && props.isAutofocussed}\n                    isDisabled={props.isDisabled}\n                    isReadOnly={props.isReadOnly}\n                    hasError={Boolean(\n                      props.hasError || (props.errors && props.errors[language])\n                    )}\n                    hasWarning={Boolean(\n                      props.hasWarning ||\n                        (props.warnings && props.warnings[language])\n                    )}\n                    warning={props.warnings && props.warnings[language]}\n                    {...createLocalizedDataAttributes(props, language)}\n                    /* ARIA */\n                    aria-invalid={props['aria-invalid']}\n                    aria-errormessage={props['aria-errormessage']}\n                  />\n                  {props.errors && props.errors[language]}\n                  {props.warnings && props.warnings[language]}\n                </Stack>\n              </div>\n            );\n          })}\n        </Stack>\n        {shouldRenderLanguagesButton && (\n          <LocalizedInputToggle\n            isOpen={areLanguagesExpanded}\n            onClick={onLocalizedInputToggle}\n            isDisabled={\n              areLanguagesExpanded &&\n              Boolean(\n                hasErrorInRemainingLanguages || hasWarningInRemainingLanguages\n              )\n            }\n            remainingLocalizations={languages.length - 1}\n          />\n        )}\n      </Stack>\n    </Constraints.Horizontal>\n  );\n};\n\nLocalizedTextInput.displayName = 'LocalizedTextInput';\n\nLocalizedTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;\n\nLocalizedTextInput.defaultProps = {\n  horizontalConstraint: 'scale',\n};\n\nLocalizedTextInput.getId = getId;\n\nLocalizedTextInput.getName = getName;\n\nLocalizedTextInput.createLocalizedString = createLocalizedString;\n\nLocalizedTextInput.isEmpty = isEmpty;\n\nLocalizedTextInput.omitEmptyTranslations = omitEmptyTranslations;\n\nLocalizedTextInput.isTouched = isTouched;\n\nexport default LocalizedTextInput;\n"]} */",
|
|
72
72
|
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
73
73
|
};
|
|
74
74
|
const LocalizedInput = props => {
|
|
@@ -116,6 +116,8 @@ LocalizedInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
|
116
116
|
isDisabled: _pt.bool,
|
|
117
117
|
isReadOnly: _pt.bool,
|
|
118
118
|
hasError: _pt.bool,
|
|
119
|
+
hasWarning: _pt.bool,
|
|
120
|
+
warning: _pt.node,
|
|
119
121
|
placeholder: _pt.string
|
|
120
122
|
} : {};
|
|
121
123
|
LocalizedInput.displayName = 'LocalizedInput';
|
|
@@ -135,10 +137,11 @@ const LocalizedTextInput = props => {
|
|
|
135
137
|
const languages = sortLanguages(props.selectedLanguage, _Object$keys(props.value));
|
|
136
138
|
const id = useFieldId(props.id, sequentialId);
|
|
137
139
|
const hasErrorInRemainingLanguages = props.hasError || getHasErrorOnRemainingLanguages(props.errors, props.selectedLanguage);
|
|
138
|
-
|
|
140
|
+
const hasWarningInRemainingLanguages = props.hasWarning || getHasWarningOnRemainingLanguages(props.warnings, props.selectedLanguage);
|
|
141
|
+
if (hasErrorInRemainingLanguages || hasWarningInRemainingLanguages) {
|
|
139
142
|
// this update within render replaces the old `getDerivedStateFromProps` functionality
|
|
140
143
|
// https://reactjs.org/docs/hooks-faq.html#how-do-i-implement-getderivedstatefromprops
|
|
141
|
-
if (
|
|
144
|
+
if (!areLanguagesExpanded) {
|
|
142
145
|
toggleLanguages();
|
|
143
146
|
}
|
|
144
147
|
}
|
|
@@ -173,19 +176,21 @@ const LocalizedTextInput = props => {
|
|
|
173
176
|
isAutofocussed: index === 0 && props.isAutofocussed,
|
|
174
177
|
isDisabled: props.isDisabled,
|
|
175
178
|
isReadOnly: props.isReadOnly,
|
|
176
|
-
hasError: Boolean(props.hasError || props.errors && props.errors[language])
|
|
179
|
+
hasError: Boolean(props.hasError || props.errors && props.errors[language]),
|
|
180
|
+
hasWarning: Boolean(props.hasWarning || props.warnings && props.warnings[language]),
|
|
181
|
+
warning: props.warnings && props.warnings[language]
|
|
177
182
|
}, createLocalizedDataAttributes(props, language)), {}, {
|
|
178
183
|
/* ARIA */
|
|
179
184
|
"aria-invalid": props['aria-invalid'],
|
|
180
185
|
"aria-errormessage": props['aria-errormessage']
|
|
181
|
-
})), props.errors && props.errors[language]]
|
|
186
|
+
})), props.errors && props.errors[language], props.warnings && props.warnings[language]]
|
|
182
187
|
})
|
|
183
188
|
}, language);
|
|
184
189
|
})
|
|
185
190
|
}), shouldRenderLanguagesButton && jsx(LocalizedInputToggle, {
|
|
186
191
|
isOpen: areLanguagesExpanded,
|
|
187
192
|
onClick: onLocalizedInputToggle,
|
|
188
|
-
isDisabled: areLanguagesExpanded && hasErrorInRemainingLanguages,
|
|
193
|
+
isDisabled: areLanguagesExpanded && Boolean(hasErrorInRemainingLanguages || hasWarningInRemainingLanguages),
|
|
189
194
|
remainingLocalizations: languages.length - 1
|
|
190
195
|
})]
|
|
191
196
|
})
|
|
@@ -210,7 +215,9 @@ LocalizedTextInput.propTypes = process.env.NODE_ENV !== "production" ? {
|
|
|
210
215
|
placeholder: _pt.objectOf(_pt.string),
|
|
211
216
|
horizontalConstraint: _pt.oneOf([1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 'scale', 'auto']),
|
|
212
217
|
hasError: _pt.bool,
|
|
213
|
-
|
|
218
|
+
hasWarning: _pt.bool,
|
|
219
|
+
errors: _pt.objectOf(_pt.string),
|
|
220
|
+
warnings: _pt.objectOf(_pt.node)
|
|
214
221
|
} : {};
|
|
215
222
|
LocalizedTextInput.displayName = 'LocalizedTextInput';
|
|
216
223
|
LocalizedTextInput.RequiredValueErrorMessage = RequiredValueErrorMessage;
|
|
@@ -226,6 +233,6 @@ LocalizedTextInput.isTouched = isTouched;
|
|
|
226
233
|
var LocalizedTextInput$1 = LocalizedTextInput;
|
|
227
234
|
|
|
228
235
|
// NOTE: This string will be replaced on build time with the package version.
|
|
229
|
-
var version = "
|
|
236
|
+
var version = "17.0.1";
|
|
230
237
|
|
|
231
238
|
export { LocalizedTextInput$1 as default, version };
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { type FocusEventHandler, type ChangeEventHandler } from 'react';
|
|
1
|
+
import { type FocusEventHandler, type ChangeEventHandler, type ReactNode } from 'react';
|
|
2
2
|
interface HTMLLocalizedInputElement extends HTMLInputElement {
|
|
3
3
|
language: string;
|
|
4
4
|
}
|
|
@@ -21,7 +21,9 @@ export type TLocalizedTextInputProps = {
|
|
|
21
21
|
placeholder?: Record<string, string>;
|
|
22
22
|
horizontalConstraint?: 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 'scale' | 'auto';
|
|
23
23
|
hasError?: boolean;
|
|
24
|
+
hasWarning?: boolean;
|
|
24
25
|
errors?: Record<string, string>;
|
|
26
|
+
warnings?: Record<string, ReactNode>;
|
|
25
27
|
};
|
|
26
28
|
export type TLocalizedInputProps = {
|
|
27
29
|
id?: string;
|
|
@@ -36,6 +38,8 @@ export type TLocalizedInputProps = {
|
|
|
36
38
|
isDisabled?: boolean;
|
|
37
39
|
isReadOnly?: boolean;
|
|
38
40
|
hasError?: boolean;
|
|
41
|
+
hasWarning?: boolean;
|
|
42
|
+
warning?: ReactNode;
|
|
39
43
|
placeholder?: string;
|
|
40
44
|
};
|
|
41
45
|
declare const LocalizedTextInput: {
|
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@commercetools-uikit/localized-text-input",
|
|
3
3
|
"description": "A controlled text input component for localized single-line strings with validation states.",
|
|
4
|
-
"version": "
|
|
4
|
+
"version": "17.0.1",
|
|
5
5
|
"bugs": "https://github.com/commercetools/ui-kit/issues",
|
|
6
6
|
"repository": {
|
|
7
7
|
"type": "git",
|
|
@@ -21,18 +21,18 @@
|
|
|
21
21
|
"dependencies": {
|
|
22
22
|
"@babel/runtime": "^7.20.13",
|
|
23
23
|
"@babel/runtime-corejs3": "^7.20.13",
|
|
24
|
-
"@commercetools-uikit/constraints": "
|
|
25
|
-
"@commercetools-uikit/design-system": "
|
|
26
|
-
"@commercetools-uikit/flat-button": "
|
|
27
|
-
"@commercetools-uikit/hooks": "
|
|
28
|
-
"@commercetools-uikit/icons": "
|
|
29
|
-
"@commercetools-uikit/input-utils": "
|
|
30
|
-
"@commercetools-uikit/localized-utils": "
|
|
31
|
-
"@commercetools-uikit/messages": "
|
|
32
|
-
"@commercetools-uikit/spacings-stack": "
|
|
33
|
-
"@commercetools-uikit/text": "
|
|
34
|
-
"@commercetools-uikit/text-input": "
|
|
35
|
-
"@commercetools-uikit/utils": "
|
|
24
|
+
"@commercetools-uikit/constraints": "17.0.1",
|
|
25
|
+
"@commercetools-uikit/design-system": "17.0.1",
|
|
26
|
+
"@commercetools-uikit/flat-button": "17.0.1",
|
|
27
|
+
"@commercetools-uikit/hooks": "17.0.1",
|
|
28
|
+
"@commercetools-uikit/icons": "17.0.1",
|
|
29
|
+
"@commercetools-uikit/input-utils": "17.0.1",
|
|
30
|
+
"@commercetools-uikit/localized-utils": "17.0.1",
|
|
31
|
+
"@commercetools-uikit/messages": "17.0.1",
|
|
32
|
+
"@commercetools-uikit/spacings-stack": "17.0.1",
|
|
33
|
+
"@commercetools-uikit/text": "17.0.1",
|
|
34
|
+
"@commercetools-uikit/text-input": "17.0.1",
|
|
35
|
+
"@commercetools-uikit/utils": "17.0.1",
|
|
36
36
|
"@emotion/react": "^11.10.5",
|
|
37
37
|
"@emotion/styled": "^11.10.5",
|
|
38
38
|
"prop-types": "15.8.1"
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"commercetools-uikit-localized-text-input.cjs.d.ts","sourceRoot":"","sources":["./declarations/src/index.d.ts"],"names":[],"mappings":"AAAA"}
|