@ultraviolet/ui 1.83.2 → 1.84.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (40) hide show
  1. package/dist/components/Chip/index.d.ts +1 -1
  2. package/dist/components/NumberInputV2/index.cjs +11 -7
  3. package/dist/components/NumberInputV2/index.js +11 -7
  4. package/dist/components/Pagination/PaginationButtons.cjs +84 -0
  5. package/dist/components/Pagination/PaginationButtons.d.ts +12 -0
  6. package/dist/components/Pagination/PaginationButtons.js +82 -0
  7. package/dist/components/Pagination/PerPage.cjs +65 -0
  8. package/dist/components/Pagination/PerPage.d.ts +12 -0
  9. package/dist/components/Pagination/PerPage.js +63 -0
  10. package/dist/components/Pagination/getPageNumbers.cjs +17 -17
  11. package/dist/components/Pagination/getPageNumbers.d.ts +2 -2
  12. package/dist/components/Pagination/getPageNumbers.js +17 -17
  13. package/dist/components/Pagination/index.cjs +15 -65
  14. package/dist/components/Pagination/index.d.ts +29 -2
  15. package/dist/components/Pagination/index.js +16 -64
  16. package/dist/components/Popover/index.cjs +14 -3
  17. package/dist/components/Popover/index.js +15 -4
  18. package/dist/components/Slider/components/DoubleSlider.cjs +6 -6
  19. package/dist/components/Slider/components/DoubleSlider.js +6 -6
  20. package/dist/components/Slider/components/SingleSlider.cjs +3 -3
  21. package/dist/components/Slider/components/SingleSlider.js +3 -3
  22. package/dist/components/TagList/index.cjs +6 -4
  23. package/dist/components/TagList/index.d.ts +5 -1
  24. package/dist/components/TagList/index.js +6 -4
  25. package/dist/components/TimeInputV2/constants.cjs +19 -0
  26. package/dist/components/TimeInputV2/constants.d.ts +13 -0
  27. package/dist/components/TimeInputV2/constants.js +19 -0
  28. package/dist/components/TimeInputV2/helpers.cjs +50 -0
  29. package/dist/components/TimeInputV2/helpers.d.ts +9 -0
  30. package/dist/components/TimeInputV2/helpers.js +50 -0
  31. package/dist/components/TimeInputV2/index.cjs +341 -0
  32. package/dist/components/TimeInputV2/index.d.ts +44 -0
  33. package/dist/components/TimeInputV2/index.js +339 -0
  34. package/dist/components/VerificationCode/index.cjs +70 -38
  35. package/dist/components/VerificationCode/index.d.ts +7 -2
  36. package/dist/components/VerificationCode/index.js +63 -31
  37. package/dist/components/index.d.ts +1 -0
  38. package/dist/index.cjs +17 -15
  39. package/dist/index.js +2 -0
  40. package/package.json +3 -3
@@ -0,0 +1,341 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const jsxRuntime = require("@emotion/react/jsx-runtime");
4
+ const _styled = require("@emotion/styled/base");
5
+ const Icon = require("@ultraviolet/icons");
6
+ const React = require("react");
7
+ const index$2 = require("../Button/index.cjs");
8
+ const index = require("../Stack/index.cjs");
9
+ const index$1 = require("../Text/index.cjs");
10
+ const constants = require("./constants.cjs");
11
+ const helpers = require("./helpers.cjs");
12
+ const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
13
+ const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
14
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() {
15
+ return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop).";
16
+ }
17
+ const TimeInputWrapper = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
18
+ target: "e8pjt8k3"
19
+ } : {
20
+ target: "e8pjt8k3",
21
+ label: "TimeInputWrapper"
22
+ })("display:flex;cursor:text;padding:", ({
23
+ theme
24
+ }) => theme.space[1], ";box-shadow:none;background:", ({
25
+ theme
26
+ }) => theme.colors.neutral.background, ";border-radius:", ({
27
+ theme
28
+ }) => theme.radii.default, ";border:1px solid ", ({
29
+ theme
30
+ }) => theme.colors.neutral.border, ';&:not([data-disabled="true"]):not([data-readonly="true"]):active{border-color:', ({
31
+ theme
32
+ }) => theme.colors.primary.borderHover, ";box-shadow:", ({
33
+ theme
34
+ }) => theme.shadows.focusPrimary, ';}&[data-disabled="false"]:hover,[data-disabled="false"]:focus{border-color:', ({
35
+ theme
36
+ }) => theme.colors.primary.borderHover, ";outline:none;}&:focus-within{border-color:", ({
37
+ theme
38
+ }) => theme.colors.primary.borderHover, ";}&[data-size='small']{height:", ({
39
+ theme
40
+ }) => theme.sizing[constants.INPUT_SIZE_HEIGHT.small], ";padding-left:", ({
41
+ theme
42
+ }) => theme.space[1], ";}&[data-size='medium']{height:", ({
43
+ theme
44
+ }) => theme.sizing[constants.INPUT_SIZE_HEIGHT.medium], ";}&[data-size='large']{height:", ({
45
+ theme
46
+ }) => theme.sizing[constants.INPUT_SIZE_HEIGHT.large], ";}&[data-readonly='true']{background:", ({
47
+ theme
48
+ }) => theme.colors.neutral.backgroundWeak, ";border-color:", ({
49
+ theme
50
+ }) => theme.colors.neutral.border, ";cursor:default;}&[data-disabled='true']{background:", ({
51
+ theme
52
+ }) => theme.colors.neutral.backgroundDisabled, ";border-color:", ({
53
+ theme
54
+ }) => theme.colors.neutral.borderDisabled, ";cursor:not-allowed;user-select:none;}&[data-error='true']{border:1px solid ", ({
55
+ theme
56
+ }) => theme.colors.danger.border, ';&:not([data-disabled="true"]):not([data-readonly="true"]):active{border-color:', ({
57
+ theme
58
+ }) => theme.colors.danger.borderHover, ";box-shadow:", ({
59
+ theme
60
+ }) => theme.shadows.focusDanger, ';}&:not([data-disabled="true"]):not([data-readonly="true"]):hover{border-color:', ({
61
+ theme
62
+ }) => 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":"AAqCE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, AsteriskIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { LabelProp } from '../../types'\nimport { Button } from '../Button'\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`\nconst StyledText = styled(Text)`\ncursor: text;\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date\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} & LabelProp\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 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      <Stack direction=\"row\" gap={1} alignItems=\"center\">\n        <StyledText\n          as=\"label\"\n          prominence=\"strong\"\n          sentiment=\"neutral\"\n          variant=\"body\"\n        >\n          {label}\n        </StyledText>\n        {required ? <AsteriskIcon size={8} sentiment=\"danger\" /> : null}\n        {labelDescription ? (\n          <StyledText as=\"label\" variant=\"bodySmall\">\n            {labelDescription}\n          </StyledText>\n        ) : null}\n      </Stack>\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"]} */"));
63
+ const Input = /* @__PURE__ */ _styled__default.default("input", process.env.NODE_ENV === "production" ? {
64
+ target: "e8pjt8k2"
65
+ } : {
66
+ target: "e8pjt8k2",
67
+ label: "Input"
68
+ })("border:none;outline:none;background:transparent;font-size:", ({
69
+ theme
70
+ }) => theme.typography.bodySmall.fontSize, ";width:", ({
71
+ theme
72
+ }) => theme.sizing[312], ";height:", ({
73
+ theme
74
+ }) => theme.sizing[300], ";text-align:center;border-radius:", ({
75
+ theme
76
+ }) => theme.radii.default, ";color:", ({
77
+ theme
78
+ }) => theme.colors.neutral.text, ";caret-color:transparent;&[data-size='large']{font-size:", ({
79
+ theme
80
+ }) => theme.typography.body.fontSize, ";}&:not(:disabled):hover{background-color:", ({
81
+ theme
82
+ }) => theme.colors.neutral.backgroundHover, ";color:", ({
83
+ theme
84
+ }) => theme.colors.neutral.textWeak, ";}&:not(:disabled):active,:not(:disabled):focus{background-color:", ({
85
+ theme
86
+ }) => theme.colors.neutral.backgroundStrong, ";color:", ({
87
+ theme
88
+ }) => theme.colors.neutral.text, ';}&:read-only{cursor:default;}&:disabled{cursor:not-allowed;user-select:none;}&[data-period="true"]{color:', ({
89
+ theme
90
+ }) => 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":"AAwGE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, AsteriskIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { LabelProp } from '../../types'\nimport { Button } from '../Button'\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`\nconst StyledText = styled(Text)`\ncursor: text;\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date\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} & LabelProp\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 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      <Stack direction=\"row\" gap={1} alignItems=\"center\">\n        <StyledText\n          as=\"label\"\n          prominence=\"strong\"\n          sentiment=\"neutral\"\n          variant=\"body\"\n        >\n          {label}\n        </StyledText>\n        {required ? <AsteriskIcon size={8} sentiment=\"danger\" /> : null}\n        {labelDescription ? (\n          <StyledText as=\"label\" variant=\"bodySmall\">\n            {labelDescription}\n          </StyledText>\n        ) : null}\n      </Stack>\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"]} */"));
91
+ const CustomText = /* @__PURE__ */ _styled__default.default(index$1.Text, process.env.NODE_ENV === "production" ? {
92
+ target: "e8pjt8k1"
93
+ } : {
94
+ target: "e8pjt8k1",
95
+ label: "CustomText"
96
+ })("padding-inline:", ({
97
+ theme
98
+ }) => 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":"AAyJ+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, AsteriskIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { LabelProp } from '../../types'\nimport { Button } from '../Button'\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`\nconst StyledText = styled(Text)`\ncursor: text;\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date\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} & LabelProp\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 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      <Stack direction=\"row\" gap={1} alignItems=\"center\">\n        <StyledText\n          as=\"label\"\n          prominence=\"strong\"\n          sentiment=\"neutral\"\n          variant=\"body\"\n        >\n          {label}\n        </StyledText>\n        {required ? <AsteriskIcon size={8} sentiment=\"danger\" /> : null}\n        {labelDescription ? (\n          <StyledText as=\"label\" variant=\"bodySmall\">\n            {labelDescription}\n          </StyledText>\n        ) : null}\n      </Stack>\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"]} */"));
99
+ const StyledText = /* @__PURE__ */ _styled__default.default(index$1.Text, process.env.NODE_ENV === "production" ? {
100
+ target: "e8pjt8k0"
101
+ } : {
102
+ target: "e8pjt8k0",
103
+ label: "StyledText"
104
+ })(process.env.NODE_ENV === "production" ? {
105
+ name: "1jbm2s3",
106
+ styles: "cursor:text"
107
+ } : {
108
+ name: "1jbm2s3",
109
+ styles: "cursor:text/*# 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":"AA4J+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TimeInputV2/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { AlertCircleIcon, AsteriskIcon } from '@ultraviolet/icons'\nimport type { FocusEvent, ReactNode } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport type { LabelProp } from '../../types'\nimport { Button } from '../Button'\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`\nconst StyledText = styled(Text)`\ncursor: text;\n`\n\ntype TimeInputProps = {\n  placeholder?: Time\n  value?: Date\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} & LabelProp\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 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      <Stack direction=\"row\" gap={1} alignItems=\"center\">\n        <StyledText\n          as=\"label\"\n          prominence=\"strong\"\n          sentiment=\"neutral\"\n          variant=\"body\"\n        >\n          {label}\n        </StyledText>\n        {required ? <AsteriskIcon size={8} sentiment=\"danger\" /> : null}\n        {labelDescription ? (\n          <StyledText as=\"label\" variant=\"bodySmall\">\n            {labelDescription}\n          </StyledText>\n        ) : null}\n      </Stack>\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"]} */",
110
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
111
+ });
112
+ const TimeInputV2 = ({
113
+ label,
114
+ timeFormat = 24,
115
+ value,
116
+ clearable,
117
+ required,
118
+ labelDescription,
119
+ helper,
120
+ size = "medium",
121
+ disabled = false,
122
+ readOnly = false,
123
+ error = false,
124
+ onChange,
125
+ onBlur,
126
+ onFocus,
127
+ className,
128
+ id,
129
+ autoFocus,
130
+ "data-testid": dataTestId,
131
+ placeholder = constants.DEFAULT_PLACEHOLDER,
132
+ "aria-label": ariaLabel
133
+ }) => {
134
+ const defaultPeriod = React.useMemo(() => {
135
+ if (value) return value.getHours() >= 12 ? "pm" : "am";
136
+ return void 0;
137
+ }, [value]);
138
+ const [time, setTime] = React.useState(value);
139
+ const [period, setPeriod] = React.useState(defaultPeriod);
140
+ const [filled, setFilled] = React.useState(value ? {
141
+ h: true,
142
+ m: true,
143
+ s: true
144
+ } : {
145
+ h: false,
146
+ m: false,
147
+ s: false
148
+ });
149
+ const refHours = React.useRef(null);
150
+ const refSeconds = React.useRef(null);
151
+ const refMinutes = React.useRef(null);
152
+ const refPeriod = React.useRef(null);
153
+ React.useEffect(() => {
154
+ if (value) {
155
+ setTime(value);
156
+ if (value.getTime() !== time?.getTime()) {
157
+ setFilled({
158
+ h: true,
159
+ m: true,
160
+ s: true
161
+ });
162
+ }
163
+ }
164
+ }, [value]);
165
+ const handleChangePeriod = (key) => {
166
+ if (!time) {
167
+ setPeriod(`${key}m`);
168
+ } else if (key.toLowerCase() === "a") {
169
+ if (time.getHours() >= 12) {
170
+ const newTime = new Date(time);
171
+ newTime.setHours(newTime.getHours() - 12);
172
+ setTime(newTime);
173
+ onChange?.(newTime);
174
+ }
175
+ setPeriod("am");
176
+ } else {
177
+ if (time.getHours() < 12) {
178
+ const newTime = new Date(time);
179
+ newTime.setHours(newTime.getHours() + 12);
180
+ setTime(newTime);
181
+ onChange?.(newTime);
182
+ }
183
+ setPeriod("pm");
184
+ }
185
+ };
186
+ const handleChange = (type, key) => {
187
+ const newTime = time ? new Date(time) : constants.DEFAULT_DATE;
188
+ const valueToChange = helpers.getValueByType(type, time);
189
+ if (helpers.canConcat(valueToChange, type, key, timeFormat)) {
190
+ const newValue2 = valueToChange % 10 * 10 + key;
191
+ helpers.setValueByType(type, newTime, newValue2);
192
+ } else helpers.setValueByType(type, newTime, key);
193
+ const newValue = helpers.getValueByType(type, newTime);
194
+ if (type === "s" && newTime && newValue >= 7 && timeFormat === 12) {
195
+ refPeriod.current?.focus();
196
+ } else if (type === "m" && newTime && newValue >= 6) {
197
+ refSeconds.current?.focus();
198
+ }
199
+ if (type === "h") {
200
+ if (helpers.isCompleteHour(timeFormat, newValue)) {
201
+ refMinutes.current?.focus();
202
+ }
203
+ }
204
+ const newFilled = {
205
+ ...filled
206
+ };
207
+ newFilled[type] = true;
208
+ setTime(newTime);
209
+ onChange?.(newTime);
210
+ setFilled(newFilled);
211
+ };
212
+ const handleIncrease = (type) => {
213
+ const newTime = time ? new Date(time) : constants.DEFAULT_DATE;
214
+ const currentValue = helpers.getValueByType(type, newTime);
215
+ if (type === "h" && timeFormat === 24) {
216
+ helpers.setValueByType(type, newTime, currentValue === 23 ? 0 : currentValue + 1);
217
+ } else if (type === "h" && timeFormat === 12) {
218
+ helpers.setValueByType(type, newTime, currentValue === 12 ? 1 : currentValue + 1);
219
+ } else {
220
+ helpers.setValueByType(type, newTime, currentValue === 59 ? 0 : currentValue + 1);
221
+ }
222
+ const newFilled = {
223
+ ...filled
224
+ };
225
+ newFilled[type] = true;
226
+ setTime(newTime);
227
+ onChange?.(newTime);
228
+ setFilled(newFilled);
229
+ };
230
+ const handleDecrease = (type) => {
231
+ const newTime = time ? new Date(time) : constants.DEFAULT_DATE;
232
+ const currentValue = helpers.getValueByType(type, newTime);
233
+ if (type === "h" && timeFormat === 24) {
234
+ helpers.setValueByType(type, newTime, currentValue === 0 ? 23 : currentValue - 1);
235
+ } else if (type === "h" && timeFormat === 12) {
236
+ helpers.setValueByType(type, newTime, currentValue === 1 ? 12 : currentValue - 1);
237
+ } else {
238
+ helpers.setValueByType(type, newTime, currentValue === 0 ? 59 : currentValue - 1);
239
+ }
240
+ const newFilled = {
241
+ ...filled
242
+ };
243
+ newFilled[type] = true;
244
+ setTime(newTime);
245
+ onChange?.(newTime);
246
+ setFilled(newFilled);
247
+ };
248
+ const handleNext = (type) => {
249
+ if (type === "h") refMinutes.current?.focus();
250
+ if (type === "m") refSeconds.current?.focus();
251
+ if (type === "s" && timeFormat === 12) refPeriod.current?.focus();
252
+ };
253
+ const handlePrevious = (type) => {
254
+ if (type === "m") refHours.current?.focus();
255
+ if (type === "s") refMinutes.current?.focus();
256
+ };
257
+ return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: 0.5, className, children: [
258
+ /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: 1, alignItems: "center", children: [
259
+ /* @__PURE__ */ jsxRuntime.jsx(StyledText, { as: "label", prominence: "strong", sentiment: "neutral", variant: "body", children: label }),
260
+ required ? /* @__PURE__ */ jsxRuntime.jsx(Icon.AsteriskIcon, { size: 8, sentiment: "danger" }) : null,
261
+ labelDescription ? /* @__PURE__ */ jsxRuntime.jsx(StyledText, { as: "label", variant: "bodySmall", children: labelDescription }) : null
262
+ ] }),
263
+ /* @__PURE__ */ jsxRuntime.jsxs(TimeInputWrapper, { "data-readonly": readOnly, "data-disabled": disabled, "data-size": size, "data-error": !!error, direction: "row", alignItems: "center", justifyContent: "space-between", onBlur, onFocus, "aria-required": required, onClick: () => refHours.current?.focus(), id, "data-testid": dataTestId, "aria-label": ariaLabel, children: [
264
+ /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", children: [
265
+ constants.TIME_KEYS.map((type) => {
266
+ const computedRef = () => {
267
+ if (type === "h") return refHours;
268
+ if (type === "m") return refMinutes;
269
+ return refSeconds;
270
+ };
271
+ const fullName = () => {
272
+ if (type === "h") return "hours";
273
+ if (type === "m") return "minutes";
274
+ return "seconds";
275
+ };
276
+ const computeMaxValue = () => {
277
+ if (type === "h" && timeFormat === 12) return 12;
278
+ if (type === "h" && timeFormat === 24) return 23;
279
+ return 59;
280
+ };
281
+ return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", children: [
282
+ /* @__PURE__ */ jsxRuntime.jsx(Input, { value: filled[type] ? helpers.format(helpers.getValueByType(type, time), type, timeFormat) : "", placeholder: placeholder[type], "data-size": size, readOnly, disabled, "aria-label": ariaLabel, "data-testid": `${fullName()}-input`, onClick: (event) => {
283
+ event.stopPropagation();
284
+ }, ref: computedRef(), role: "spinbutton", "aria-valuemax": computeMaxValue(), "aria-valuemin": type === "h" && timeFormat === 12 ? 1 : 0, "aria-valuenow": filled[type] ? Number.parseInt(helpers.format(helpers.getValueByType(type, time), type, timeFormat), 10) : void 0, onChange: (event) => {
285
+ if (!readOnly && !disabled) {
286
+ const key = helpers.getLastTypedChar(event.target.value, helpers.getValueByType(type, time));
287
+ if (helpers.isNumber(key)) {
288
+ handleChange(type, Number.parseInt(key, 10));
289
+ }
290
+ }
291
+ }, onKeyDown: (event) => {
292
+ if (!readOnly && !disabled) {
293
+ if (event.key === "ArrowUp") {
294
+ event.preventDefault();
295
+ handleIncrease(type);
296
+ } else if (event.key === "ArrowDown") {
297
+ event.preventDefault();
298
+ handleDecrease(type);
299
+ } else if (event.key === "ArrowLeft") {
300
+ event.preventDefault();
301
+ handlePrevious(type);
302
+ } else if (event.key === "ArrowRight") {
303
+ event.preventDefault();
304
+ handleNext(type);
305
+ }
306
+ }
307
+ }, autoFocus: autoFocus && type === "h" }),
308
+ type === "s" ? null : /* @__PURE__ */ jsxRuntime.jsx(CustomText, { as: "span", variant: "body", prominence: "default", sentiment: "neutral", children: ":" })
309
+ ] }, type);
310
+ }),
311
+ timeFormat === 12 ? /* @__PURE__ */ jsxRuntime.jsx(Input, { value: period?.toUpperCase(), placeholder: placeholder.period ?? "AM", "data-size": size, "data-period": true, readOnly, disabled, "aria-label": ariaLabel, "data-testid": "am-pm-input", onChange: (event) => {
312
+ if (!readOnly && !disabled) {
313
+ const key = event.target.value.slice(-1);
314
+ if (helpers.isAOrP(key)) handleChangePeriod(key);
315
+ }
316
+ }, onKeyDown: (event) => {
317
+ if (!readOnly && !disabled) {
318
+ if (event.key === "ArrowUp" || event.key === "ArrowDown") {
319
+ event.preventDefault();
320
+ handleChangePeriod(period === "am" ? "p" : "a");
321
+ } else if (event.key === "ArrowLeft") {
322
+ event.preventDefault();
323
+ refSeconds.current?.focus();
324
+ }
325
+ }
326
+ }, ref: refPeriod, onClick: (event) => event.stopPropagation(), role: "spinbutton", "aria-valuemax": 12, "aria-valuemin": 0, "aria-valuenow": period === "am" ? 0 : 12, "aria-valuetext": period }) : null
327
+ ] }),
328
+ error || clearable ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", alignItems: "center", gap: "1", children: [
329
+ error ? /* @__PURE__ */ jsxRuntime.jsx(Icon.AlertCircleIcon, { sentiment: "danger" }) : null,
330
+ clearable ? /* @__PURE__ */ jsxRuntime.jsx(index$2.Button, { "aria-label": "clear value", disabled: disabled || readOnly, variant: "ghost", size: "small", icon: "close", onClick: (event) => {
331
+ event.stopPropagation();
332
+ setTime(void 0);
333
+ onChange?.(void 0);
334
+ }, sentiment: "neutral", "data-testid": "clear" }) : null
335
+ ] }) : null
336
+ ] }),
337
+ helper || error ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "caption", sentiment: error ? "danger" : "neutral", prominence: error ? "default" : "weak", disabled, children: error || helper }) : null
338
+ ] });
339
+ };
340
+ exports.Input = Input;
341
+ exports.TimeInputV2 = TimeInputV2;
@@ -0,0 +1,44 @@
1
+ import type { FocusEvent, ReactNode } from 'react';
2
+ import type { LabelProp } from '../../types';
3
+ export type Time = {
4
+ h: string;
5
+ m: string;
6
+ s: string;
7
+ period?: string;
8
+ };
9
+ export declare const Input: import("@emotion/styled").StyledComponent<{
10
+ theme?: import("@emotion/react").Theme;
11
+ as?: React.ElementType;
12
+ } & {
13
+ 'data-size': "small" | "medium" | "large";
14
+ 'data-period'?: boolean;
15
+ }, import("react").DetailedHTMLProps<import("react").InputHTMLAttributes<HTMLInputElement>, HTMLInputElement>, {}>;
16
+ type TimeInputProps = {
17
+ placeholder?: Time;
18
+ value?: Date;
19
+ clearable?: boolean;
20
+ required?: boolean;
21
+ labelDescription?: ReactNode;
22
+ helper?: ReactNode;
23
+ disabled?: boolean;
24
+ readOnly?: boolean;
25
+ error?: boolean | string;
26
+ 'data-testid'?: string;
27
+ onChange?: (value: Date | undefined, valuePeriod?: string) => void;
28
+ onBlur?: (event: FocusEvent<HTMLInputElement>) => void;
29
+ onFocus?: (event: FocusEvent<HTMLInputElement>) => void;
30
+ className?: string;
31
+ id?: string;
32
+ size?: 'small' | 'medium' | 'large';
33
+ timeFormat?: 12 | 24;
34
+ /**
35
+ * Automatically focus on the element on render. Autofocus is applied to the hour input
36
+ */
37
+ autoFocus?: boolean;
38
+ } & LabelProp;
39
+ /**
40
+ * A time input component that allows users to type a time in a 24 or 12-hour format.
41
+ * @experimental This component is experimental and may be subject to breaking changes in the future.
42
+ */
43
+ export declare const TimeInputV2: ({ label, timeFormat, value, clearable, required, labelDescription, helper, size, disabled, readOnly, error, onChange, onBlur, onFocus, className, id, autoFocus, "data-testid": dataTestId, placeholder, "aria-label": ariaLabel, }: TimeInputProps) => import("@emotion/react/jsx-runtime").JSX.Element;
44
+ export {};