@ultraviolet/ui 1.94.0 → 1.94.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.
- package/dist/components/Chip/index.d.ts +1 -1
- package/dist/components/SelectableCard/index.cjs +14 -12
- package/dist/components/SelectableCard/index.js +14 -12
- package/dist/components/TimeInputV2/index.cjs +5 -5
- package/dist/components/TimeInputV2/index.js +5 -5
- package/dist/components/VerificationCode/index.cjs +3 -3
- package/dist/components/VerificationCode/index.js +3 -3
- package/package.json +3 -3
|
@@ -54,7 +54,7 @@ const TimeInputWrapper = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV ===
|
|
|
54
54
|
theme
|
|
55
55
|
}) => theme.shadows.focusDanger, ';}&:not([data-disabled="true"]):not([data-readonly="true"]):hover{border-color:', ({
|
|
56
56
|
theme
|
|
57
|
-
}) => theme.colors.danger.borderHover, ";}}" + (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/TimeInputV2/index.tsx"],"names":[],"mappings":"AAuCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport {\n  DEFAULT_DATE,\n  DEFAULT_PLACEHOLDER,\n  INPUT_SIZE_HEIGHT,\n  TIME_KEYS,\n} from './constants'\nimport {\n  canConcat,\n  format,\n  getLastTypedChar,\n  getValueByType,\n  isAOrP,\n  isCompleteHour,\n  isNumber,\n  setValueByType,\n} from './helpers'\n\nexport type Time = {\n  h: string\n  m: string\n  s: string\n  period?: string\n}\n\nconst TimeInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-error': boolean\n}>`\n  display: flex;\n  cursor: text;\n  padding: ${({ theme }) => theme.space[1]};\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n  &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n  \n  &[data-disabled=\"false\"]:hover,\n  [data-disabled=\"false\"]:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):hover {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n`\n\nexport const Input = styled.input<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-period'?: boolean\n}>`\n  border: none;\n  outline: none;\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  width: ${({ theme }) => theme.sizing[312]};\n  height: ${({ theme }) => theme.sizing[300]};\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  caret-color: transparent;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &:not(:disabled):hover {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:not(:disabled):active, \n  :not(:disabled):focus{\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrong};\n    color:  ${({ theme }) => theme.colors.neutral.text};\n  }\n\n  &:read-only {\n    cursor: default;\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-period=\"true\"] {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  ::-moz-selection {\n    background: none;\n  }\n\n  ::selection {\n    background: none;\n  }\n`\n\nconst CustomText = styled(Text)`\npadding-inline: ${({ theme }) => theme.space['0.25']};\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date | null\n  clearable?: boolean\n  required?: boolean\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  disabled?: boolean\n  readOnly?: boolean\n  error?: boolean | string\n  'data-testid'?: string\n  onChange?: (value: Date | undefined, valuePeriod?: string) => void\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  className?: string\n  id?: string\n  size?: 'small' | 'medium' | 'large'\n  timeFormat?: 12 | 24\n  /**\n   * Automatically focus on the element on render. Autofocus is applied to the hour input\n   */\n  autoFocus?: boolean\n} & (\n  | {\n      label?: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n)\n\n/**\n * A time input component that allows users to type a time in a 24 or 12-hour format.\n * @experimental This component is experimental and may be subject to breaking changes in the future.\n */\nexport const TimeInputV2 = ({\n  label,\n  timeFormat = 24,\n  value,\n  clearable,\n  required,\n  labelDescription,\n  helper,\n  size = 'medium',\n  disabled = false,\n  readOnly = false,\n  error = false,\n  onChange,\n  onBlur,\n  onFocus,\n  className,\n  id,\n  autoFocus,\n  'data-testid': dataTestId,\n  placeholder = DEFAULT_PLACEHOLDER,\n  'aria-label': ariaLabel,\n}: TimeInputProps) => {\n  const localId = useId()\n  const defaultPeriod = useMemo(() => {\n    if (value) return value.getHours() >= 12 ? 'pm' : 'am'\n\n    return undefined\n  }, [value])\n\n  const [time, setTime] = useState(value)\n  const [period, setPeriod] = useState<'pm' | 'am' | undefined>(defaultPeriod)\n  const [filled, setFilled] = useState(\n    value ? { h: true, m: true, s: true } : { h: false, m: false, s: false },\n  ) // to not show 00 when there should be a placeholder\n\n  const refHours = useRef<HTMLInputElement>(null)\n  const refSeconds = useRef<HTMLInputElement>(null)\n  const refMinutes = useRef<HTMLInputElement>(null)\n  const refPeriod = useRef<HTMLInputElement>(null)\n\n  useEffect(() => {\n    if (value) {\n      setTime(value)\n\n      // without this condition, every time an input value changes, the other ones will be set to 0 if they used to be undefined\n      // instead of leaving them empty (and showing the placeholder)\n      if (value.getTime() !== time?.getTime()) {\n        setFilled({ h: true, m: true, s: true })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [value])\n\n  const handleChangePeriod = (key: 'a' | 'p') => {\n    if (!time) {\n      setPeriod(`${key}m`)\n    } else if (key.toLowerCase() === 'a') {\n      if (time.getHours() >= 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() - 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('am')\n    } else {\n      if (time.getHours() < 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() + 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('pm')\n    }\n  }\n  const handleChange = (type: 'h' | 'm' | 's', key: number) => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const valueToChange = getValueByType(type, time)\n\n    if (canConcat(valueToChange, type, key, timeFormat)) {\n      const newValue = (valueToChange % 10) * 10 + key\n\n      setValueByType(type, newTime, newValue)\n    } else setValueByType(type, newTime, key)\n\n    const newValue = getValueByType(type, newTime)\n    // Focus to next input if the current input has a valid time\n    if (type === 's' && newTime && newValue >= 7 && timeFormat === 12) {\n      refPeriod.current?.focus()\n    } else if (type === 'm' && newTime && newValue >= 6) {\n      refSeconds.current?.focus()\n    }\n\n    if (type === 'h') {\n      if (isCompleteHour(timeFormat, newValue)) {\n        refMinutes.current?.focus()\n      }\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n\n    setFilled(newFilled)\n  }\n\n  // Increase time with arrow up\n  const handleIncrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Decrease time with arrow down\n  const handleDecrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Go to next input\n  const handleNext = (type: 'h' | 'm' | 's') => {\n    if (type === 'h') refMinutes.current?.focus()\n    if (type === 'm') refSeconds.current?.focus()\n    if (type === 's' && timeFormat === 12) refPeriod.current?.focus()\n  }\n\n  // Go to previous input\n  const handlePrevious = (type: 'h' | 'm' | 's') => {\n    if (type === 'm') refHours.current?.focus()\n    if (type === 's') refMinutes.current?.focus()\n  }\n\n  return (\n    <Stack gap={0.5} className={className}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <TimeInputWrapper\n        data-readonly={readOnly}\n        data-disabled={disabled}\n        data-size={size}\n        data-error={!!error}\n        direction=\"row\"\n        alignItems=\"center\"\n        justifyContent=\"space-between\"\n        onBlur={onBlur}\n        onFocus={onFocus}\n        aria-required={required}\n        onClick={() => refHours.current?.focus()}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n      >\n        <Stack direction=\"row\">\n          {TIME_KEYS.map(type => {\n            const computedRef = () => {\n              if (type === 'h') return refHours\n              if (type === 'm') return refMinutes\n\n              return refSeconds\n            }\n            const fullName = () => {\n              if (type === 'h') return 'hours'\n              if (type === 'm') return 'minutes'\n\n              return 'seconds'\n            }\n\n            const computeMaxValue = () => {\n              if (type === 'h' && timeFormat === 12) return 12\n              if (type === 'h' && timeFormat === 24) return 23\n\n              return 59\n            }\n\n            return (\n              <Stack key={type} direction=\"row\">\n                <Input\n                  value={\n                    filled[type]\n                      ? format(getValueByType(type, time), type, timeFormat)\n                      : ''\n                  }\n                  placeholder={placeholder[type]}\n                  data-size={size}\n                  readOnly={readOnly}\n                  disabled={disabled}\n                  aria-label={ariaLabel}\n                  data-testid={`${fullName()}-input`}\n                  onClick={event => {\n                    event.stopPropagation()\n                  }}\n                  ref={computedRef()}\n                  role=\"spinbutton\"\n                  aria-valuemax={computeMaxValue()}\n                  aria-valuemin={type === 'h' && timeFormat === 12 ? 1 : 0}\n                  aria-valuenow={\n                    filled[type]\n                      ? Number.parseInt(\n                          format(getValueByType(type, time), type, timeFormat),\n                          10,\n                        )\n                      : undefined\n                  }\n                  onChange={event => {\n                    if (!readOnly && !disabled) {\n                      const key = getLastTypedChar(\n                        event.target.value,\n                        getValueByType(type, time),\n                      )\n                      if (isNumber(key)) {\n                        handleChange(type, Number.parseInt(key, 10))\n                      }\n                    }\n                  }}\n                  onKeyDown={event => {\n                    if (!readOnly && !disabled) {\n                      if (event.key === 'ArrowUp') {\n                        event.preventDefault()\n                        handleIncrease(type)\n                      } else if (event.key === 'ArrowDown') {\n                        event.preventDefault()\n                        handleDecrease(type)\n                      } else if (event.key === 'ArrowLeft') {\n                        event.preventDefault()\n                        handlePrevious(type)\n                      } else if (event.key === 'ArrowRight') {\n                        event.preventDefault()\n                        handleNext(type)\n                      }\n                    }\n                  }}\n                  autoFocus={autoFocus && type === 'h'}\n                />\n                {type === 's' ? null : (\n                  <CustomText\n                    as=\"span\"\n                    variant=\"body\"\n                    prominence=\"default\"\n                    sentiment=\"neutral\"\n                  >\n                    :\n                  </CustomText>\n                )}\n              </Stack>\n            )\n          })}\n          {timeFormat === 12 ? (\n            <Input\n              value={period?.toUpperCase()}\n              placeholder={placeholder.period ?? 'AM'}\n              data-size={size}\n              data-period\n              readOnly={readOnly}\n              disabled={disabled}\n              aria-label={ariaLabel}\n              data-testid=\"am-pm-input\"\n              onChange={event => {\n                if (!readOnly && !disabled) {\n                  const key = event.target.value.slice(-1)\n                  if (isAOrP(key)) handleChangePeriod(key as 'a' | 'p')\n                }\n              }}\n              onKeyDown={event => {\n                if (!readOnly && !disabled) {\n                  if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {\n                    event.preventDefault()\n                    handleChangePeriod(period === 'am' ? 'p' : 'a')\n                  } else if (event.key === 'ArrowLeft') {\n                    event.preventDefault()\n                    refSeconds.current?.focus()\n                  }\n                }\n              }}\n              ref={refPeriod}\n              onClick={event => event.stopPropagation()}\n              role=\"spinbutton\"\n              aria-valuemax={12}\n              aria-valuemin={0}\n              aria-valuenow={period === 'am' ? 0 : 12}\n              aria-valuetext={period}\n            />\n          ) : null}\n        </Stack>\n        {error || clearable ? (\n          <Stack direction=\"row\" alignItems=\"center\" gap=\"1\">\n            {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            {clearable ? (\n              <Button\n                aria-label=\"clear value\"\n                disabled={disabled || readOnly}\n                variant=\"ghost\"\n                size=\"small\"\n                icon=\"close\"\n                onClick={event => {\n                  event.stopPropagation()\n                  setTime(undefined)\n                  onChange?.(undefined)\n                }}\n                sentiment=\"neutral\"\n                data-testid=\"clear\"\n              />\n            ) : null}\n          </Stack>\n        ) : null}\n      </TimeInputWrapper>\n      {helper || error ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={error ? 'danger' : 'neutral'}\n          prominence={error ? 'default' : 'weak'}\n          disabled={disabled}\n        >\n          {error || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
|
|
57
|
+
}) => theme.colors.danger.borderHover, ";}}" + (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/TimeInputV2/index.tsx"],"names":[],"mappings":"AAuCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport {\n  DEFAULT_DATE,\n  DEFAULT_PLACEHOLDER,\n  INPUT_SIZE_HEIGHT,\n  TIME_KEYS,\n} from './constants'\nimport {\n  canConcat,\n  format,\n  getLastTypedChar,\n  getValueByType,\n  isAOrP,\n  isCompleteHour,\n  isNumber,\n  setValueByType,\n} from './helpers'\n\nexport type Time = {\n  h: string\n  m: string\n  s: string\n  period?: string\n}\n\nconst TimeInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-error': boolean\n}>`\n  display: flex;\n  cursor: text;\n  padding: ${({ theme }) => theme.space[1]};\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n  &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n\n  &[data-disabled=\"false\"]:hover,\n  [data-disabled=\"false\"]:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):hover {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n`\n\nexport const Input = styled.input<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-period'?: boolean\n}>`\n  border: none;\n  outline: none;\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  width: ${({ theme }) => theme.sizing[312]};\n  height: ${({ theme }) => theme.sizing[300]};\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  caret-color: transparent;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &:not(:disabled):hover {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:not(:disabled):active,\n  :not(:disabled):focus{\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrong};\n    color:  ${({ theme }) => theme.colors.neutral.text};\n  }\n\n  &:read-only {\n    cursor: default;\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-period=\"true\"] {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  ::-moz-selection {\n    background: none;\n  }\n\n  ::selection {\n    background: none;\n  }\n`\n\nconst CustomText = styled(Text)`\npadding-inline: ${({ theme }) => theme.space['0.25']};\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date | null\n  clearable?: boolean\n  required?: boolean\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  disabled?: boolean\n  readOnly?: boolean\n  error?: boolean | string\n  'data-testid'?: string\n  onChange?: (value: Date | undefined, valuePeriod?: string) => void\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  className?: string\n  id?: string\n  size?: 'small' | 'medium' | 'large'\n  timeFormat?: 12 | 24\n  /**\n   * Automatically focus on the element on render. Autofocus is applied to the hour input\n   */\n  autoFocus?: boolean\n} & (\n  | {\n      label?: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n)\n\n/**\n * A time input component that allows users to type a time in a 24 or 12-hour format.\n * @experimental This component is experimental and may be subject to breaking changes in the future.\n */\nexport const TimeInputV2 = ({\n  label,\n  timeFormat = 24,\n  value,\n  clearable,\n  required,\n  labelDescription,\n  helper,\n  size = 'medium',\n  disabled = false,\n  readOnly = false,\n  error = false,\n  onChange,\n  onBlur,\n  onFocus,\n  className,\n  id,\n  autoFocus,\n  'data-testid': dataTestId,\n  placeholder = DEFAULT_PLACEHOLDER,\n  'aria-label': ariaLabel,\n}: TimeInputProps) => {\n  const localId = useId()\n  const defaultPeriod = useMemo(() => {\n    if (value) return value.getHours() >= 12 ? 'pm' : 'am'\n\n    return undefined\n  }, [value])\n\n  const [time, setTime] = useState(value)\n  const [period, setPeriod] = useState<'pm' | 'am' | undefined>(defaultPeriod)\n  const [filled, setFilled] = useState(\n    value ? { h: true, m: true, s: true } : { h: false, m: false, s: false },\n  ) // to not show 00 when there should be a placeholder\n\n  const refHours = useRef<HTMLInputElement>(null)\n  const refSeconds = useRef<HTMLInputElement>(null)\n  const refMinutes = useRef<HTMLInputElement>(null)\n  const refPeriod = useRef<HTMLInputElement>(null)\n\n  useEffect(() => {\n    if (value) {\n      setTime(value)\n\n      // without this condition, every time an input value changes, the other ones will be set to 0 if they used to be undefined\n      // instead of leaving them empty (and showing the placeholder)\n      if (value.getTime() !== time?.getTime()) {\n        setFilled({ h: true, m: true, s: true })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [value])\n\n  const handleChangePeriod = (key: 'a' | 'p') => {\n    if (!time) {\n      setPeriod(`${key}m`)\n    } else if (key.toLowerCase() === 'a') {\n      if (time.getHours() >= 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() - 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('am')\n    } else {\n      if (time.getHours() < 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() + 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('pm')\n    }\n  }\n  const handleChange = (type: 'h' | 'm' | 's', key: number) => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const valueToChange = getValueByType(type, time)\n\n    if (canConcat(valueToChange, type, key, timeFormat)) {\n      const newValue = (valueToChange % 10) * 10 + key\n\n      setValueByType(type, newTime, newValue)\n    } else setValueByType(type, newTime, key)\n\n    const newValue = getValueByType(type, newTime)\n    // Focus to next input if the current input has a valid time\n    if (type === 's' && newTime && newValue >= 7 && timeFormat === 12) {\n      refPeriod.current?.focus()\n    } else if (type === 'm' && newTime && newValue >= 6) {\n      refSeconds.current?.focus()\n    }\n\n    if (type === 'h') {\n      if (isCompleteHour(timeFormat, newValue)) {\n        refMinutes.current?.focus()\n      }\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n\n    setFilled(newFilled)\n  }\n\n  // Increase time with arrow up\n  const handleIncrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Decrease time with arrow down\n  const handleDecrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Go to next input\n  const handleNext = (type: 'h' | 'm' | 's') => {\n    if (type === 'h') refMinutes.current?.focus()\n    if (type === 'm') refSeconds.current?.focus()\n    if (type === 's' && timeFormat === 12) refPeriod.current?.focus()\n  }\n\n  // Go to previous input\n  const handlePrevious = (type: 'h' | 'm' | 's') => {\n    if (type === 'm') refHours.current?.focus()\n    if (type === 's') refMinutes.current?.focus()\n  }\n\n  return (\n    <Stack gap={0.5} className={className}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <TimeInputWrapper\n        data-readonly={readOnly}\n        data-disabled={disabled}\n        data-size={size}\n        data-error={!!error}\n        direction=\"row\"\n        alignItems=\"center\"\n        justifyContent=\"space-between\"\n        onBlur={onBlur}\n        onFocus={onFocus}\n        aria-required={required}\n        onClick={() => refHours.current?.focus()}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n      >\n        <Stack direction=\"row\">\n          {TIME_KEYS.map(type => {\n            const computedRef = () => {\n              if (type === 'h') return refHours\n              if (type === 'm') return refMinutes\n\n              return refSeconds\n            }\n            const fullName = () => {\n              if (type === 'h') return 'hours'\n              if (type === 'm') return 'minutes'\n\n              return 'seconds'\n            }\n\n            const computeMaxValue = () => {\n              if (type === 'h' && timeFormat === 12) return 12\n              if (type === 'h' && timeFormat === 24) return 23\n\n              return 59\n            }\n\n            return (\n              <Stack key={type} direction=\"row\">\n                <Input\n                  value={\n                    filled[type]\n                      ? format(getValueByType(type, time), type, timeFormat)\n                      : ''\n                  }\n                  placeholder={placeholder[type]}\n                  data-size={size}\n                  readOnly={readOnly}\n                  disabled={disabled}\n                  aria-label={ariaLabel}\n                  data-testid={`${fullName()}-input`}\n                  onClick={event => {\n                    event.stopPropagation()\n                  }}\n                  ref={computedRef()}\n                  role=\"spinbutton\"\n                  autoComplete=\"false\"\n                  aria-valuemax={computeMaxValue()}\n                  aria-valuemin={type === 'h' && timeFormat === 12 ? 1 : 0}\n                  aria-valuenow={\n                    filled[type]\n                      ? Number.parseInt(\n                          format(getValueByType(type, time), type, timeFormat),\n                          10,\n                        )\n                      : undefined\n                  }\n                  onChange={event => {\n                    if (!readOnly && !disabled) {\n                      const key = getLastTypedChar(\n                        event.target.value,\n                        getValueByType(type, time),\n                      )\n                      if (isNumber(key)) {\n                        handleChange(type, Number.parseInt(key, 10))\n                      }\n                    }\n                  }}\n                  onKeyDown={event => {\n                    if (!readOnly && !disabled) {\n                      if (event.key === 'ArrowUp') {\n                        event.preventDefault()\n                        handleIncrease(type)\n                      } else if (event.key === 'ArrowDown') {\n                        event.preventDefault()\n                        handleDecrease(type)\n                      } else if (event.key === 'ArrowLeft') {\n                        event.preventDefault()\n                        handlePrevious(type)\n                      } else if (event.key === 'ArrowRight') {\n                        event.preventDefault()\n                        handleNext(type)\n                      }\n                    }\n                  }}\n                  autoFocus={autoFocus && type === 'h'}\n                />\n                {type === 's' ? null : (\n                  <CustomText\n                    as=\"span\"\n                    variant=\"body\"\n                    prominence=\"default\"\n                    sentiment=\"neutral\"\n                  >\n                    :\n                  </CustomText>\n                )}\n              </Stack>\n            )\n          })}\n          {timeFormat === 12 ? (\n            <Input\n              value={period?.toUpperCase()}\n              placeholder={placeholder.period ?? 'AM'}\n              data-size={size}\n              data-period\n              readOnly={readOnly}\n              disabled={disabled}\n              aria-label={ariaLabel}\n              data-testid=\"am-pm-input\"\n              onChange={event => {\n                if (!readOnly && !disabled) {\n                  const key = event.target.value.slice(-1)\n                  if (isAOrP(key)) handleChangePeriod(key as 'a' | 'p')\n                }\n              }}\n              onKeyDown={event => {\n                if (!readOnly && !disabled) {\n                  if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {\n                    event.preventDefault()\n                    handleChangePeriod(period === 'am' ? 'p' : 'a')\n                  } else if (event.key === 'ArrowLeft') {\n                    event.preventDefault()\n                    refSeconds.current?.focus()\n                  }\n                }\n              }}\n              ref={refPeriod}\n              onClick={event => event.stopPropagation()}\n              role=\"spinbutton\"\n              autoComplete=\"false\"\n              aria-valuemax={12}\n              aria-valuemin={0}\n              aria-valuenow={period === 'am' ? 0 : 12}\n              aria-valuetext={period}\n            />\n          ) : null}\n        </Stack>\n        {error || clearable ? (\n          <Stack direction=\"row\" alignItems=\"center\" gap=\"1\">\n            {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            {clearable ? (\n              <Button\n                aria-label=\"clear value\"\n                disabled={disabled || readOnly}\n                variant=\"ghost\"\n                size=\"small\"\n                icon=\"close\"\n                onClick={event => {\n                  event.stopPropagation()\n                  setTime(undefined)\n                  onChange?.(undefined)\n                }}\n                sentiment=\"neutral\"\n                data-testid=\"clear\"\n              />\n            ) : null}\n          </Stack>\n        ) : null}\n      </TimeInputWrapper>\n      {helper || error ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={error ? 'danger' : 'neutral'}\n          prominence={error ? 'default' : 'weak'}\n          disabled={disabled}\n        >\n          {error || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
|
|
58
58
|
const Input = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "production" ? {
|
|
59
59
|
target: "e8pjt8k1"
|
|
60
60
|
} : {
|
|
@@ -82,7 +82,7 @@ const Input = /* @__PURE__ */ _styled("input", process.env.NODE_ENV === "product
|
|
|
82
82
|
theme
|
|
83
83
|
}) => theme.colors.neutral.text, ';}&:read-only{cursor:default;}&:disabled{cursor:not-allowed;user-select:none;}&[data-period="true"]{color:', ({
|
|
84
84
|
theme
|
|
85
|
-
}) => theme.colors.neutral.textWeak, ";}::-moz-selection{background:none;}::selection{background:none;}" + (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/TimeInputV2/index.tsx"],"names":[],"mappings":"AA0GE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport {\n  DEFAULT_DATE,\n  DEFAULT_PLACEHOLDER,\n  INPUT_SIZE_HEIGHT,\n  TIME_KEYS,\n} from './constants'\nimport {\n  canConcat,\n  format,\n  getLastTypedChar,\n  getValueByType,\n  isAOrP,\n  isCompleteHour,\n  isNumber,\n  setValueByType,\n} from './helpers'\n\nexport type Time = {\n  h: string\n  m: string\n  s: string\n  period?: string\n}\n\nconst TimeInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-error': boolean\n}>`\n  display: flex;\n  cursor: text;\n  padding: ${({ theme }) => theme.space[1]};\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n  &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n  \n  &[data-disabled=\"false\"]:hover,\n  [data-disabled=\"false\"]:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):hover {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n`\n\nexport const Input = styled.input<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-period'?: boolean\n}>`\n  border: none;\n  outline: none;\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  width: ${({ theme }) => theme.sizing[312]};\n  height: ${({ theme }) => theme.sizing[300]};\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  caret-color: transparent;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &:not(:disabled):hover {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:not(:disabled):active, \n  :not(:disabled):focus{\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrong};\n    color:  ${({ theme }) => theme.colors.neutral.text};\n  }\n\n  &:read-only {\n    cursor: default;\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-period=\"true\"] {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  ::-moz-selection {\n    background: none;\n  }\n\n  ::selection {\n    background: none;\n  }\n`\n\nconst CustomText = styled(Text)`\npadding-inline: ${({ theme }) => theme.space['0.25']};\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date | null\n  clearable?: boolean\n  required?: boolean\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  disabled?: boolean\n  readOnly?: boolean\n  error?: boolean | string\n  'data-testid'?: string\n  onChange?: (value: Date | undefined, valuePeriod?: string) => void\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  className?: string\n  id?: string\n  size?: 'small' | 'medium' | 'large'\n  timeFormat?: 12 | 24\n  /**\n   * Automatically focus on the element on render. Autofocus is applied to the hour input\n   */\n  autoFocus?: boolean\n} & (\n  | {\n      label?: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n)\n\n/**\n * A time input component that allows users to type a time in a 24 or 12-hour format.\n * @experimental This component is experimental and may be subject to breaking changes in the future.\n */\nexport const TimeInputV2 = ({\n  label,\n  timeFormat = 24,\n  value,\n  clearable,\n  required,\n  labelDescription,\n  helper,\n  size = 'medium',\n  disabled = false,\n  readOnly = false,\n  error = false,\n  onChange,\n  onBlur,\n  onFocus,\n  className,\n  id,\n  autoFocus,\n  'data-testid': dataTestId,\n  placeholder = DEFAULT_PLACEHOLDER,\n  'aria-label': ariaLabel,\n}: TimeInputProps) => {\n  const localId = useId()\n  const defaultPeriod = useMemo(() => {\n    if (value) return value.getHours() >= 12 ? 'pm' : 'am'\n\n    return undefined\n  }, [value])\n\n  const [time, setTime] = useState(value)\n  const [period, setPeriod] = useState<'pm' | 'am' | undefined>(defaultPeriod)\n  const [filled, setFilled] = useState(\n    value ? { h: true, m: true, s: true } : { h: false, m: false, s: false },\n  ) // to not show 00 when there should be a placeholder\n\n  const refHours = useRef<HTMLInputElement>(null)\n  const refSeconds = useRef<HTMLInputElement>(null)\n  const refMinutes = useRef<HTMLInputElement>(null)\n  const refPeriod = useRef<HTMLInputElement>(null)\n\n  useEffect(() => {\n    if (value) {\n      setTime(value)\n\n      // without this condition, every time an input value changes, the other ones will be set to 0 if they used to be undefined\n      // instead of leaving them empty (and showing the placeholder)\n      if (value.getTime() !== time?.getTime()) {\n        setFilled({ h: true, m: true, s: true })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [value])\n\n  const handleChangePeriod = (key: 'a' | 'p') => {\n    if (!time) {\n      setPeriod(`${key}m`)\n    } else if (key.toLowerCase() === 'a') {\n      if (time.getHours() >= 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() - 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('am')\n    } else {\n      if (time.getHours() < 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() + 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('pm')\n    }\n  }\n  const handleChange = (type: 'h' | 'm' | 's', key: number) => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const valueToChange = getValueByType(type, time)\n\n    if (canConcat(valueToChange, type, key, timeFormat)) {\n      const newValue = (valueToChange % 10) * 10 + key\n\n      setValueByType(type, newTime, newValue)\n    } else setValueByType(type, newTime, key)\n\n    const newValue = getValueByType(type, newTime)\n    // Focus to next input if the current input has a valid time\n    if (type === 's' && newTime && newValue >= 7 && timeFormat === 12) {\n      refPeriod.current?.focus()\n    } else if (type === 'm' && newTime && newValue >= 6) {\n      refSeconds.current?.focus()\n    }\n\n    if (type === 'h') {\n      if (isCompleteHour(timeFormat, newValue)) {\n        refMinutes.current?.focus()\n      }\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n\n    setFilled(newFilled)\n  }\n\n  // Increase time with arrow up\n  const handleIncrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Decrease time with arrow down\n  const handleDecrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Go to next input\n  const handleNext = (type: 'h' | 'm' | 's') => {\n    if (type === 'h') refMinutes.current?.focus()\n    if (type === 'm') refSeconds.current?.focus()\n    if (type === 's' && timeFormat === 12) refPeriod.current?.focus()\n  }\n\n  // Go to previous input\n  const handlePrevious = (type: 'h' | 'm' | 's') => {\n    if (type === 'm') refHours.current?.focus()\n    if (type === 's') refMinutes.current?.focus()\n  }\n\n  return (\n    <Stack gap={0.5} className={className}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <TimeInputWrapper\n        data-readonly={readOnly}\n        data-disabled={disabled}\n        data-size={size}\n        data-error={!!error}\n        direction=\"row\"\n        alignItems=\"center\"\n        justifyContent=\"space-between\"\n        onBlur={onBlur}\n        onFocus={onFocus}\n        aria-required={required}\n        onClick={() => refHours.current?.focus()}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n      >\n        <Stack direction=\"row\">\n          {TIME_KEYS.map(type => {\n            const computedRef = () => {\n              if (type === 'h') return refHours\n              if (type === 'm') return refMinutes\n\n              return refSeconds\n            }\n            const fullName = () => {\n              if (type === 'h') return 'hours'\n              if (type === 'm') return 'minutes'\n\n              return 'seconds'\n            }\n\n            const computeMaxValue = () => {\n              if (type === 'h' && timeFormat === 12) return 12\n              if (type === 'h' && timeFormat === 24) return 23\n\n              return 59\n            }\n\n            return (\n              <Stack key={type} direction=\"row\">\n                <Input\n                  value={\n                    filled[type]\n                      ? format(getValueByType(type, time), type, timeFormat)\n                      : ''\n                  }\n                  placeholder={placeholder[type]}\n                  data-size={size}\n                  readOnly={readOnly}\n                  disabled={disabled}\n                  aria-label={ariaLabel}\n                  data-testid={`${fullName()}-input`}\n                  onClick={event => {\n                    event.stopPropagation()\n                  }}\n                  ref={computedRef()}\n                  role=\"spinbutton\"\n                  aria-valuemax={computeMaxValue()}\n                  aria-valuemin={type === 'h' && timeFormat === 12 ? 1 : 0}\n                  aria-valuenow={\n                    filled[type]\n                      ? Number.parseInt(\n                          format(getValueByType(type, time), type, timeFormat),\n                          10,\n                        )\n                      : undefined\n                  }\n                  onChange={event => {\n                    if (!readOnly && !disabled) {\n                      const key = getLastTypedChar(\n                        event.target.value,\n                        getValueByType(type, time),\n                      )\n                      if (isNumber(key)) {\n                        handleChange(type, Number.parseInt(key, 10))\n                      }\n                    }\n                  }}\n                  onKeyDown={event => {\n                    if (!readOnly && !disabled) {\n                      if (event.key === 'ArrowUp') {\n                        event.preventDefault()\n                        handleIncrease(type)\n                      } else if (event.key === 'ArrowDown') {\n                        event.preventDefault()\n                        handleDecrease(type)\n                      } else if (event.key === 'ArrowLeft') {\n                        event.preventDefault()\n                        handlePrevious(type)\n                      } else if (event.key === 'ArrowRight') {\n                        event.preventDefault()\n                        handleNext(type)\n                      }\n                    }\n                  }}\n                  autoFocus={autoFocus && type === 'h'}\n                />\n                {type === 's' ? null : (\n                  <CustomText\n                    as=\"span\"\n                    variant=\"body\"\n                    prominence=\"default\"\n                    sentiment=\"neutral\"\n                  >\n                    :\n                  </CustomText>\n                )}\n              </Stack>\n            )\n          })}\n          {timeFormat === 12 ? (\n            <Input\n              value={period?.toUpperCase()}\n              placeholder={placeholder.period ?? 'AM'}\n              data-size={size}\n              data-period\n              readOnly={readOnly}\n              disabled={disabled}\n              aria-label={ariaLabel}\n              data-testid=\"am-pm-input\"\n              onChange={event => {\n                if (!readOnly && !disabled) {\n                  const key = event.target.value.slice(-1)\n                  if (isAOrP(key)) handleChangePeriod(key as 'a' | 'p')\n                }\n              }}\n              onKeyDown={event => {\n                if (!readOnly && !disabled) {\n                  if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {\n                    event.preventDefault()\n                    handleChangePeriod(period === 'am' ? 'p' : 'a')\n                  } else if (event.key === 'ArrowLeft') {\n                    event.preventDefault()\n                    refSeconds.current?.focus()\n                  }\n                }\n              }}\n              ref={refPeriod}\n              onClick={event => event.stopPropagation()}\n              role=\"spinbutton\"\n              aria-valuemax={12}\n              aria-valuemin={0}\n              aria-valuenow={period === 'am' ? 0 : 12}\n              aria-valuetext={period}\n            />\n          ) : null}\n        </Stack>\n        {error || clearable ? (\n          <Stack direction=\"row\" alignItems=\"center\" gap=\"1\">\n            {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            {clearable ? (\n              <Button\n                aria-label=\"clear value\"\n                disabled={disabled || readOnly}\n                variant=\"ghost\"\n                size=\"small\"\n                icon=\"close\"\n                onClick={event => {\n                  event.stopPropagation()\n                  setTime(undefined)\n                  onChange?.(undefined)\n                }}\n                sentiment=\"neutral\"\n                data-testid=\"clear\"\n              />\n            ) : null}\n          </Stack>\n        ) : null}\n      </TimeInputWrapper>\n      {helper || error ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={error ? 'danger' : 'neutral'}\n          prominence={error ? 'default' : 'weak'}\n          disabled={disabled}\n        >\n          {error || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
|
|
85
|
+
}) => theme.colors.neutral.textWeak, ";}::-moz-selection{background:none;}::selection{background:none;}" + (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/TimeInputV2/index.tsx"],"names":[],"mappings":"AA0GE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport {\n  DEFAULT_DATE,\n  DEFAULT_PLACEHOLDER,\n  INPUT_SIZE_HEIGHT,\n  TIME_KEYS,\n} from './constants'\nimport {\n  canConcat,\n  format,\n  getLastTypedChar,\n  getValueByType,\n  isAOrP,\n  isCompleteHour,\n  isNumber,\n  setValueByType,\n} from './helpers'\n\nexport type Time = {\n  h: string\n  m: string\n  s: string\n  period?: string\n}\n\nconst TimeInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-error': boolean\n}>`\n  display: flex;\n  cursor: text;\n  padding: ${({ theme }) => theme.space[1]};\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n  &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n\n  &[data-disabled=\"false\"]:hover,\n  [data-disabled=\"false\"]:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):hover {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n`\n\nexport const Input = styled.input<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-period'?: boolean\n}>`\n  border: none;\n  outline: none;\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  width: ${({ theme }) => theme.sizing[312]};\n  height: ${({ theme }) => theme.sizing[300]};\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  caret-color: transparent;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &:not(:disabled):hover {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:not(:disabled):active,\n  :not(:disabled):focus{\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrong};\n    color:  ${({ theme }) => theme.colors.neutral.text};\n  }\n\n  &:read-only {\n    cursor: default;\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-period=\"true\"] {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  ::-moz-selection {\n    background: none;\n  }\n\n  ::selection {\n    background: none;\n  }\n`\n\nconst CustomText = styled(Text)`\npadding-inline: ${({ theme }) => theme.space['0.25']};\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date | null\n  clearable?: boolean\n  required?: boolean\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  disabled?: boolean\n  readOnly?: boolean\n  error?: boolean | string\n  'data-testid'?: string\n  onChange?: (value: Date | undefined, valuePeriod?: string) => void\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  className?: string\n  id?: string\n  size?: 'small' | 'medium' | 'large'\n  timeFormat?: 12 | 24\n  /**\n   * Automatically focus on the element on render. Autofocus is applied to the hour input\n   */\n  autoFocus?: boolean\n} & (\n  | {\n      label?: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n)\n\n/**\n * A time input component that allows users to type a time in a 24 or 12-hour format.\n * @experimental This component is experimental and may be subject to breaking changes in the future.\n */\nexport const TimeInputV2 = ({\n  label,\n  timeFormat = 24,\n  value,\n  clearable,\n  required,\n  labelDescription,\n  helper,\n  size = 'medium',\n  disabled = false,\n  readOnly = false,\n  error = false,\n  onChange,\n  onBlur,\n  onFocus,\n  className,\n  id,\n  autoFocus,\n  'data-testid': dataTestId,\n  placeholder = DEFAULT_PLACEHOLDER,\n  'aria-label': ariaLabel,\n}: TimeInputProps) => {\n  const localId = useId()\n  const defaultPeriod = useMemo(() => {\n    if (value) return value.getHours() >= 12 ? 'pm' : 'am'\n\n    return undefined\n  }, [value])\n\n  const [time, setTime] = useState(value)\n  const [period, setPeriod] = useState<'pm' | 'am' | undefined>(defaultPeriod)\n  const [filled, setFilled] = useState(\n    value ? { h: true, m: true, s: true } : { h: false, m: false, s: false },\n  ) // to not show 00 when there should be a placeholder\n\n  const refHours = useRef<HTMLInputElement>(null)\n  const refSeconds = useRef<HTMLInputElement>(null)\n  const refMinutes = useRef<HTMLInputElement>(null)\n  const refPeriod = useRef<HTMLInputElement>(null)\n\n  useEffect(() => {\n    if (value) {\n      setTime(value)\n\n      // without this condition, every time an input value changes, the other ones will be set to 0 if they used to be undefined\n      // instead of leaving them empty (and showing the placeholder)\n      if (value.getTime() !== time?.getTime()) {\n        setFilled({ h: true, m: true, s: true })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [value])\n\n  const handleChangePeriod = (key: 'a' | 'p') => {\n    if (!time) {\n      setPeriod(`${key}m`)\n    } else if (key.toLowerCase() === 'a') {\n      if (time.getHours() >= 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() - 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('am')\n    } else {\n      if (time.getHours() < 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() + 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('pm')\n    }\n  }\n  const handleChange = (type: 'h' | 'm' | 's', key: number) => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const valueToChange = getValueByType(type, time)\n\n    if (canConcat(valueToChange, type, key, timeFormat)) {\n      const newValue = (valueToChange % 10) * 10 + key\n\n      setValueByType(type, newTime, newValue)\n    } else setValueByType(type, newTime, key)\n\n    const newValue = getValueByType(type, newTime)\n    // Focus to next input if the current input has a valid time\n    if (type === 's' && newTime && newValue >= 7 && timeFormat === 12) {\n      refPeriod.current?.focus()\n    } else if (type === 'm' && newTime && newValue >= 6) {\n      refSeconds.current?.focus()\n    }\n\n    if (type === 'h') {\n      if (isCompleteHour(timeFormat, newValue)) {\n        refMinutes.current?.focus()\n      }\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n\n    setFilled(newFilled)\n  }\n\n  // Increase time with arrow up\n  const handleIncrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Decrease time with arrow down\n  const handleDecrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Go to next input\n  const handleNext = (type: 'h' | 'm' | 's') => {\n    if (type === 'h') refMinutes.current?.focus()\n    if (type === 'm') refSeconds.current?.focus()\n    if (type === 's' && timeFormat === 12) refPeriod.current?.focus()\n  }\n\n  // Go to previous input\n  const handlePrevious = (type: 'h' | 'm' | 's') => {\n    if (type === 'm') refHours.current?.focus()\n    if (type === 's') refMinutes.current?.focus()\n  }\n\n  return (\n    <Stack gap={0.5} className={className}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <TimeInputWrapper\n        data-readonly={readOnly}\n        data-disabled={disabled}\n        data-size={size}\n        data-error={!!error}\n        direction=\"row\"\n        alignItems=\"center\"\n        justifyContent=\"space-between\"\n        onBlur={onBlur}\n        onFocus={onFocus}\n        aria-required={required}\n        onClick={() => refHours.current?.focus()}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n      >\n        <Stack direction=\"row\">\n          {TIME_KEYS.map(type => {\n            const computedRef = () => {\n              if (type === 'h') return refHours\n              if (type === 'm') return refMinutes\n\n              return refSeconds\n            }\n            const fullName = () => {\n              if (type === 'h') return 'hours'\n              if (type === 'm') return 'minutes'\n\n              return 'seconds'\n            }\n\n            const computeMaxValue = () => {\n              if (type === 'h' && timeFormat === 12) return 12\n              if (type === 'h' && timeFormat === 24) return 23\n\n              return 59\n            }\n\n            return (\n              <Stack key={type} direction=\"row\">\n                <Input\n                  value={\n                    filled[type]\n                      ? format(getValueByType(type, time), type, timeFormat)\n                      : ''\n                  }\n                  placeholder={placeholder[type]}\n                  data-size={size}\n                  readOnly={readOnly}\n                  disabled={disabled}\n                  aria-label={ariaLabel}\n                  data-testid={`${fullName()}-input`}\n                  onClick={event => {\n                    event.stopPropagation()\n                  }}\n                  ref={computedRef()}\n                  role=\"spinbutton\"\n                  autoComplete=\"false\"\n                  aria-valuemax={computeMaxValue()}\n                  aria-valuemin={type === 'h' && timeFormat === 12 ? 1 : 0}\n                  aria-valuenow={\n                    filled[type]\n                      ? Number.parseInt(\n                          format(getValueByType(type, time), type, timeFormat),\n                          10,\n                        )\n                      : undefined\n                  }\n                  onChange={event => {\n                    if (!readOnly && !disabled) {\n                      const key = getLastTypedChar(\n                        event.target.value,\n                        getValueByType(type, time),\n                      )\n                      if (isNumber(key)) {\n                        handleChange(type, Number.parseInt(key, 10))\n                      }\n                    }\n                  }}\n                  onKeyDown={event => {\n                    if (!readOnly && !disabled) {\n                      if (event.key === 'ArrowUp') {\n                        event.preventDefault()\n                        handleIncrease(type)\n                      } else if (event.key === 'ArrowDown') {\n                        event.preventDefault()\n                        handleDecrease(type)\n                      } else if (event.key === 'ArrowLeft') {\n                        event.preventDefault()\n                        handlePrevious(type)\n                      } else if (event.key === 'ArrowRight') {\n                        event.preventDefault()\n                        handleNext(type)\n                      }\n                    }\n                  }}\n                  autoFocus={autoFocus && type === 'h'}\n                />\n                {type === 's' ? null : (\n                  <CustomText\n                    as=\"span\"\n                    variant=\"body\"\n                    prominence=\"default\"\n                    sentiment=\"neutral\"\n                  >\n                    :\n                  </CustomText>\n                )}\n              </Stack>\n            )\n          })}\n          {timeFormat === 12 ? (\n            <Input\n              value={period?.toUpperCase()}\n              placeholder={placeholder.period ?? 'AM'}\n              data-size={size}\n              data-period\n              readOnly={readOnly}\n              disabled={disabled}\n              aria-label={ariaLabel}\n              data-testid=\"am-pm-input\"\n              onChange={event => {\n                if (!readOnly && !disabled) {\n                  const key = event.target.value.slice(-1)\n                  if (isAOrP(key)) handleChangePeriod(key as 'a' | 'p')\n                }\n              }}\n              onKeyDown={event => {\n                if (!readOnly && !disabled) {\n                  if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {\n                    event.preventDefault()\n                    handleChangePeriod(period === 'am' ? 'p' : 'a')\n                  } else if (event.key === 'ArrowLeft') {\n                    event.preventDefault()\n                    refSeconds.current?.focus()\n                  }\n                }\n              }}\n              ref={refPeriod}\n              onClick={event => event.stopPropagation()}\n              role=\"spinbutton\"\n              autoComplete=\"false\"\n              aria-valuemax={12}\n              aria-valuemin={0}\n              aria-valuenow={period === 'am' ? 0 : 12}\n              aria-valuetext={period}\n            />\n          ) : null}\n        </Stack>\n        {error || clearable ? (\n          <Stack direction=\"row\" alignItems=\"center\" gap=\"1\">\n            {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            {clearable ? (\n              <Button\n                aria-label=\"clear value\"\n                disabled={disabled || readOnly}\n                variant=\"ghost\"\n                size=\"small\"\n                icon=\"close\"\n                onClick={event => {\n                  event.stopPropagation()\n                  setTime(undefined)\n                  onChange?.(undefined)\n                }}\n                sentiment=\"neutral\"\n                data-testid=\"clear\"\n              />\n            ) : null}\n          </Stack>\n        ) : null}\n      </TimeInputWrapper>\n      {helper || error ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={error ? 'danger' : 'neutral'}\n          prominence={error ? 'default' : 'weak'}\n          disabled={disabled}\n        >\n          {error || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
|
|
86
86
|
const CustomText = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
|
|
87
87
|
target: "e8pjt8k0"
|
|
88
88
|
} : {
|
|
@@ -90,7 +90,7 @@ const CustomText = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "produ
|
|
|
90
90
|
label: "CustomText"
|
|
91
91
|
})("padding-inline:", ({
|
|
92
92
|
theme
|
|
93
|
-
}) => theme.space["0.25"], ";" + (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/TimeInputV2/index.tsx"],"names":[],"mappings":"AA2J+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport {\n  DEFAULT_DATE,\n  DEFAULT_PLACEHOLDER,\n  INPUT_SIZE_HEIGHT,\n  TIME_KEYS,\n} from './constants'\nimport {\n  canConcat,\n  format,\n  getLastTypedChar,\n  getValueByType,\n  isAOrP,\n  isCompleteHour,\n  isNumber,\n  setValueByType,\n} from './helpers'\n\nexport type Time = {\n  h: string\n  m: string\n  s: string\n  period?: string\n}\n\nconst TimeInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-error': boolean\n}>`\n  display: flex;\n  cursor: text;\n  padding: ${({ theme }) => theme.space[1]};\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n  &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n  \n  &[data-disabled=\"false\"]:hover,\n  [data-disabled=\"false\"]:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):hover {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n`\n\nexport const Input = styled.input<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-period'?: boolean\n}>`\n  border: none;\n  outline: none;\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  width: ${({ theme }) => theme.sizing[312]};\n  height: ${({ theme }) => theme.sizing[300]};\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  caret-color: transparent;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &:not(:disabled):hover {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:not(:disabled):active, \n  :not(:disabled):focus{\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrong};\n    color:  ${({ theme }) => theme.colors.neutral.text};\n  }\n\n  &:read-only {\n    cursor: default;\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-period=\"true\"] {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  ::-moz-selection {\n    background: none;\n  }\n\n  ::selection {\n    background: none;\n  }\n`\n\nconst CustomText = styled(Text)`\npadding-inline: ${({ theme }) => theme.space['0.25']};\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date | null\n  clearable?: boolean\n  required?: boolean\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  disabled?: boolean\n  readOnly?: boolean\n  error?: boolean | string\n  'data-testid'?: string\n  onChange?: (value: Date | undefined, valuePeriod?: string) => void\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  className?: string\n  id?: string\n  size?: 'small' | 'medium' | 'large'\n  timeFormat?: 12 | 24\n  /**\n   * Automatically focus on the element on render. Autofocus is applied to the hour input\n   */\n  autoFocus?: boolean\n} & (\n  | {\n      label?: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n)\n\n/**\n * A time input component that allows users to type a time in a 24 or 12-hour format.\n * @experimental This component is experimental and may be subject to breaking changes in the future.\n */\nexport const TimeInputV2 = ({\n  label,\n  timeFormat = 24,\n  value,\n  clearable,\n  required,\n  labelDescription,\n  helper,\n  size = 'medium',\n  disabled = false,\n  readOnly = false,\n  error = false,\n  onChange,\n  onBlur,\n  onFocus,\n  className,\n  id,\n  autoFocus,\n  'data-testid': dataTestId,\n  placeholder = DEFAULT_PLACEHOLDER,\n  'aria-label': ariaLabel,\n}: TimeInputProps) => {\n  const localId = useId()\n  const defaultPeriod = useMemo(() => {\n    if (value) return value.getHours() >= 12 ? 'pm' : 'am'\n\n    return undefined\n  }, [value])\n\n  const [time, setTime] = useState(value)\n  const [period, setPeriod] = useState<'pm' | 'am' | undefined>(defaultPeriod)\n  const [filled, setFilled] = useState(\n    value ? { h: true, m: true, s: true } : { h: false, m: false, s: false },\n  ) // to not show 00 when there should be a placeholder\n\n  const refHours = useRef<HTMLInputElement>(null)\n  const refSeconds = useRef<HTMLInputElement>(null)\n  const refMinutes = useRef<HTMLInputElement>(null)\n  const refPeriod = useRef<HTMLInputElement>(null)\n\n  useEffect(() => {\n    if (value) {\n      setTime(value)\n\n      // without this condition, every time an input value changes, the other ones will be set to 0 if they used to be undefined\n      // instead of leaving them empty (and showing the placeholder)\n      if (value.getTime() !== time?.getTime()) {\n        setFilled({ h: true, m: true, s: true })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [value])\n\n  const handleChangePeriod = (key: 'a' | 'p') => {\n    if (!time) {\n      setPeriod(`${key}m`)\n    } else if (key.toLowerCase() === 'a') {\n      if (time.getHours() >= 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() - 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('am')\n    } else {\n      if (time.getHours() < 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() + 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('pm')\n    }\n  }\n  const handleChange = (type: 'h' | 'm' | 's', key: number) => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const valueToChange = getValueByType(type, time)\n\n    if (canConcat(valueToChange, type, key, timeFormat)) {\n      const newValue = (valueToChange % 10) * 10 + key\n\n      setValueByType(type, newTime, newValue)\n    } else setValueByType(type, newTime, key)\n\n    const newValue = getValueByType(type, newTime)\n    // Focus to next input if the current input has a valid time\n    if (type === 's' && newTime && newValue >= 7 && timeFormat === 12) {\n      refPeriod.current?.focus()\n    } else if (type === 'm' && newTime && newValue >= 6) {\n      refSeconds.current?.focus()\n    }\n\n    if (type === 'h') {\n      if (isCompleteHour(timeFormat, newValue)) {\n        refMinutes.current?.focus()\n      }\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n\n    setFilled(newFilled)\n  }\n\n  // Increase time with arrow up\n  const handleIncrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Decrease time with arrow down\n  const handleDecrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Go to next input\n  const handleNext = (type: 'h' | 'm' | 's') => {\n    if (type === 'h') refMinutes.current?.focus()\n    if (type === 'm') refSeconds.current?.focus()\n    if (type === 's' && timeFormat === 12) refPeriod.current?.focus()\n  }\n\n  // Go to previous input\n  const handlePrevious = (type: 'h' | 'm' | 's') => {\n    if (type === 'm') refHours.current?.focus()\n    if (type === 's') refMinutes.current?.focus()\n  }\n\n  return (\n    <Stack gap={0.5} className={className}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <TimeInputWrapper\n        data-readonly={readOnly}\n        data-disabled={disabled}\n        data-size={size}\n        data-error={!!error}\n        direction=\"row\"\n        alignItems=\"center\"\n        justifyContent=\"space-between\"\n        onBlur={onBlur}\n        onFocus={onFocus}\n        aria-required={required}\n        onClick={() => refHours.current?.focus()}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n      >\n        <Stack direction=\"row\">\n          {TIME_KEYS.map(type => {\n            const computedRef = () => {\n              if (type === 'h') return refHours\n              if (type === 'm') return refMinutes\n\n              return refSeconds\n            }\n            const fullName = () => {\n              if (type === 'h') return 'hours'\n              if (type === 'm') return 'minutes'\n\n              return 'seconds'\n            }\n\n            const computeMaxValue = () => {\n              if (type === 'h' && timeFormat === 12) return 12\n              if (type === 'h' && timeFormat === 24) return 23\n\n              return 59\n            }\n\n            return (\n              <Stack key={type} direction=\"row\">\n                <Input\n                  value={\n                    filled[type]\n                      ? format(getValueByType(type, time), type, timeFormat)\n                      : ''\n                  }\n                  placeholder={placeholder[type]}\n                  data-size={size}\n                  readOnly={readOnly}\n                  disabled={disabled}\n                  aria-label={ariaLabel}\n                  data-testid={`${fullName()}-input`}\n                  onClick={event => {\n                    event.stopPropagation()\n                  }}\n                  ref={computedRef()}\n                  role=\"spinbutton\"\n                  aria-valuemax={computeMaxValue()}\n                  aria-valuemin={type === 'h' && timeFormat === 12 ? 1 : 0}\n                  aria-valuenow={\n                    filled[type]\n                      ? Number.parseInt(\n                          format(getValueByType(type, time), type, timeFormat),\n                          10,\n                        )\n                      : undefined\n                  }\n                  onChange={event => {\n                    if (!readOnly && !disabled) {\n                      const key = getLastTypedChar(\n                        event.target.value,\n                        getValueByType(type, time),\n                      )\n                      if (isNumber(key)) {\n                        handleChange(type, Number.parseInt(key, 10))\n                      }\n                    }\n                  }}\n                  onKeyDown={event => {\n                    if (!readOnly && !disabled) {\n                      if (event.key === 'ArrowUp') {\n                        event.preventDefault()\n                        handleIncrease(type)\n                      } else if (event.key === 'ArrowDown') {\n                        event.preventDefault()\n                        handleDecrease(type)\n                      } else if (event.key === 'ArrowLeft') {\n                        event.preventDefault()\n                        handlePrevious(type)\n                      } else if (event.key === 'ArrowRight') {\n                        event.preventDefault()\n                        handleNext(type)\n                      }\n                    }\n                  }}\n                  autoFocus={autoFocus && type === 'h'}\n                />\n                {type === 's' ? null : (\n                  <CustomText\n                    as=\"span\"\n                    variant=\"body\"\n                    prominence=\"default\"\n                    sentiment=\"neutral\"\n                  >\n                    :\n                  </CustomText>\n                )}\n              </Stack>\n            )\n          })}\n          {timeFormat === 12 ? (\n            <Input\n              value={period?.toUpperCase()}\n              placeholder={placeholder.period ?? 'AM'}\n              data-size={size}\n              data-period\n              readOnly={readOnly}\n              disabled={disabled}\n              aria-label={ariaLabel}\n              data-testid=\"am-pm-input\"\n              onChange={event => {\n                if (!readOnly && !disabled) {\n                  const key = event.target.value.slice(-1)\n                  if (isAOrP(key)) handleChangePeriod(key as 'a' | 'p')\n                }\n              }}\n              onKeyDown={event => {\n                if (!readOnly && !disabled) {\n                  if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {\n                    event.preventDefault()\n                    handleChangePeriod(period === 'am' ? 'p' : 'a')\n                  } else if (event.key === 'ArrowLeft') {\n                    event.preventDefault()\n                    refSeconds.current?.focus()\n                  }\n                }\n              }}\n              ref={refPeriod}\n              onClick={event => event.stopPropagation()}\n              role=\"spinbutton\"\n              aria-valuemax={12}\n              aria-valuemin={0}\n              aria-valuenow={period === 'am' ? 0 : 12}\n              aria-valuetext={period}\n            />\n          ) : null}\n        </Stack>\n        {error || clearable ? (\n          <Stack direction=\"row\" alignItems=\"center\" gap=\"1\">\n            {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            {clearable ? (\n              <Button\n                aria-label=\"clear value\"\n                disabled={disabled || readOnly}\n                variant=\"ghost\"\n                size=\"small\"\n                icon=\"close\"\n                onClick={event => {\n                  event.stopPropagation()\n                  setTime(undefined)\n                  onChange?.(undefined)\n                }}\n                sentiment=\"neutral\"\n                data-testid=\"clear\"\n              />\n            ) : null}\n          </Stack>\n        ) : null}\n      </TimeInputWrapper>\n      {helper || error ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={error ? 'danger' : 'neutral'}\n          prominence={error ? 'default' : 'weak'}\n          disabled={disabled}\n        >\n          {error || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
|
|
93
|
+
}) => theme.space["0.25"], ";" + (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/TimeInputV2/index.tsx"],"names":[],"mappings":"AA2J+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { AlertCircleIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useId, useMemo, useRef, useState } from 'react'\nimport { Button } from '../Button'\nimport { Label } from '../Label'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport {\n  DEFAULT_DATE,\n  DEFAULT_PLACEHOLDER,\n  INPUT_SIZE_HEIGHT,\n  TIME_KEYS,\n} from './constants'\nimport {\n  canConcat,\n  format,\n  getLastTypedChar,\n  getValueByType,\n  isAOrP,\n  isCompleteHour,\n  isNumber,\n  setValueByType,\n} from './helpers'\n\nexport type Time = {\n  h: string\n  m: string\n  s: string\n  period?: string\n}\n\nconst TimeInputWrapper = styled(Stack)<{\n  'data-readonly': boolean\n  'data-disabled': boolean\n  'data-size': 'small' | 'medium' | 'large'\n  'data-error': boolean\n}>`\n  display: flex;\n  cursor: text;\n  padding: ${({ theme }) => theme.space[1]};\n  box-shadow: none;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border-radius: ${({ theme }) => theme.radii.default};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n\n  &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n  }\n\n  &[data-disabled=\"false\"]:hover,\n  [data-disabled=\"false\"]:focus {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n    outline: none;\n  }\n\n  &:focus-within {\n    border-color: ${({ theme }) => theme.colors.primary.borderHover};\n  }\n\n  &[data-size='small'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.small]};\n    padding-left: ${({ theme }) => theme.space[1]};\n  }\n\n  &[data-size='medium'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.medium]};\n  }\n\n  &[data-size='large'] {\n    height: ${({ theme }) => theme.sizing[INPUT_SIZE_HEIGHT.large]};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n    cursor: default;\n  }\n\n  &[data-disabled='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-error='true'] {\n    border: 1px solid ${({ theme }) => theme.colors.danger.border};\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):active {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n      box-shadow: ${({ theme }) => theme.shadows.focusDanger};\n    }\n\n    &:not([data-disabled=\"true\"]):not([data-readonly=\"true\"]):hover {\n      border-color: ${({ theme }) => theme.colors.danger.borderHover};\n    }\n  }\n`\n\nexport const Input = styled.input<{\n  'data-size': 'small' | 'medium' | 'large'\n  'data-period'?: boolean\n}>`\n  border: none;\n  outline: none;\n  background: transparent;\n  font-size: ${({ theme }) => theme.typography.bodySmall.fontSize};\n  width: ${({ theme }) => theme.sizing[312]};\n  height: ${({ theme }) => theme.sizing[300]};\n  text-align: center;\n  border-radius: ${({ theme }) => theme.radii.default};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  caret-color: transparent;\n\n  &[data-size='large'] {\n    font-size: ${({ theme }) => theme.typography.body.fontSize};\n  }\n\n  &:not(:disabled):hover {\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundHover};\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  &:not(:disabled):active,\n  :not(:disabled):focus{\n    background-color: ${({ theme }) => theme.colors.neutral.backgroundStrong};\n    color:  ${({ theme }) => theme.colors.neutral.text};\n  }\n\n  &:read-only {\n    cursor: default;\n  }\n\n  &:disabled {\n    cursor: not-allowed;\n    user-select: none;\n  }\n\n  &[data-period=\"true\"] {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n\n  ::-moz-selection {\n    background: none;\n  }\n\n  ::selection {\n    background: none;\n  }\n`\n\nconst CustomText = styled(Text)`\npadding-inline: ${({ theme }) => theme.space['0.25']};\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date | null\n  clearable?: boolean\n  required?: boolean\n  labelDescription?: ReactNode\n  helper?: ReactNode\n  disabled?: boolean\n  readOnly?: boolean\n  error?: boolean | string\n  'data-testid'?: string\n  onChange?: (value: Date | undefined, valuePeriod?: string) => void\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  className?: string\n  id?: string\n  size?: 'small' | 'medium' | 'large'\n  timeFormat?: 12 | 24\n  /**\n   * Automatically focus on the element on render. Autofocus is applied to the hour input\n   */\n  autoFocus?: boolean\n} & (\n  | {\n      label?: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n)\n\n/**\n * A time input component that allows users to type a time in a 24 or 12-hour format.\n * @experimental This component is experimental and may be subject to breaking changes in the future.\n */\nexport const TimeInputV2 = ({\n  label,\n  timeFormat = 24,\n  value,\n  clearable,\n  required,\n  labelDescription,\n  helper,\n  size = 'medium',\n  disabled = false,\n  readOnly = false,\n  error = false,\n  onChange,\n  onBlur,\n  onFocus,\n  className,\n  id,\n  autoFocus,\n  'data-testid': dataTestId,\n  placeholder = DEFAULT_PLACEHOLDER,\n  'aria-label': ariaLabel,\n}: TimeInputProps) => {\n  const localId = useId()\n  const defaultPeriod = useMemo(() => {\n    if (value) return value.getHours() >= 12 ? 'pm' : 'am'\n\n    return undefined\n  }, [value])\n\n  const [time, setTime] = useState(value)\n  const [period, setPeriod] = useState<'pm' | 'am' | undefined>(defaultPeriod)\n  const [filled, setFilled] = useState(\n    value ? { h: true, m: true, s: true } : { h: false, m: false, s: false },\n  ) // to not show 00 when there should be a placeholder\n\n  const refHours = useRef<HTMLInputElement>(null)\n  const refSeconds = useRef<HTMLInputElement>(null)\n  const refMinutes = useRef<HTMLInputElement>(null)\n  const refPeriod = useRef<HTMLInputElement>(null)\n\n  useEffect(() => {\n    if (value) {\n      setTime(value)\n\n      // without this condition, every time an input value changes, the other ones will be set to 0 if they used to be undefined\n      // instead of leaving them empty (and showing the placeholder)\n      if (value.getTime() !== time?.getTime()) {\n        setFilled({ h: true, m: true, s: true })\n      }\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [value])\n\n  const handleChangePeriod = (key: 'a' | 'p') => {\n    if (!time) {\n      setPeriod(`${key}m`)\n    } else if (key.toLowerCase() === 'a') {\n      if (time.getHours() >= 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() - 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('am')\n    } else {\n      if (time.getHours() < 12) {\n        const newTime = new Date(time)\n        newTime.setHours(newTime.getHours() + 12)\n        setTime(newTime)\n        onChange?.(newTime)\n      }\n      setPeriod('pm')\n    }\n  }\n  const handleChange = (type: 'h' | 'm' | 's', key: number) => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const valueToChange = getValueByType(type, time)\n\n    if (canConcat(valueToChange, type, key, timeFormat)) {\n      const newValue = (valueToChange % 10) * 10 + key\n\n      setValueByType(type, newTime, newValue)\n    } else setValueByType(type, newTime, key)\n\n    const newValue = getValueByType(type, newTime)\n    // Focus to next input if the current input has a valid time\n    if (type === 's' && newTime && newValue >= 7 && timeFormat === 12) {\n      refPeriod.current?.focus()\n    } else if (type === 'm' && newTime && newValue >= 6) {\n      refSeconds.current?.focus()\n    }\n\n    if (type === 'h') {\n      if (isCompleteHour(timeFormat, newValue)) {\n        refMinutes.current?.focus()\n      }\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n\n    setFilled(newFilled)\n  }\n\n  // Increase time with arrow up\n  const handleIncrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Decrease time with arrow down\n  const handleDecrease = (type: 'h' | 'm' | 's') => {\n    const newTime = time ? new Date(time) : DEFAULT_DATE\n    const currentValue = getValueByType(type, newTime)\n\n    if (type === 'h' && timeFormat === 24) {\n      setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1)\n    } else if (type === 'h' && timeFormat === 12) {\n      setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1)\n    } else {\n      setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1)\n    }\n    const newFilled = { ...filled }\n    newFilled[type] = true\n\n    setTime(newTime)\n    onChange?.(newTime)\n    setFilled(newFilled)\n  }\n\n  // Go to next input\n  const handleNext = (type: 'h' | 'm' | 's') => {\n    if (type === 'h') refMinutes.current?.focus()\n    if (type === 'm') refSeconds.current?.focus()\n    if (type === 's' && timeFormat === 12) refPeriod.current?.focus()\n  }\n\n  // Go to previous input\n  const handlePrevious = (type: 'h' | 'm' | 's') => {\n    if (type === 'm') refHours.current?.focus()\n    if (type === 's') refMinutes.current?.focus()\n  }\n\n  return (\n    <Stack gap={0.5} className={className}>\n      {label || labelDescription ? (\n        <Label\n          labelDescription={labelDescription}\n          required={required}\n          size={size}\n          htmlFor={id ?? localId}\n        >\n          {label}\n        </Label>\n      ) : null}\n      <TimeInputWrapper\n        data-readonly={readOnly}\n        data-disabled={disabled}\n        data-size={size}\n        data-error={!!error}\n        direction=\"row\"\n        alignItems=\"center\"\n        justifyContent=\"space-between\"\n        onBlur={onBlur}\n        onFocus={onFocus}\n        aria-required={required}\n        onClick={() => refHours.current?.focus()}\n        id={id}\n        data-testid={dataTestId}\n        aria-label={ariaLabel}\n      >\n        <Stack direction=\"row\">\n          {TIME_KEYS.map(type => {\n            const computedRef = () => {\n              if (type === 'h') return refHours\n              if (type === 'm') return refMinutes\n\n              return refSeconds\n            }\n            const fullName = () => {\n              if (type === 'h') return 'hours'\n              if (type === 'm') return 'minutes'\n\n              return 'seconds'\n            }\n\n            const computeMaxValue = () => {\n              if (type === 'h' && timeFormat === 12) return 12\n              if (type === 'h' && timeFormat === 24) return 23\n\n              return 59\n            }\n\n            return (\n              <Stack key={type} direction=\"row\">\n                <Input\n                  value={\n                    filled[type]\n                      ? format(getValueByType(type, time), type, timeFormat)\n                      : ''\n                  }\n                  placeholder={placeholder[type]}\n                  data-size={size}\n                  readOnly={readOnly}\n                  disabled={disabled}\n                  aria-label={ariaLabel}\n                  data-testid={`${fullName()}-input`}\n                  onClick={event => {\n                    event.stopPropagation()\n                  }}\n                  ref={computedRef()}\n                  role=\"spinbutton\"\n                  autoComplete=\"false\"\n                  aria-valuemax={computeMaxValue()}\n                  aria-valuemin={type === 'h' && timeFormat === 12 ? 1 : 0}\n                  aria-valuenow={\n                    filled[type]\n                      ? Number.parseInt(\n                          format(getValueByType(type, time), type, timeFormat),\n                          10,\n                        )\n                      : undefined\n                  }\n                  onChange={event => {\n                    if (!readOnly && !disabled) {\n                      const key = getLastTypedChar(\n                        event.target.value,\n                        getValueByType(type, time),\n                      )\n                      if (isNumber(key)) {\n                        handleChange(type, Number.parseInt(key, 10))\n                      }\n                    }\n                  }}\n                  onKeyDown={event => {\n                    if (!readOnly && !disabled) {\n                      if (event.key === 'ArrowUp') {\n                        event.preventDefault()\n                        handleIncrease(type)\n                      } else if (event.key === 'ArrowDown') {\n                        event.preventDefault()\n                        handleDecrease(type)\n                      } else if (event.key === 'ArrowLeft') {\n                        event.preventDefault()\n                        handlePrevious(type)\n                      } else if (event.key === 'ArrowRight') {\n                        event.preventDefault()\n                        handleNext(type)\n                      }\n                    }\n                  }}\n                  autoFocus={autoFocus && type === 'h'}\n                />\n                {type === 's' ? null : (\n                  <CustomText\n                    as=\"span\"\n                    variant=\"body\"\n                    prominence=\"default\"\n                    sentiment=\"neutral\"\n                  >\n                    :\n                  </CustomText>\n                )}\n              </Stack>\n            )\n          })}\n          {timeFormat === 12 ? (\n            <Input\n              value={period?.toUpperCase()}\n              placeholder={placeholder.period ?? 'AM'}\n              data-size={size}\n              data-period\n              readOnly={readOnly}\n              disabled={disabled}\n              aria-label={ariaLabel}\n              data-testid=\"am-pm-input\"\n              onChange={event => {\n                if (!readOnly && !disabled) {\n                  const key = event.target.value.slice(-1)\n                  if (isAOrP(key)) handleChangePeriod(key as 'a' | 'p')\n                }\n              }}\n              onKeyDown={event => {\n                if (!readOnly && !disabled) {\n                  if (event.key === 'ArrowUp' || event.key === 'ArrowDown') {\n                    event.preventDefault()\n                    handleChangePeriod(period === 'am' ? 'p' : 'a')\n                  } else if (event.key === 'ArrowLeft') {\n                    event.preventDefault()\n                    refSeconds.current?.focus()\n                  }\n                }\n              }}\n              ref={refPeriod}\n              onClick={event => event.stopPropagation()}\n              role=\"spinbutton\"\n              autoComplete=\"false\"\n              aria-valuemax={12}\n              aria-valuemin={0}\n              aria-valuenow={period === 'am' ? 0 : 12}\n              aria-valuetext={period}\n            />\n          ) : null}\n        </Stack>\n        {error || clearable ? (\n          <Stack direction=\"row\" alignItems=\"center\" gap=\"1\">\n            {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            {clearable ? (\n              <Button\n                aria-label=\"clear value\"\n                disabled={disabled || readOnly}\n                variant=\"ghost\"\n                size=\"small\"\n                icon=\"close\"\n                onClick={event => {\n                  event.stopPropagation()\n                  setTime(undefined)\n                  onChange?.(undefined)\n                }}\n                sentiment=\"neutral\"\n                data-testid=\"clear\"\n              />\n            ) : null}\n          </Stack>\n        ) : null}\n      </TimeInputWrapper>\n      {helper || error ? (\n        <Text\n          as=\"p\"\n          variant=\"caption\"\n          sentiment={error ? 'danger' : 'neutral'}\n          prominence={error ? 'default' : 'weak'}\n          disabled={disabled}\n        >\n          {error || helper}\n        </Text>\n      ) : null}\n    </Stack>\n  )\n}\n"]} */"));
|
|
94
94
|
const TimeInputV2 = ({
|
|
95
95
|
label,
|
|
96
96
|
timeFormat = 24,
|
|
@@ -260,7 +260,7 @@ const TimeInputV2 = ({
|
|
|
260
260
|
return /* @__PURE__ */ jsxs(Stack, { direction: "row", children: [
|
|
261
261
|
/* @__PURE__ */ jsx(Input, { value: filled[type] ? format(getValueByType(type, time), type, timeFormat) : "", placeholder: placeholder[type], "data-size": size, readOnly, disabled, "aria-label": ariaLabel, "data-testid": `${fullName()}-input`, onClick: (event) => {
|
|
262
262
|
event.stopPropagation();
|
|
263
|
-
}, ref: computedRef(), role: "spinbutton", "aria-valuemax": computeMaxValue(), "aria-valuemin": type === "h" && timeFormat === 12 ? 1 : 0, "aria-valuenow": filled[type] ? Number.parseInt(format(getValueByType(type, time), type, timeFormat), 10) : void 0, onChange: (event) => {
|
|
263
|
+
}, ref: computedRef(), role: "spinbutton", autoComplete: "false", "aria-valuemax": computeMaxValue(), "aria-valuemin": type === "h" && timeFormat === 12 ? 1 : 0, "aria-valuenow": filled[type] ? Number.parseInt(format(getValueByType(type, time), type, timeFormat), 10) : void 0, onChange: (event) => {
|
|
264
264
|
if (!readOnly && !disabled) {
|
|
265
265
|
const key = getLastTypedChar(event.target.value, getValueByType(type, time));
|
|
266
266
|
if (isNumber(key)) {
|
|
@@ -302,7 +302,7 @@ const TimeInputV2 = ({
|
|
|
302
302
|
refSeconds.current?.focus();
|
|
303
303
|
}
|
|
304
304
|
}
|
|
305
|
-
}, ref: refPeriod, onClick: (event) => event.stopPropagation(), role: "spinbutton", "aria-valuemax": 12, "aria-valuemin": 0, "aria-valuenow": period === "am" ? 0 : 12, "aria-valuetext": period }) : null
|
|
305
|
+
}, ref: refPeriod, onClick: (event) => event.stopPropagation(), role: "spinbutton", autoComplete: "false", "aria-valuemax": 12, "aria-valuemin": 0, "aria-valuenow": period === "am" ? 0 : 12, "aria-valuetext": period }) : null
|
|
306
306
|
] }),
|
|
307
307
|
error || clearable ? /* @__PURE__ */ jsxs(Stack, { direction: "row", alignItems: "center", gap: "1", children: [
|
|
308
308
|
error ? /* @__PURE__ */ jsx(AlertCircleIcon, { sentiment: "danger" }) : null,
|
|
@@ -80,7 +80,7 @@ const StyledInput = /* @__PURE__ */ _styled__default.default("input", process.en
|
|
|
80
80
|
theme
|
|
81
81
|
}) => theme.colors.neutral.textDisabled, ";border:solid 1px ", ({
|
|
82
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  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        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        default:\n          break\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            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"]} */"));
|
|
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  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        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        default:\n          break\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            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            autoComplete=\"off\"\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"]} */"));
|
|
84
84
|
const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.env.NODE_ENV === "production" ? {
|
|
85
85
|
target: "e1a2bx9q0"
|
|
86
86
|
} : {
|
|
@@ -88,7 +88,7 @@ const FieldSet = /* @__PURE__ */ _styled__default.default("fieldset", process.en
|
|
|
88
88
|
label: "FieldSet"
|
|
89
89
|
})("border:none;padding:0;margin:0;display:flex;flex-direction:column;gap:", ({
|
|
90
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  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        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        default:\n          break\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            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"]} */"));
|
|
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  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        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        default:\n          break\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            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            autoComplete=\"off\"\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"]} */"));
|
|
92
92
|
const DEFAULT_ON_FUNCTION = () => {
|
|
93
93
|
};
|
|
94
94
|
const inputOnFocus = (event) => event.target.select();
|
|
@@ -217,7 +217,7 @@ const VerificationCode = ({
|
|
|
217
217
|
}, [error, success]);
|
|
218
218
|
return /* @__PURE__ */ jsxRuntime.jsxs(FieldSet, { className, "data-testid": dataTestId, children: [
|
|
219
219
|
label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsx(index.Label, { labelDescription, required, size: size === "xlarge" ? "large" : size, children: label }) : null,
|
|
220
|
-
/* @__PURE__ */ jsxRuntime.jsx("div", { children: values.map((value, index2) => /* @__PURE__ */ jsxRuntime.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":"AAuVY","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  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        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        default:\n          break\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            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": index2, value, id: `${inputId || uniqueId}-${index2}`, ref: inputRefs[index2], onChange: inputOnChange(index2), onKeyDown: inputOnKeyDown(index2), onPaste: inputOnPaste(index2), onFocus: inputOnFocus, disabled, required, placeholder: placeholder?.[index2] ?? "", "aria-label": `${ariaLabel} ${index2}` }, `field-${index2}`)) }),
|
|
220
|
+
/* @__PURE__ */ jsxRuntime.jsx("div", { children: values.map((value, index2) => /* @__PURE__ */ jsxRuntime.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":"AAuVY","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  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        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        default:\n          break\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            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            autoComplete=\"off\"\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": index2, value, id: `${inputId || uniqueId}-${index2}`, ref: inputRefs[index2], onChange: inputOnChange(index2), onKeyDown: inputOnKeyDown(index2), onPaste: inputOnPaste(index2), onFocus: inputOnFocus, disabled, required, placeholder: placeholder?.[index2] ?? "", "aria-label": `${ariaLabel} ${index2}`, autoComplete: "off" }, `field-${index2}`)) }),
|
|
221
221
|
error || typeof success === "string" || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
|
|
222
222
|
!error && !success && typeof helper !== "string" && helper ? helper : null
|
|
223
223
|
] });
|