@ultraviolet/ui 1.83.2 → 1.84.0
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/NumberInputV2/index.cjs +11 -7
- package/dist/components/NumberInputV2/index.js +11 -7
- package/dist/components/Pagination/PaginationButtons.cjs +84 -0
- package/dist/components/Pagination/PaginationButtons.d.ts +12 -0
- package/dist/components/Pagination/PaginationButtons.js +82 -0
- package/dist/components/Pagination/PerPage.cjs +65 -0
- package/dist/components/Pagination/PerPage.d.ts +12 -0
- package/dist/components/Pagination/PerPage.js +63 -0
- package/dist/components/Pagination/getPageNumbers.cjs +17 -17
- package/dist/components/Pagination/getPageNumbers.d.ts +2 -2
- package/dist/components/Pagination/getPageNumbers.js +17 -17
- package/dist/components/Pagination/index.cjs +15 -65
- package/dist/components/Pagination/index.d.ts +29 -2
- package/dist/components/Pagination/index.js +16 -64
- package/dist/components/Slider/components/DoubleSlider.cjs +6 -6
- package/dist/components/Slider/components/DoubleSlider.js +6 -6
- package/dist/components/Slider/components/SingleSlider.cjs +3 -3
- package/dist/components/Slider/components/SingleSlider.js +3 -3
- package/dist/components/TagList/index.cjs +6 -4
- package/dist/components/TagList/index.d.ts +5 -1
- package/dist/components/TagList/index.js +6 -4
- package/dist/components/TimeInputV2/constants.cjs +19 -0
- package/dist/components/TimeInputV2/constants.d.ts +13 -0
- package/dist/components/TimeInputV2/constants.js +19 -0
- package/dist/components/TimeInputV2/helpers.cjs +50 -0
- package/dist/components/TimeInputV2/helpers.d.ts +9 -0
- package/dist/components/TimeInputV2/helpers.js +50 -0
- package/dist/components/TimeInputV2/index.cjs +341 -0
- package/dist/components/TimeInputV2/index.d.ts +44 -0
- package/dist/components/TimeInputV2/index.js +339 -0
- package/dist/components/index.d.ts +1 -0
- package/dist/index.cjs +17 -15
- package/dist/index.js +2 -0
- 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 {};
|