@ultraviolet/ui 3.0.0-beta.11 → 3.0.0-beta.13

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.
Files changed (70) hide show
  1. package/dist/components/ActionBar/styles.css.cjs +0 -1
  2. package/dist/components/ActionBar/styles.css.js +0 -1
  3. package/dist/components/BarStack/styles.css.cjs +0 -1
  4. package/dist/components/BarStack/styles.css.js +0 -1
  5. package/dist/components/BarStack/variables.css.cjs +1 -0
  6. package/dist/components/BarStack/variables.css.js +1 -0
  7. package/dist/components/Bullet/constants.d.ts +10 -0
  8. package/dist/components/Bullet/index.cjs +9 -75
  9. package/dist/components/Bullet/index.d.ts +5 -14
  10. package/dist/components/Bullet/index.js +10 -74
  11. package/dist/components/Bullet/styles.css.cjs +6 -0
  12. package/dist/components/Bullet/styles.css.d.ts +23 -0
  13. package/dist/components/Bullet/styles.css.js +6 -0
  14. package/dist/components/Button/constants.cjs +0 -1
  15. package/dist/components/Button/constants.js +0 -1
  16. package/dist/components/Chip/styles.css.cjs +1 -0
  17. package/dist/components/Chip/styles.css.js +1 -0
  18. package/dist/components/Expandable/index.cjs +5 -17
  19. package/dist/components/Expandable/index.d.ts +0 -6
  20. package/dist/components/Expandable/index.js +6 -16
  21. package/dist/components/Expandable/styles.css.cjs +7 -0
  22. package/dist/components/Expandable/styles.css.d.ts +2 -0
  23. package/dist/components/Expandable/styles.css.js +7 -0
  24. package/dist/components/GlobalAlert/index.cjs +3 -30
  25. package/dist/components/GlobalAlert/index.js +3 -28
  26. package/dist/components/GlobalAlert/styles.css.cjs +7 -0
  27. package/dist/components/GlobalAlert/styles.css.d.ts +2 -0
  28. package/dist/components/GlobalAlert/styles.css.js +7 -0
  29. package/dist/components/Link/constants.d.ts +6 -0
  30. package/dist/components/Link/index.cjs +11 -114
  31. package/dist/components/Link/index.d.ts +1 -6
  32. package/dist/components/Link/index.js +12 -113
  33. package/dist/components/Link/styles.css.cjs +14 -0
  34. package/dist/components/Link/styles.css.d.ts +78 -0
  35. package/dist/components/Link/styles.css.js +14 -0
  36. package/dist/components/ProgressBar/constants.d.ts +1 -0
  37. package/dist/components/ProgressBar/index.cjs +7 -77
  38. package/dist/components/ProgressBar/index.d.ts +2 -2
  39. package/dist/components/ProgressBar/index.js +7 -75
  40. package/dist/components/ProgressBar/styles.css.cjs +11 -0
  41. package/dist/components/ProgressBar/styles.css.d.ts +4 -0
  42. package/dist/components/ProgressBar/styles.css.js +11 -0
  43. package/dist/components/ProgressBar/variables.css.cjs +5 -0
  44. package/dist/components/ProgressBar/variables.css.d.ts +1 -0
  45. package/dist/components/ProgressBar/variables.css.js +5 -0
  46. package/dist/components/Radio/index.cjs +30 -136
  47. package/dist/components/Radio/index.d.ts +0 -49
  48. package/dist/components/Radio/index.js +31 -135
  49. package/dist/components/Radio/styles.css.cjs +21 -0
  50. package/dist/components/Radio/styles.css.d.ts +9 -0
  51. package/dist/components/Radio/styles.css.js +21 -0
  52. package/dist/components/SelectableCard/index.cjs +29 -28
  53. package/dist/components/SelectableCard/index.js +21 -20
  54. package/dist/components/SelectableCardOptionGroup/components/Option.cjs +8 -8
  55. package/dist/components/SelectableCardOptionGroup/components/Option.js +6 -6
  56. package/dist/components/StepList/index.cjs +1 -1
  57. package/dist/components/StepList/index.js +1 -1
  58. package/dist/components/VerificationCode/constants.d.ts +12 -0
  59. package/dist/components/VerificationCode/index.cjs +5 -88
  60. package/dist/components/VerificationCode/index.d.ts +1 -1
  61. package/dist/components/VerificationCode/index.js +5 -86
  62. package/dist/components/VerificationCode/styles.css.cjs +9 -0
  63. package/dist/components/VerificationCode/styles.css.d.ts +3 -0
  64. package/dist/components/VerificationCode/styles.css.js +9 -0
  65. package/dist/theme/index.cjs +0 -4
  66. package/dist/theme/index.js +0 -4
  67. package/dist/ui.css +1 -1
  68. package/package.json +3 -3
  69. package/dist/utils/capitalize.cjs +0 -3
  70. package/dist/utils/capitalize.js +0 -4
@@ -2,93 +2,10 @@
2
2
  "use strict";
3
3
  Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
4
4
  const jsxRuntime = require("@emotion/react/jsx-runtime");
5
- const _styled = require("@emotion/styled/base");
6
5
  const react = require("react");
7
6
  const index = require("../Label/index.cjs");
8
7
  const index$1 = require("../Text/index.cjs");
9
- const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
10
- const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
11
- const SIZE_HEIGHT = {
12
- large: "600",
13
- medium: "500",
14
- small: "400",
15
- xlarge: "800"
16
- };
17
- const SIZE_WIDTH = {
18
- large: "500",
19
- medium: "400",
20
- small: "300",
21
- xlarge: "700"
22
- };
23
- const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.env.NODE_ENV === "production" ? {
24
- shouldForwardProp: (prop) => !["inputSize"].includes(prop),
25
- target: "e1a2bx9q1"
26
- } : {
27
- shouldForwardProp: (prop) => !["inputSize"].includes(prop),
28
- target: "e1a2bx9q1",
29
- label: "StyledInput"
30
- })("background:", ({
31
- theme
32
- }) => theme.colors.neutral.background, ";color:", ({
33
- theme
34
- }) => theme.colors.neutral.text, ";", ({
35
- inputSize,
36
- theme
37
- }) => {
38
- if (inputSize === "small") {
39
- return `
40
- font-size: ${theme.typography.caption.fontSize};
41
- font-weight: ${theme.typography.caption.weight};
42
- `;
43
- }
44
- return `
45
- font-size: ${theme.typography.body.fontSize};
46
- font-weight: ${theme.typography.body.weight};
47
- `;
48
- }, " text-align:center;border-radius:", ({
49
- theme
50
- }) => theme.radii.default, ";margin-right:", ({
51
- theme
52
- }) => theme.space["1"], ";width:", ({
53
- inputSize,
54
- theme
55
- }) => theme.sizing[SIZE_WIDTH[inputSize]], ";height:", ({
56
- inputSize,
57
- theme
58
- }) => theme.sizing[SIZE_HEIGHT[inputSize]], ";outline-style:none;transition:border-color 0.2s ease,box-shadow 0.2s ease;border:solid 1px ", ({
59
- theme
60
- }) => theme.colors.neutral.border, ";&[aria-invalid='true']{border-color:", ({
61
- theme
62
- }) => theme.colors.danger.border, ";}&[data-success='true']{border-color:", ({
63
- theme
64
- }) => theme.colors.success.border, ";}&:hover,&:focus{border-color:", ({
65
- theme
66
- }) => theme.colors.primary.borderHover, ";&[aria-invalid='true']{border-color:", ({
67
- theme
68
- }) => theme.colors.danger.borderHover, ";}&[data-success='true']{border-color:", ({
69
- theme
70
- }) => theme.colors.success.borderHover, ";}}&:focus{box-shadow:", ({
71
- theme: {
72
- shadows
73
- }
74
- }) => shadows.focusPrimary, ";}&:last-child{margin-right:0;}&::placeholder{color:", ({
75
- disabled,
76
- theme
77
- }) => disabled ? theme.colors.neutral.textWeakDisabled : theme.colors.neutral.textWeak, ";}&:disabled{cursor:not-allowed;background:", ({
78
- theme
79
- }) => theme.colors.neutral.backgroundDisabled, ";color:", ({
80
- theme
81
- }) => theme.colors.neutral.textDisabled, ";border:solid 1px ", ({
82
- theme
83
- }) => theme.colors.neutral.borderDisabled, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx"],"names":[],"mappings":"AAoCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  large: '600',\n  medium: '500',\n  small: '400',\n  xlarge: '800',\n} as const\n\nconst SIZE_WIDTH = {\n  large: '500',\n  medium: '400',\n  small: '300',\n  xlarge: '700',\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\n  ])\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace': {\n          event.preventDefault()\n\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (copiedValue: string) =>\n          // Replace non number char with empty char when type is number\n          type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = structuredClone(vals)\n\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size === 'xlarge' ? 'large' : size}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            aria-invalid={!!error}\n            aria-label={`${ariaLabel} ${index}`}\n            autoComplete=\"off\"\n            css={[inputStyle]}\n            data-success={!!success}\n            data-testid={index}\n            disabled={disabled}\n            id={`${inputId || uniqueId}-${index}`}\n            inputSize={size}\n            key={`field-${index}`}\n            onChange={inputOnChange(index)}\n            onFocus={inputOnFocus}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            placeholder={placeholder?.[index] ?? ''}\n            ref={inputRefs[index]}\n            required={required}\n            type={type === 'number' ? 'tel' : type}\n            value={value}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          disabled={disabled}\n          prominence={!error && !success ? 'weak' : 'default'}\n          sentiment={sentiment}\n          variant=\"caption\"\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
84
- const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.env.NODE_ENV === "production" ? {
85
- target: "e1a2bx9q0"
86
- } : {
87
- target: "e1a2bx9q0",
88
- label: "FieldSet"
89
- })("border:none;padding:0;margin:0;display:flex;flex-direction:column;gap:", ({
90
- theme
91
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx"],"names":[],"mappings":"AA4GgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  large: '600',\n  medium: '500',\n  small: '400',\n  xlarge: '800',\n} as const\n\nconst SIZE_WIDTH = {\n  large: '500',\n  medium: '400',\n  small: '300',\n  xlarge: '700',\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\n  ])\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace': {\n          event.preventDefault()\n\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (copiedValue: string) =>\n          // Replace non number char with empty char when type is number\n          type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = structuredClone(vals)\n\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size === 'xlarge' ? 'large' : size}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            aria-invalid={!!error}\n            aria-label={`${ariaLabel} ${index}`}\n            autoComplete=\"off\"\n            css={[inputStyle]}\n            data-success={!!success}\n            data-testid={index}\n            disabled={disabled}\n            id={`${inputId || uniqueId}-${index}`}\n            inputSize={size}\n            key={`field-${index}`}\n            onChange={inputOnChange(index)}\n            onFocus={inputOnFocus}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            placeholder={placeholder?.[index] ?? ''}\n            ref={inputRefs[index]}\n            required={required}\n            type={type === 'number' ? 'tel' : type}\n            value={value}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          disabled={disabled}\n          prominence={!error && !success ? 'weak' : 'default'}\n          sentiment={sentiment}\n          variant=\"caption\"\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
8
+ const styles_css = require("./styles.css.cjs");
92
9
  const DEFAULT_ON_FUNCTION = () => {
93
10
  };
94
11
  const inputOnFocus = (event) => event.target.select();
@@ -99,7 +16,6 @@ const VerificationCode = ({
99
16
  fields = 4,
100
17
  initialValue = "",
101
18
  inputId,
102
- inputStyle = "",
103
19
  size = "large",
104
20
  onChange = DEFAULT_ON_FUNCTION,
105
21
  onComplete = DEFAULT_ON_FUNCTION,
@@ -114,6 +30,7 @@ const VerificationCode = ({
114
30
  success
115
31
  }) => {
116
32
  const uniqueId = react.useId();
33
+ const id = inputId || uniqueId;
117
34
  const valuesArray = Object.assign(new Array(fields).fill(""), [...initialValue.substring(0, fields)]);
118
35
  const [values, setValues] = react.useState(valuesArray);
119
36
  const inputRefs = Array.from({
@@ -220,9 +137,9 @@ const VerificationCode = ({
220
137
  }
221
138
  return "neutral";
222
139
  }, [error, success]);
223
- return /* @__PURE__ */ jsxRuntime.jsxs(FieldSet, { className, "data-testid": dataTestId, children: [
224
- label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsx(index.Label, { labelDescription, required, size: size === "xlarge" ? "large" : size, children: label }) : null,
225
- /* @__PURE__ */ jsxRuntime.jsx("div", { children: values.map((value, index2) => /* @__PURE__ */ jsxRuntime.jsx(StyledInput, { "aria-invalid": !!error, "aria-label": `${ariaLabel} ${index2}`, autoComplete: "off", css: [inputStyle, process.env.NODE_ENV === "production" ? "" : ";label:VerificationCode;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx"],"names":[],"mappings":"AAgWY","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  large: '600',\n  medium: '500',\n  small: '400',\n  xlarge: '800',\n} as const\n\nconst SIZE_WIDTH = {\n  large: '500',\n  medium: '400',\n  small: '300',\n  xlarge: '700',\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\n  ])\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace': {\n          event.preventDefault()\n\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (copiedValue: string) =>\n          // Replace non number char with empty char when type is number\n          type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = structuredClone(vals)\n\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size === 'xlarge' ? 'large' : size}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            aria-invalid={!!error}\n            aria-label={`${ariaLabel} ${index}`}\n            autoComplete=\"off\"\n            css={[inputStyle]}\n            data-success={!!success}\n            data-testid={index}\n            disabled={disabled}\n            id={`${inputId || uniqueId}-${index}`}\n            inputSize={size}\n            key={`field-${index}`}\n            onChange={inputOnChange(index)}\n            onFocus={inputOnFocus}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            placeholder={placeholder?.[index] ?? ''}\n            ref={inputRefs[index]}\n            required={required}\n            type={type === 'number' ? 'tel' : type}\n            value={value}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          disabled={disabled}\n          prominence={!error && !success ? 'weak' : 'default'}\n          sentiment={sentiment}\n          variant=\"caption\"\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"], "data-success": !!success, "data-testid": index2, disabled, id: `${inputId || uniqueId}-${index2}`, inputSize: size, onChange: inputOnChange(index2), onFocus: inputOnFocus, onKeyDown: inputOnKeyDown(index2), onPaste: inputOnPaste(index2), pattern: type === "number" ? "[0-9]*" : void 0, placeholder: placeholder?.[index2] ?? "", ref: inputRefs[index2], required, type: type === "number" ? "tel" : type, value }, `field-${index2}`)) }),
140
+ return /* @__PURE__ */ jsxRuntime.jsxs("fieldset", { className: `${className ? `${className} ` : ""}${styles_css.filedSetClass}`, "data-testid": dataTestId, children: [
141
+ label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsx(index.Label, { htmlFor: `${id}-0`, id: `${id}-label`, labelDescription, required, size: size === "xlarge" ? "large" : size, children: label }) : null,
142
+ /* @__PURE__ */ jsxRuntime.jsx("div", { children: values.map((value, index2) => /* @__PURE__ */ jsxRuntime.jsx("input", { "aria-invalid": !!error, "aria-label": `${ariaLabel} ${index2}`, autoComplete: "off", className: `${styles_css.inputSizes[size]} ${styles_css.inputClass}`, "data-success": !!success, "data-testid": index2, disabled, id: `${id}-${index2}`, onChange: inputOnChange(index2), onFocus: inputOnFocus, onKeyDown: inputOnKeyDown(index2), onPaste: inputOnPaste(index2), pattern: type === "number" ? "[0-9]*" : void 0, placeholder: placeholder?.[index2] ?? "", ref: inputRefs[index2], required, type: type === "number" ? "tel" : type, value }, `field-${index2}`)) }),
226
143
  error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", disabled, prominence: !error && !success ? "weak" : "default", sentiment, variant: "caption", children: error || success || helper }) : null,
227
144
  !error && !success && typeof helper !== "string" && helper ? helper : null
228
145
  ] });
@@ -37,5 +37,5 @@ type VerificationCodeProps = {
37
37
  /**
38
38
  * Verification code allows you to enter a code in multiple fields (4 by default).
39
39
  */
40
- export declare const VerificationCode: ({ disabled, className, error, fields, initialValue, inputId, inputStyle, size, onChange, onComplete, placeholder, required, type, "data-testid": dataTestId, "aria-label": ariaLabel, label, labelDescription, helper, success, }: VerificationCodeProps) => import("@emotion/react/jsx-runtime").JSX.Element;
40
+ export declare const VerificationCode: ({ disabled, className, error, fields, initialValue, inputId, size, onChange, onComplete, placeholder, required, type, "data-testid": dataTestId, "aria-label": ariaLabel, label, labelDescription, helper, success, }: VerificationCodeProps) => import("@emotion/react/jsx-runtime").JSX.Element;
41
41
  export {};
@@ -1,90 +1,9 @@
1
1
  "use client";
2
2
  import { jsxs, jsx } from "@emotion/react/jsx-runtime";
3
- import _styled from "@emotion/styled/base";
4
3
  import { useId, useState, createRef, useMemo } from "react";
5
4
  import { Label } from "../Label/index.js";
6
5
  import { Text } from "../Text/index.js";
7
- const SIZE_HEIGHT = {
8
- large: "600",
9
- medium: "500",
10
- small: "400",
11
- xlarge: "800"
12
- };
13
- const SIZE_WIDTH = {
14
- large: "500",
15
- medium: "400",
16
- small: "300",
17
- xlarge: "700"
18
- };
19
- const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
20
- shouldForwardProp: (prop) => !["inputSize"].includes(prop),
21
- target: "e1a2bx9q1"
22
- } : {
23
- shouldForwardProp: (prop) => !["inputSize"].includes(prop),
24
- target: "e1a2bx9q1",
25
- label: "StyledInput"
26
- })("background:", ({
27
- theme
28
- }) => theme.colors.neutral.background, ";color:", ({
29
- theme
30
- }) => theme.colors.neutral.text, ";", ({
31
- inputSize,
32
- theme
33
- }) => {
34
- if (inputSize === "small") {
35
- return `
36
- font-size: ${theme.typography.caption.fontSize};
37
- font-weight: ${theme.typography.caption.weight};
38
- `;
39
- }
40
- return `
41
- font-size: ${theme.typography.body.fontSize};
42
- font-weight: ${theme.typography.body.weight};
43
- `;
44
- }, " text-align:center;border-radius:", ({
45
- theme
46
- }) => theme.radii.default, ";margin-right:", ({
47
- theme
48
- }) => theme.space["1"], ";width:", ({
49
- inputSize,
50
- theme
51
- }) => theme.sizing[SIZE_WIDTH[inputSize]], ";height:", ({
52
- inputSize,
53
- theme
54
- }) => theme.sizing[SIZE_HEIGHT[inputSize]], ";outline-style:none;transition:border-color 0.2s ease,box-shadow 0.2s ease;border:solid 1px ", ({
55
- theme
56
- }) => theme.colors.neutral.border, ";&[aria-invalid='true']{border-color:", ({
57
- theme
58
- }) => theme.colors.danger.border, ";}&[data-success='true']{border-color:", ({
59
- theme
60
- }) => theme.colors.success.border, ";}&:hover,&:focus{border-color:", ({
61
- theme
62
- }) => theme.colors.primary.borderHover, ";&[aria-invalid='true']{border-color:", ({
63
- theme
64
- }) => theme.colors.danger.borderHover, ";}&[data-success='true']{border-color:", ({
65
- theme
66
- }) => theme.colors.success.borderHover, ";}}&:focus{box-shadow:", ({
67
- theme: {
68
- shadows
69
- }
70
- }) => shadows.focusPrimary, ";}&:last-child{margin-right:0;}&::placeholder{color:", ({
71
- disabled,
72
- theme
73
- }) => disabled ? theme.colors.neutral.textWeakDisabled : theme.colors.neutral.textWeak, ";}&:disabled{cursor:not-allowed;background:", ({
74
- theme
75
- }) => theme.colors.neutral.backgroundDisabled, ";color:", ({
76
- theme
77
- }) => theme.colors.neutral.textDisabled, ";border:solid 1px ", ({
78
- theme
79
- }) => theme.colors.neutral.borderDisabled, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx"],"names":[],"mappings":"AAoCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  large: '600',\n  medium: '500',\n  small: '400',\n  xlarge: '800',\n} as const\n\nconst SIZE_WIDTH = {\n  large: '500',\n  medium: '400',\n  small: '300',\n  xlarge: '700',\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\n  ])\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace': {\n          event.preventDefault()\n\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (copiedValue: string) =>\n          // Replace non number char with empty char when type is number\n          type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = structuredClone(vals)\n\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size === 'xlarge' ? 'large' : size}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            aria-invalid={!!error}\n            aria-label={`${ariaLabel} ${index}`}\n            autoComplete=\"off\"\n            css={[inputStyle]}\n            data-success={!!success}\n            data-testid={index}\n            disabled={disabled}\n            id={`${inputId || uniqueId}-${index}`}\n            inputSize={size}\n            key={`field-${index}`}\n            onChange={inputOnChange(index)}\n            onFocus={inputOnFocus}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            placeholder={placeholder?.[index] ?? ''}\n            ref={inputRefs[index]}\n            required={required}\n            type={type === 'number' ? 'tel' : type}\n            value={value}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          disabled={disabled}\n          prominence={!error && !success ? 'weak' : 'default'}\n          sentiment={sentiment}\n          variant=\"caption\"\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
80
- const FieldSet = /* @__PURE__ */ _styled("fieldset", process.env.NODE_ENV === "production" ? {
81
- target: "e1a2bx9q0"
82
- } : {
83
- target: "e1a2bx9q0",
84
- label: "FieldSet"
85
- })("border:none;padding:0;margin:0;display:flex;flex-direction:column;gap:", ({
86
- theme
87
- }) => theme.space["0.5"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx"],"names":[],"mappings":"AA4GgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  large: '600',\n  medium: '500',\n  small: '400',\n  xlarge: '800',\n} as const\n\nconst SIZE_WIDTH = {\n  large: '500',\n  medium: '400',\n  small: '300',\n  xlarge: '700',\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\n  ])\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace': {\n          event.preventDefault()\n\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (copiedValue: string) =>\n          // Replace non number char with empty char when type is number\n          type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = structuredClone(vals)\n\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size === 'xlarge' ? 'large' : size}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            aria-invalid={!!error}\n            aria-label={`${ariaLabel} ${index}`}\n            autoComplete=\"off\"\n            css={[inputStyle]}\n            data-success={!!success}\n            data-testid={index}\n            disabled={disabled}\n            id={`${inputId || uniqueId}-${index}`}\n            inputSize={size}\n            key={`field-${index}`}\n            onChange={inputOnChange(index)}\n            onFocus={inputOnFocus}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            placeholder={placeholder?.[index] ?? ''}\n            ref={inputRefs[index]}\n            required={required}\n            type={type === 'number' ? 'tel' : type}\n            value={value}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          disabled={disabled}\n          prominence={!error && !success ? 'weak' : 'default'}\n          sentiment={sentiment}\n          variant=\"caption\"\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"));
6
+ import { filedSetClass, inputSizes, inputClass } from "./styles.css.js";
88
7
  const DEFAULT_ON_FUNCTION = () => {
89
8
  };
90
9
  const inputOnFocus = (event) => event.target.select();
@@ -95,7 +14,6 @@ const VerificationCode = ({
95
14
  fields = 4,
96
15
  initialValue = "",
97
16
  inputId,
98
- inputStyle = "",
99
17
  size = "large",
100
18
  onChange = DEFAULT_ON_FUNCTION,
101
19
  onComplete = DEFAULT_ON_FUNCTION,
@@ -110,6 +28,7 @@ const VerificationCode = ({
110
28
  success
111
29
  }) => {
112
30
  const uniqueId = useId();
31
+ const id = inputId || uniqueId;
113
32
  const valuesArray = Object.assign(new Array(fields).fill(""), [...initialValue.substring(0, fields)]);
114
33
  const [values, setValues] = useState(valuesArray);
115
34
  const inputRefs = Array.from({
@@ -216,9 +135,9 @@ const VerificationCode = ({
216
135
  }
217
136
  return "neutral";
218
137
  }, [error, success]);
219
- return /* @__PURE__ */ jsxs(FieldSet, { className, "data-testid": dataTestId, children: [
220
- label || labelDescription ? /* @__PURE__ */ jsx(Label, { labelDescription, required, size: size === "xlarge" ? "large" : size, children: label }) : null,
221
- /* @__PURE__ */ jsx("div", { children: values.map((value, index) => /* @__PURE__ */ jsx(StyledInput, { "aria-invalid": !!error, "aria-label": `${ariaLabel} ${index}`, autoComplete: "off", css: [inputStyle, process.env.NODE_ENV === "production" ? "" : ";label:VerificationCode;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx"],"names":[],"mappings":"AAgWY","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Label } from '../Label'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  large: '600',\n  medium: '500',\n  small: '400',\n  xlarge: '800',\n} as const\n\nconst SIZE_WIDTH = {\n  large: '500',\n  medium: '400',\n  small: '300',\n  xlarge: '700',\n} as const\n\nexport const verificationCodeSizes = Object.keys(SIZE_HEIGHT) as Size[]\n\nconst StyledInput = styled('input', {\n  shouldForwardProp: prop => !['inputSize'].includes(prop),\n})<{\n  inputSize: Size\n}>`\n  background: ${({ theme }) => theme.colors.neutral.background};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  ${({ inputSize, theme }) => {\n    if (inputSize === 'small') {\n      return `\n           font-size: ${theme.typography.caption.fontSize};\n           font-weight: ${theme.typography.caption.weight};\n        `\n    }\n\n    return `\n           font-size: ${theme.typography.body.fontSize};\n           font-weight: ${theme.typography.body.weight};\n         `\n  }}\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  margin-right: ${({ theme }) => theme.space['1']};\n  width: ${({ inputSize, theme }) => theme.sizing[SIZE_WIDTH[inputSize]]};\n  height: ${({ inputSize, theme }) => theme.sizing[SIZE_HEIGHT[inputSize]]};\n  outline-style: none;\n  transition:\n    border-color 0.2s ease,\n    box-shadow 0.2s ease;\n\n  border: solid 1px ${({ theme }) => theme.colors.neutral.border};\n\n  &[aria-invalid='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &:hover,\n  &:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n\n    &[aria-invalid='true'] {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n\n    &[data-success='true'] {\n      border-color: ${({ theme }) => theme.colors.success.borderHover};\n    }\n  }\n\n  &:focus {\n    box-shadow: ${({ theme: { shadows } }) => shadows.focusPrimary};\n  }\n\n  &:last-child {\n    margin-right: 0;\n  }\n\n  &::placeholder {\n    color: ${({ disabled, theme }) =>\n      disabled\n        ? theme.colors.neutral.textWeakDisabled\n        : theme.colors.neutral.textWeak};\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n    border: solid 1px ${({ theme }) => theme.colors.neutral.borderDisabled};\n  }\n`\n\nconst FieldSet = styled.fieldset`\n  border: none;\n  padding: 0;\n  margin: 0;\n  display: flex;\n  flex-direction: column;\n  gap: ${({ theme }) => theme.space['0.5']};\n`\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean | string\n  className?: string\n  /**\n   * Amount of field you want\n   */\n  fields?: number\n  initialValue?: string\n  inputId?: string\n  inputStyle?: string\n  size?: 'small' | 'medium' | 'large' | 'xlarge'\n  /**\n   * Triggered when a field change\n   */\n  onChange?: (data: unknown) => void\n  /**\n   * Triggered when all fields are completed\n   */\n  onComplete?: (data: unknown) => void\n  placeholder?: string\n  required?: boolean\n  /**\n   * Type of the fields\n   */\n  type?: 'text' | 'number'\n  'data-testid'?: string\n  'aria-label'?: string\n  label?: string\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  success?: boolean | string\n}\n\n/**\n * Verification code allows you to enter a code in multiple fields (4 by default).\n */\nexport const VerificationCode = ({\n  disabled = false,\n  className,\n  error = false,\n  fields = 4,\n  initialValue = '',\n  inputId,\n  inputStyle = '',\n  size = 'large',\n  onChange = DEFAULT_ON_FUNCTION,\n  onComplete = DEFAULT_ON_FUNCTION,\n  placeholder = '',\n  required = false,\n  type = 'number',\n  'data-testid': dataTestId,\n  'aria-label': ariaLabel = 'Verification code',\n  label,\n  labelDescription,\n  helper,\n  success,\n}: VerificationCodeProps) => {\n  const uniqueId = useId()\n  const valuesArray = Object.assign(new Array(fields).fill(''), [\n    ...initialValue.substring(0, fields),\n  ])\n  const [values, setValues] = useState<string[]>(valuesArray)\n\n  const inputRefs = Array.from({ length: fields }, () =>\n    createRef<HTMLInputElement>(),\n  )\n\n  const triggerChange = (inputValues: string[]) => {\n    const stringValue = inputValues.join('')\n    if (onChange) {\n      onChange(stringValue)\n    }\n    if (onComplete && stringValue.length >= fields) {\n      onComplete(stringValue)\n    }\n  }\n\n  const inputOnChange =\n    (index: number) => (event: ChangeEvent<HTMLInputElement>) => {\n      let { value } = event.target\n      if (type === 'number') {\n        value = event.target.value.replace(/[^\\d]/gi, '')\n      }\n      const newValues = [...values]\n\n      if (\n        value === '' ||\n        (type === 'number' && !new RegExp(event.target.pattern).test(value))\n      ) {\n        newValues[index] = ''\n        setValues(newValues)\n\n        return\n      }\n\n      const sanitizedValue = value[0] // in case more than 1 char, we just take the first one\n      newValues[index] = sanitizedValue ?? ''\n      setValues(newValues)\n      const nextIndex = Math.min(index + 1, fields - 1)\n      const next = inputRefs[nextIndex]\n\n      next?.current?.focus()\n\n      triggerChange(newValues)\n    }\n\n  const inputOnKeyDown =\n    (index: number): KeyboardEventHandler<HTMLInputElement> =>\n    event => {\n      const prevIndex = index - 1\n      const nextIndex = index + 1\n      const first = inputRefs[0]\n      const last = inputRefs[inputRefs.length - 1]\n      const prev = inputRefs[prevIndex]\n      const next = inputRefs[nextIndex]\n      const vals = [...values]\n\n      switch (event.key) {\n        case 'Backspace': {\n          event.preventDefault()\n\n          if (values[index]) {\n            vals[index] = ''\n            setValues(vals)\n            triggerChange(vals)\n          } else if (prev) {\n            vals[prevIndex] = ''\n            prev?.current?.focus()\n            setValues(vals)\n            triggerChange(vals)\n          }\n          break\n        }\n\n        case 'ArrowLeft': {\n          event.preventDefault()\n          prev?.current?.focus()\n          break\n        }\n\n        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\n        }\n\n        case 'ArrowDown': {\n          event.preventDefault()\n          last?.current?.focus()\n\n          break\n        }\n\n        default: {\n          break\n        }\n      }\n    }\n\n  const inputOnPaste =\n    (currentIndex: number): ClipboardEventHandler<HTMLInputElement> =>\n    event => {\n      event.preventDefault()\n      const pastedValue = [...event.clipboardData.getData('Text')].map(\n        (copiedValue: string) =>\n          // Replace non number char with empty char when type is number\n          type === 'number' ? copiedValue.replace(/[^\\d]/gi, '') : copiedValue,\n      )\n\n      // Trim array to avoid array overflow\n      pastedValue.splice(\n        fields - currentIndex < pastedValue.length\n          ? fields - currentIndex\n          : pastedValue.length,\n      )\n\n      setValues((vals: string[]) => {\n        const newArray = structuredClone(vals)\n\n        newArray.splice(currentIndex, pastedValue.length, ...pastedValue)\n\n        return newArray\n      })\n\n      // we select min value between the end of inputs and valid pasted chars\n      const nextIndex = Math.min(\n        currentIndex + pastedValue.filter(item => item !== '').length,\n        inputRefs.length - 1,\n      )\n      const next = inputRefs[nextIndex]\n      next?.current?.focus()\n      triggerChange(pastedValue)\n    }\n\n  const sentiment = useMemo(() => {\n    if (error) {\n      return 'danger'\n    }\n\n    if (success) {\n      return 'success'\n    }\n\n    return 'neutral'\n  }, [error, success])\n\n  return (\n    <FieldSet className={className} data-testid={dataTestId}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size === 'xlarge' ? 'large' : size}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            aria-invalid={!!error}\n            aria-label={`${ariaLabel} ${index}`}\n            autoComplete=\"off\"\n            css={[inputStyle]}\n            data-success={!!success}\n            data-testid={index}\n            disabled={disabled}\n            id={`${inputId || uniqueId}-${index}`}\n            inputSize={size}\n            key={`field-${index}`}\n            onChange={inputOnChange(index)}\n            onFocus={inputOnFocus}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            placeholder={placeholder?.[index] ?? ''}\n            ref={inputRefs[index]}\n            required={required}\n            type={type === 'number' ? 'tel' : type}\n            value={value}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          disabled={disabled}\n          prominence={!error && !success ? 'weak' : 'default'}\n          sentiment={sentiment}\n          variant=\"caption\"\n        >\n          {error || success || helper}\n        </Text>\n      ) : null}\n      {!error && !success && typeof helper !== 'string' && helper\n        ? helper\n        : null}\n    </FieldSet>\n  )\n}\n"]} */"], "data-success": !!success, "data-testid": index, disabled, id: `${inputId || uniqueId}-${index}`, inputSize: size, onChange: inputOnChange(index), onFocus: inputOnFocus, onKeyDown: inputOnKeyDown(index), onPaste: inputOnPaste(index), pattern: type === "number" ? "[0-9]*" : void 0, placeholder: placeholder?.[index] ?? "", ref: inputRefs[index], required, type: type === "number" ? "tel" : type, value }, `field-${index}`)) }),
138
+ return /* @__PURE__ */ jsxs("fieldset", { className: `${className ? `${className} ` : ""}${filedSetClass}`, "data-testid": dataTestId, children: [
139
+ label || labelDescription ? /* @__PURE__ */ jsx(Label, { htmlFor: `${id}-0`, id: `${id}-label`, labelDescription, required, size: size === "xlarge" ? "large" : size, children: label }) : null,
140
+ /* @__PURE__ */ jsx("div", { children: values.map((value, index) => /* @__PURE__ */ jsx("input", { "aria-invalid": !!error, "aria-label": `${ariaLabel} ${index}`, autoComplete: "off", className: `${inputSizes[size]} ${inputClass}`, "data-success": !!success, "data-testid": index, disabled, id: `${id}-${index}`, onChange: inputOnChange(index), onFocus: inputOnFocus, onKeyDown: inputOnKeyDown(index), onPaste: inputOnPaste(index), pattern: type === "number" ? "[0-9]*" : void 0, placeholder: placeholder?.[index] ?? "", ref: inputRefs[index], required, type: type === "number" ? "tel" : type, value }, `field-${index}`)) }),
222
141
  error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsx(Text, { as: "p", disabled, prominence: !error && !success ? "weak" : "default", sentiment, variant: "caption", children: error || success || helper }) : null,
223
142
  !error && !success && typeof helper !== "string" && helper ? helper : null
224
143
  ] });
@@ -0,0 +1,9 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ ;/* empty css */
4
+ var inputSizes = { small: "uv_1k9synd0", medium: "uv_1k9synd1", large: "uv_1k9synd2", xlarge: "uv_1k9synd3" };
5
+ var inputClass = "uv_1k9synd4";
6
+ var filedSetClass = "uv_1k9synd5";
7
+ exports.filedSetClass = filedSetClass;
8
+ exports.inputClass = inputClass;
9
+ exports.inputSizes = inputSizes;
@@ -0,0 +1,3 @@
1
+ export declare const inputSizes: Record<"small" | "large" | "medium" | "xlarge", string>;
2
+ export declare const inputClass: string;
3
+ export declare const filedSetClass: string;
@@ -0,0 +1,9 @@
1
+ /* empty css */
2
+ var inputSizes = { small: "uv_1k9synd0", medium: "uv_1k9synd1", large: "uv_1k9synd2", xlarge: "uv_1k9synd3" };
3
+ var inputClass = "uv_1k9synd4";
4
+ var filedSetClass = "uv_1k9synd5";
5
+ export {
6
+ filedSetClass,
7
+ inputClass,
8
+ inputSizes
9
+ };
@@ -13,8 +13,6 @@ const {
13
13
  breakpoints
14
14
  } = themes.consoleLightTheme;
15
15
  const extendTheme = (extendedTheme) => deepmerge__default.default(themes.consoleLightTheme, extendedTheme);
16
- const SENTIMENTS = ["primary", "secondary", "neutral", "success", "danger", "warning", "info"];
17
- const SENTIMENTS_WITHOUT_NEUTRAL = SENTIMENTS.filter((sentiment) => sentiment !== "neutral");
18
16
  Object.defineProperty(exports, "darkTheme", {
19
17
  enumerable: true,
20
18
  get: () => themes.consoleDarkTheme
@@ -23,8 +21,6 @@ Object.defineProperty(exports, "default", {
23
21
  enumerable: true,
24
22
  get: () => themes.consoleLightTheme
25
23
  });
26
- exports.SENTIMENTS = SENTIMENTS;
27
- exports.SENTIMENTS_WITHOUT_NEUTRAL = SENTIMENTS_WITHOUT_NEUTRAL;
28
24
  exports.colors = colors;
29
25
  exports.extendTheme = extendTheme;
30
26
  exports.radii = radii;
@@ -10,11 +10,7 @@ const {
10
10
  breakpoints
11
11
  } = consoleLightTheme;
12
12
  const extendTheme = (extendedTheme) => deepmerge(consoleLightTheme, extendedTheme);
13
- const SENTIMENTS = ["primary", "secondary", "neutral", "success", "danger", "warning", "info"];
14
- const SENTIMENTS_WITHOUT_NEUTRAL = SENTIMENTS.filter((sentiment) => sentiment !== "neutral");
15
13
  export {
16
- SENTIMENTS,
17
- SENTIMENTS_WITHOUT_NEUTRAL,
18
14
  colors,
19
15
  consoleDarkTheme as darkTheme,
20
16
  consoleLightTheme2 as default,