@ultraviolet/ui 1.84.0 → 1.84.2

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.
@@ -1,6 +1,9 @@
1
- import { jsx } from "@emotion/react/jsx-runtime";
1
+ import { jsxs, jsx } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
- import { useId, useState, createRef } from "react";
3
+ import { AsteriskIcon } from "@ultraviolet/icons";
4
+ import { useId, useState, createRef, useMemo } from "react";
5
+ import { Stack } from "../Stack/index.js";
6
+ import { Text } from "../Text/index.js";
4
7
  const SIZE_HEIGHT = {
5
8
  xlarge: "800",
6
9
  large: "600",
@@ -15,20 +18,16 @@ const SIZE_WIDTH = {
15
18
  };
16
19
  const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
17
20
  shouldForwardProp: (prop) => !["inputSize"].includes(prop),
18
- target: "e1a2bx9q0"
21
+ target: "e1a2bx9q1"
19
22
  } : {
20
23
  shouldForwardProp: (prop) => !["inputSize"].includes(prop),
21
- target: "e1a2bx9q0",
24
+ target: "e1a2bx9q1",
22
25
  label: "StyledInput"
23
26
  })("background:", ({
24
27
  theme
25
- }) => theme.colors.neutral.background, ";border:solid 1px ", ({
26
- "aria-invalid": error,
27
- theme
28
- }) => error ? theme.colors.danger.border : theme.colors.neutral.border, ";color:", ({
29
- "aria-invalid": error,
28
+ }) => theme.colors.neutral.background, ";color:", ({
30
29
  theme
31
- }) => error ? theme.colors.danger.text : theme.colors.neutral.text, ";", ({
30
+ }) => theme.colors.neutral.text, ";", ({
32
31
  inputSize,
33
32
  theme
34
33
  }) => {
@@ -52,31 +51,40 @@ const StyledInput = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "p
52
51
  }) => theme.sizing[SIZE_WIDTH[inputSize]], ";height:", ({
53
52
  inputSize,
54
53
  theme
55
- }) => theme.sizing[SIZE_HEIGHT[inputSize]], ";outline-style:none;transition:border-color 0.2s ease,box-shadow 0.2s ease;&:hover,&:focus{border-color:", ({
56
- "aria-invalid": error,
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:", ({
57
61
  theme
58
- }) => error ? theme.colors.danger.borderHover : theme.colors.primary.borderHover, ";}&:focus{box-shadow:", ({
59
- "aria-invalid": error,
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:", ({
60
67
  theme: {
61
68
  shadows
62
69
  }
63
- }) => error ? shadows.focusDanger : shadows.focusPrimary, ";}&:last-child{margin-right:0;}&::placeholder{color:", ({
70
+ }) => shadows.focusPrimary, ";}&:last-child{margin-right:0;}&::placeholder{color:", ({
64
71
  disabled,
65
72
  theme
66
- }) => disabled ? theme.colors.neutral.textWeakDisabled : theme.colors.neutral.textWeak, ";}", ({
67
- disabled,
68
- theme: {
69
- colors
70
- }
71
- }) => disabled && `cursor: default;
72
- background-color: ${colors.neutral.backgroundDisabled};
73
- border-color: ${colors.neutral.borderDisabled};
74
- color: ${colors.neutral.textDisabled};
75
-
76
- &:hover {
77
- border: ${colors.neutral.borderDisabled}
78
- }
79
- `, ";" + (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":"AA+BE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n} from 'react'\nimport { createRef, useId, useState } from 'react'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  border: solid 1px\n    ${({ 'aria-invalid': error, theme }) =>\n      error ? theme.colors.danger.border : theme.colors.neutral.border};\n  color: ${({ 'aria-invalid': error, theme }) =>\n    error ? theme.colors.danger.text : 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  &:hover,\n  &:focus {\n    border-color: ${({ 'aria-invalid': error, theme }) =>\n      error\n        ? theme.colors.danger.borderHover\n        : theme.colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ 'aria-invalid': error, theme: { shadows } }) =>\n      error ? shadows.focusDanger : 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, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};\n\n    &:hover {\n      border: ${colors.neutral.borderDisabled}\n    }\n    `}\n`\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean\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}\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\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}: 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        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\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  return (\n    <div className={className} data-testid={dataTestId}>\n      {values.map((value: string, index: number) => (\n        <StyledInput\n          css={[inputStyle]}\n          aria-invalid={error}\n          inputSize={size}\n          type={type === 'number' ? 'tel' : type}\n          pattern={type === 'number' ? '[0-9]*' : undefined}\n          key={`field-${index}`}\n          data-testid={index}\n          value={value}\n          id={`${inputId || uniqueId}-${index}`}\n          ref={inputRefs[index]}\n          onChange={inputOnChange(index)}\n          onKeyDown={inputOnKeyDown(index)}\n          onPaste={inputOnPaste(index)}\n          onFocus={inputOnFocus}\n          disabled={disabled}\n          required={required}\n          placeholder={placeholder?.[index] ?? ''}\n          aria-label={`${ariaLabel} ${index}`}\n        />\n      ))}\n    </div>\n  )\n}\n"]} */"));
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":"AAmCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\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        <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n          {label ? (\n            <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n              <Text\n                as=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${index}`}\n            data-testid={index}\n            value={value}\n            id={`${inputId || uniqueId}-${index}`}\n            ref={inputRefs[index]}\n            onChange={inputOnChange(index)}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            onFocus={inputOnFocus}\n            disabled={disabled}\n            required={required}\n            placeholder={placeholder?.[index] ?? ''}\n            aria-label={`${ariaLabel} ${index}`}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\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":"AA2GgC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\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        <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n          {label ? (\n            <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n              <Text\n                as=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${index}`}\n            data-testid={index}\n            value={value}\n            id={`${inputId || uniqueId}-${index}`}\n            ref={inputRefs[index]}\n            onChange={inputOnChange(index)}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            onFocus={inputOnFocus}\n            disabled={disabled}\n            required={required}\n            placeholder={placeholder?.[index] ?? ''}\n            aria-label={`${ariaLabel} ${index}`}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\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
88
  const DEFAULT_ON_FUNCTION = () => {
81
89
  };
82
90
  const inputOnFocus = (event) => event.target.select();
@@ -95,7 +103,11 @@ const VerificationCode = ({
95
103
  required = false,
96
104
  type = "number",
97
105
  "data-testid": dataTestId,
98
- "aria-label": ariaLabel = "Verification code"
106
+ "aria-label": ariaLabel = "Verification code",
107
+ label,
108
+ labelDescription,
109
+ helper,
110
+ success
99
111
  }) => {
100
112
  const uniqueId = useId();
101
113
  const valuesArray = Object.assign(new Array(fields).fill(""), [...initialValue.substring(0, fields)]);
@@ -195,7 +207,27 @@ const VerificationCode = ({
195
207
  next?.current?.focus();
196
208
  triggerChange(pastedValue);
197
209
  };
198
- return /* @__PURE__ */ jsx("div", { className, "data-testid": dataTestId, children: values.map((value, index) => /* @__PURE__ */ jsx(StyledInput, { 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":"AAySU","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n} from 'react'\nimport { createRef, useId, useState } from 'react'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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  border: solid 1px\n    ${({ 'aria-invalid': error, theme }) =>\n      error ? theme.colors.danger.border : theme.colors.neutral.border};\n  color: ${({ 'aria-invalid': error, theme }) =>\n    error ? theme.colors.danger.text : 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  &:hover,\n  &:focus {\n    border-color: ${({ 'aria-invalid': error, theme }) =>\n      error\n        ? theme.colors.danger.borderHover\n        : theme.colors.primary.borderHover};\n  }\n\n  &:focus {\n    box-shadow: ${({ 'aria-invalid': error, theme: { shadows } }) =>\n      error ? shadows.focusDanger : 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, theme: { colors } }) =>\n    disabled &&\n    `cursor: default;\n    background-color: ${colors.neutral.backgroundDisabled};\n    border-color: ${colors.neutral.borderDisabled};\n    color: ${colors.neutral.textDisabled};\n\n    &:hover {\n      border: ${colors.neutral.borderDisabled}\n    }\n    `}\n`\n\ntype VerificationCodeProps = {\n  disabled?: boolean\n  error?: boolean\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}\n\nconst DEFAULT_ON_FUNCTION = () => {}\n\nconst inputOnFocus: FocusEventHandler<HTMLInputElement> = event =>\n  event.target.select()\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}: 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        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\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  return (\n    <div className={className} data-testid={dataTestId}>\n      {values.map((value: string, index: number) => (\n        <StyledInput\n          css={[inputStyle]}\n          aria-invalid={error}\n          inputSize={size}\n          type={type === 'number' ? 'tel' : type}\n          pattern={type === 'number' ? '[0-9]*' : undefined}\n          key={`field-${index}`}\n          data-testid={index}\n          value={value}\n          id={`${inputId || uniqueId}-${index}`}\n          ref={inputRefs[index]}\n          onChange={inputOnChange(index)}\n          onKeyDown={inputOnKeyDown(index)}\n          onPaste={inputOnPaste(index)}\n          onFocus={inputOnFocus}\n          disabled={disabled}\n          required={required}\n          placeholder={placeholder?.[index] ?? ''}\n          aria-label={`${ariaLabel} ${index}`}\n        />\n      ))}\n    </div>\n  )\n}\n"]} */"], "aria-invalid": error, inputSize: size, type: type === "number" ? "tel" : type, pattern: type === "number" ? "[0-9]*" : void 0, "data-testid": index, value, id: `${inputId || uniqueId}-${index}`, ref: inputRefs[index], onChange: inputOnChange(index), onKeyDown: inputOnKeyDown(index), onPaste: inputOnPaste(index), onFocus: inputOnFocus, disabled, required, placeholder: placeholder?.[index] ?? "", "aria-label": `${ariaLabel} ${index}` }, `field-${index}`)) });
210
+ const sentiment = useMemo(() => {
211
+ if (error) {
212
+ return "danger";
213
+ }
214
+ if (success) {
215
+ return "success";
216
+ }
217
+ return "neutral";
218
+ }, [error, success]);
219
+ return /* @__PURE__ */ jsxs(FieldSet, { className, "data-testid": dataTestId, children: [
220
+ label || labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", alignItems: "center", children: [
221
+ label ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
222
+ /* @__PURE__ */ jsx(Text, { as: "legend", variant: ["xlarge", "large"].includes(size) ? "bodyStrong" : "bodySmallStrong", sentiment: "neutral", prominence: "strong", children: label }),
223
+ required ? /* @__PURE__ */ jsx(AsteriskIcon, { sentiment: "danger", size: 8 }) : null
224
+ ] }) : null,
225
+ labelDescription ?? null
226
+ ] }) : null,
227
+ /* @__PURE__ */ jsx("div", { children: values.map((value, index) => /* @__PURE__ */ jsx(StyledInput, { 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":"AAsWY","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/VerificationCode/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AsteriskIcon } from '@ultraviolet/icons'\nimport type {\n  ChangeEvent,\n  ClipboardEventHandler,\n  FocusEventHandler,\n  KeyboardEventHandler,\n  ReactNode,\n} from 'react'\nimport { createRef, useId, useMemo, useState } from 'react'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\n\ntype Size = 'small' | 'medium' | 'large' | 'xlarge'\n\nconst SIZE_HEIGHT = {\n  xlarge: '800',\n  large: '600',\n  medium: '500',\n  small: '400',\n} as const\n\nconst SIZE_WIDTH = {\n  xlarge: '700',\n  large: '500',\n  medium: '400',\n  small: '300',\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        case 'ArrowRight': {\n          event.preventDefault()\n          next?.current?.focus()\n          break\n        }\n        case 'ArrowUp': {\n          event.preventDefault()\n          first?.current?.focus()\n          break\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        <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n          {label ? (\n            <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n              <Text\n                as=\"legend\"\n                variant={\n                  ['xlarge', 'large'].includes(size)\n                    ? 'bodyStrong'\n                    : 'bodySmallStrong'\n                }\n                sentiment=\"neutral\"\n                prominence=\"strong\"\n              >\n                {label}\n              </Text>\n              {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n            </Stack>\n          ) : null}\n          {labelDescription ?? null}\n        </Stack>\n      ) : null}\n      <div>\n        {values.map((value: string, index: number) => (\n          <StyledInput\n            css={[inputStyle]}\n            aria-invalid={!!error}\n            data-success={!!success}\n            inputSize={size}\n            type={type === 'number' ? 'tel' : type}\n            pattern={type === 'number' ? '[0-9]*' : undefined}\n            key={`field-${index}`}\n            data-testid={index}\n            value={value}\n            id={`${inputId || uniqueId}-${index}`}\n            ref={inputRefs[index]}\n            onChange={inputOnChange(index)}\n            onKeyDown={inputOnKeyDown(index)}\n            onPaste={inputOnPaste(index)}\n            onFocus={inputOnFocus}\n            disabled={disabled}\n            required={required}\n            placeholder={placeholder?.[index] ?? ''}\n            aria-label={`${ariaLabel} ${index}`}\n          />\n        ))}\n      </div>\n      {error || typeof success === 'string' || typeof helper === 'string' ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={sentiment}\n          prominence={!error && !success ? 'weak' : 'default'}\n          disabled={disabled}\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"]} */"], "aria-invalid": !!error, "data-success": !!success, inputSize: size, type: type === "number" ? "tel" : type, pattern: type === "number" ? "[0-9]*" : void 0, "data-testid": index, value, id: `${inputId || uniqueId}-${index}`, ref: inputRefs[index], onChange: inputOnChange(index), onKeyDown: inputOnKeyDown(index), onPaste: inputOnPaste(index), onFocus: inputOnFocus, disabled, required, placeholder: placeholder?.[index] ?? "", "aria-label": `${ariaLabel} ${index}` }, `field-${index}`)) }),
228
+ error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsx(Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
229
+ !error && !success && typeof helper !== "string" && helper ? helper : null
230
+ ] });
199
231
  };
200
232
  export {
201
233
  VerificationCode
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@ultraviolet/ui",
3
- "version": "1.84.0",
3
+ "version": "1.84.2",
4
4
  "description": "Ultraviolet UI",
5
5
  "homepage": "https://github.com/scaleway/ultraviolet#readme",
6
6
  "repository": {
@@ -85,8 +85,8 @@
85
85
  "react-toastify": "10.0.6",
86
86
  "react-use-clipboard": "1.0.9",
87
87
  "reakit": "1.3.11",
88
- "@ultraviolet/icons": "3.8.2",
89
- "@ultraviolet/themes": "1.15.0"
88
+ "@ultraviolet/icons": "3.8.4",
89
+ "@ultraviolet/themes": "1.16.0"
90
90
  },
91
91
  "scripts": {
92
92
  "prebuild": "shx rm -rf dist",