@ultraviolet/ui 1.85.1 → 1.85.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/dist/components/Alert/index.cjs +16 -15
  2. package/dist/components/Alert/index.js +16 -15
  3. package/dist/components/Avatar/index.cjs +4 -4
  4. package/dist/components/Avatar/index.js +4 -4
  5. package/dist/components/AvatarV2/index.cjs +8 -7
  6. package/dist/components/AvatarV2/index.js +8 -7
  7. package/dist/components/Button/constants.cjs +24 -0
  8. package/dist/components/Button/constants.d.ts +18 -0
  9. package/dist/components/Button/constants.js +24 -0
  10. package/dist/components/Button/index.cjs +8 -27
  11. package/dist/components/Button/index.d.ts +25 -6
  12. package/dist/components/Button/index.js +5 -24
  13. package/dist/components/DateInput/Context.cjs +2 -1
  14. package/dist/components/DateInput/Context.d.ts +1 -0
  15. package/dist/components/DateInput/Context.js +2 -1
  16. package/dist/components/DateInput/components/CalendarContent.cjs +77 -0
  17. package/dist/components/DateInput/components/CalendarContent.d.ts +1 -0
  18. package/dist/components/DateInput/components/CalendarContent.js +75 -0
  19. package/dist/components/DateInput/components/CalendarDaily.cjs +9 -9
  20. package/dist/components/DateInput/components/CalendarDaily.d.ts +1 -3
  21. package/dist/components/DateInput/components/CalendarDaily.js +9 -9
  22. package/dist/components/DateInput/components/CalendarMonthly.cjs +8 -8
  23. package/dist/components/DateInput/components/CalendarMonthly.d.ts +1 -3
  24. package/dist/components/DateInput/components/CalendarMonthly.js +8 -8
  25. package/dist/components/DateInput/components/Popup.cjs +8 -80
  26. package/dist/components/DateInput/components/Popup.d.ts +2 -1
  27. package/dist/components/DateInput/components/Popup.js +9 -81
  28. package/dist/components/DateInput/constants.cjs +0 -4
  29. package/dist/components/DateInput/constants.d.ts +0 -2
  30. package/dist/components/DateInput/constants.js +0 -4
  31. package/dist/components/DateInput/helpers.cjs +11 -4
  32. package/dist/components/DateInput/helpers.d.ts +5 -3
  33. package/dist/components/DateInput/helpers.js +12 -5
  34. package/dist/components/DateInput/index.cjs +26 -6
  35. package/dist/components/DateInput/index.d.ts +5 -1
  36. package/dist/components/DateInput/index.js +27 -7
  37. package/dist/components/GlobalAlert/index.cjs +3 -2
  38. package/dist/components/GlobalAlert/index.js +4 -3
  39. package/dist/components/TextArea/index.cjs +13 -12
  40. package/dist/components/TextArea/index.js +5 -4
  41. package/package.json +2 -2
@@ -1,28 +1,40 @@
1
- import { jsx } from "@emotion/react/jsx-runtime";
1
+ import { jsx, jsxs } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
3
  import { CalendarRangeIcon } from "@ultraviolet/icons";
4
4
  import { useMemo, useState, useRef, useEffect } from "react";
5
+ import { Card } from "../Card/index.js";
6
+ import { Stack } from "../Stack/index.js";
7
+ import { Text } from "../Text/index.js";
5
8
  import { TextInputV2 } from "../TextInputV2/index.js";
6
9
  import { DateInputContext } from "./Context.js";
10
+ import { CalendarContent } from "./components/CalendarContent.js";
7
11
  import { CalendarPopup } from "./components/Popup.js";
8
- import { formatValue } from "./helpers.js";
12
+ import { styleCalendarContainer, formatValue } from "./helpers.js";
9
13
  import { getMonths, getDays, getLocalizedMonths } from "./helpersLocale.js";
10
14
  function _EMOTION_STRINGIFIED_CSS_ERROR__() {
11
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).";
12
16
  }
13
17
  const Container = /* @__PURE__ */ _styled("div", process.env.NODE_ENV === "production" ? {
14
- target: "e1bm75lk0"
18
+ target: "e1bm75lk1"
15
19
  } : {
16
- target: "e1bm75lk0",
20
+ target: "e1bm75lk1",
17
21
  label: "Container"
18
22
  })(process.env.NODE_ENV === "production" ? {
19
23
  name: "1d3w5wq",
20
24
  styles: "width:100%"
21
25
  } : {
22
26
  name: "1d3w5wq",
23
- styles: "width:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/DateInput/index.tsx"],"names":[],"mappings":"AAW4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/DateInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { CalendarRangeIcon } from '@ultraviolet/icons'\nimport type { Locale } from 'date-fns'\nimport type { ChangeEvent, FocusEvent } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { TextInputV2 } from '../TextInputV2'\nimport { type ContextProps, DateInputContext } from './Context'\nimport { CalendarPopup } from './components/Popup'\nimport { formatValue } from './helpers'\nimport { getDays, getLocalizedMonths, getMonths } from './helpersLocale'\n\nconst Container = styled.div`\nwidth: 100%;`\n\ntype DateInputProps<IsRange extends undefined | boolean = false> = {\n  autoFocus?: boolean\n  locale?: string | Locale\n  disabled?: boolean\n  maxDate?: Date | null\n  minDate?: Date | null\n  name?: string\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  error?: string\n  required?: boolean\n  format?: (value?: Date) => string | undefined\n  /**\n   * Label of the field\n   */\n  label?: string\n  value?: Date | string | null\n  className?: string\n  'data-testid'?: string\n  excludeDates?: Date[]\n  id?: string\n  labelDescription?: string\n  success?: string | boolean\n  helper?: string\n  size?: 'small' | 'medium' | 'large'\n  readOnly?: boolean\n  tooltip?: string\n  showMonthYearPicker?: boolean\n  placeholder?: string\n  startDate?: Date | null\n  endDate?: Date | null\n  selectsRange?: IsRange\n  onChange?: IsRange extends true\n    ? (\n        date: Date[] | [Date | null, Date | null],\n        event?: React.SyntheticEvent,\n      ) => void\n    : (date: Date | null, event?: React.SyntheticEvent) => void\n}\n\n/**\n * DateInput can be used to select a specific date\n */\nexport const DateInput = <IsRange extends undefined | boolean>({\n  autoFocus = false,\n  disabled = false,\n  error,\n  format,\n  label,\n  labelDescription,\n  locale = 'en-US',\n  maxDate,\n  minDate,\n  startDate,\n  endDate,\n  name,\n  onBlur,\n  onChange,\n  placeholder,\n  onFocus,\n  required = false,\n  excludeDates,\n  value,\n  className,\n  id,\n  success,\n  helper,\n  size = 'large',\n  readOnly = false,\n  tooltip,\n  selectsRange = false,\n  showMonthYearPicker = false,\n  'data-testid': dataTestId,\n}: DateInputProps<IsRange>) => {\n  const defaultMonthToShow = useMemo(() => {\n    if (value) return new Date(value).getMonth() + 1\n    if (startDate && selectsRange) return startDate.getMonth() + 1\n    if (endDate && selectsRange) return endDate.getMonth() + 1\n\n    return new Date().getMonth() + 1\n  }, [endDate, selectsRange, startDate, value])\n\n  const defaultYearToShow = useMemo(() => {\n    if (value) return new Date(value).getFullYear()\n    if (startDate && selectsRange) return startDate.getFullYear()\n    if (endDate && selectsRange) return endDate.getFullYear()\n\n    return new Date().getFullYear()\n  }, [endDate, selectsRange, startDate, value])\n\n  const [computedValue, setValue] = useState(\n    value && !selectsRange ? new Date(value) : null,\n  )\n  const [computedRange, setRange] = useState({\n    start: startDate ?? null,\n    end: endDate ?? null,\n  })\n  const [isPopupVisible, setVisible] = useState(false)\n  const [monthToShow, setMonthToShow] = useState(defaultMonthToShow)\n  const [yearToShow, setYearToShow] = useState(defaultYearToShow)\n  const [inputValue, setInputValue] = useState(\n    formatValue(\n      computedValue,\n      computedRange,\n      showMonthYearPicker,\n      selectsRange,\n      format,\n    ),\n  )\n  const refInput = useRef<HTMLInputElement>(null)\n  const MONTHS = getMonths(locale)\n  const DAYS = getDays(locale)\n  const MONTHS_ARR = getLocalizedMonths(locale)\n\n  const valueContext = useMemo(\n    () =>\n      ({\n        showMonthYearPicker,\n        disabled,\n        value: computedValue,\n        range: computedRange,\n        setRange,\n        setValue,\n        monthToShow,\n        yearToShow,\n        setMonthToShow,\n        setYearToShow,\n        excludeDates,\n        maxDate,\n        minDate,\n        MONTHS,\n        MONTHS_ARR,\n        DAYS,\n        onChange,\n        selectsRange,\n        format,\n        setInputValue,\n        setVisible,\n      }) as ContextProps,\n    [\n      showMonthYearPicker,\n      disabled,\n      selectsRange,\n      computedValue,\n      computedRange,\n      monthToShow,\n      yearToShow,\n      excludeDates,\n      maxDate,\n      minDate,\n      MONTHS,\n      MONTHS_ARR,\n      DAYS,\n      onChange,\n      format,\n      setInputValue,\n      setVisible,\n    ],\n  )\n\n  useEffect(() => {\n    if (value && !selectsRange) {\n      setValue(new Date(value))\n      setInputValue(\n        formatValue(\n          new Date(value),\n          null,\n          showMonthYearPicker,\n          selectsRange,\n          format,\n        ),\n      )\n    }\n    if (selectsRange) {\n      setRange({\n        start: startDate ?? computedRange.start,\n        end: endDate ?? computedRange.end,\n      })\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [endDate, startDate, value])\n\n  const manageOnChange = (event: ChangeEvent<HTMLInputElement>) => {\n    const newValue = event.currentTarget.value\n\n    if (selectsRange) {\n      const [startDateInput, endDateInput] = newValue.split(' - ').map(val => {\n        if (showMonthYearPicker) {\n          // Force YYYY/MM (since MM/YYYY not recognised as a date in typescript)\n          const res = val.split(/\\D+/).map(aa => Number.parseInt(aa, 10))\n\n          return new Date(Math.max(...res), Math.min(...res) - 1)\n        }\n\n        return new Date(val)\n      })\n\n      const computedNewRange: [Date | null, Date | null] = [\n        startDateInput instanceof Date &&\n        !Number.isNaN(startDateInput.getTime())\n          ? startDateInput\n          : null,\n        endDateInput instanceof Date && !Number.isNaN(endDateInput.getTime())\n          ? endDateInput\n          : null,\n      ]\n\n      setRange({ start: computedNewRange[0], end: computedNewRange[1] })\n      setInputValue(newValue)\n\n      if (computedNewRange[0]) {\n        setMonthToShow(computedNewRange[0].getMonth() + 1)\n        setYearToShow(computedNewRange[0].getFullYear())\n      }\n      // TypeScript fails to automatically get the correct type of onChange here\n      ;(\n        onChange as (\n          date: Date[] | [Date | null, Date | null],\n          event: React.SyntheticEvent | undefined,\n        ) => void\n      )?.(computedNewRange, event)\n    } else {\n      const computedDate = new Date(newValue)\n      setInputValue(newValue)\n\n      if (Date.parse(newValue)) {\n        setValue(computedDate)\n        setMonthToShow(computedDate.getMonth() + 1)\n        setYearToShow(computedDate.getFullYear())\n\n        // TypeScript fails to automatically get the correct type of onChange here\n        ;(\n          onChange as (date: Date | null, event?: React.SyntheticEvent) => void\n        )?.(computedDate, event)\n      }\n    }\n  }\n\n  return (\n    <DateInputContext.Provider value={valueContext}>\n      <Container\n        id={id}\n        className={className}\n        data-testid={dataTestId}\n        onBlur={onBlur}\n        onFocus={onFocus}\n        onClick={() => {\n          if (!isPopupVisible) setVisible(true)\n        }}\n      >\n        <CalendarPopup\n          visible={isPopupVisible}\n          setVisible={setVisible}\n          refInput={refInput}\n        >\n          <TextInputV2\n            label={label}\n            placeholder={placeholder}\n            value={inputValue}\n            required={required}\n            error={error}\n            success={success}\n            readOnly={readOnly}\n            disabled={disabled}\n            size={size}\n            autoFocus={autoFocus}\n            helper={helper}\n            labelDescription={labelDescription}\n            name={name}\n            suffix={\n              <CalendarRangeIcon\n                size=\"large\"\n                sentiment=\"neutral\"\n                disabled={disabled}\n              />\n            }\n            ref={refInput}\n            tooltip={tooltip}\n            autoComplete=\"false\"\n            onChange={manageOnChange}\n          />\n        </CalendarPopup>\n      </Container>\n    </DateInputContext.Provider>\n  )\n}\n"]} */",
27
+ styles: "width:100%/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/DateInput/index.tsx"],"names":[],"mappings":"AAe4B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/DateInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { CalendarRangeIcon } from '@ultraviolet/icons'\nimport type { Locale } from 'date-fns'\nimport type { ChangeEvent, FocusEvent } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Card } from '../Card'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { TextInputV2 } from '../TextInputV2'\nimport { type ContextProps, DateInputContext } from './Context'\nimport { CalendarContent } from './components/CalendarContent'\nimport { CalendarPopup } from './components/Popup'\nimport { formatValue, styleCalendarContainer } from './helpers'\nimport { getDays, getLocalizedMonths, getMonths } from './helpersLocale'\n\nconst Container = styled.div`\nwidth: 100%;`\n\nconst StyledCard = styled(Card)`\n  ${({ theme }) => styleCalendarContainer(theme)}\n  width: 16.5rem;\n\n  &[data-disabled=\"true\"] {\n      cursor: not-allowed;\n    }\n`\n\ntype DateInputProps<IsRange extends undefined | boolean = false> = {\n  autoFocus?: boolean\n  locale?: string | Locale\n  disabled?: boolean\n  maxDate?: Date | null\n  minDate?: Date | null\n  name?: string\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  error?: string\n  required?: boolean\n  format?: (value?: Date) => string | undefined\n  /**\n   * Label of the field\n   */\n  label?: string\n  value?: Date | string | null\n  className?: string\n  'data-testid'?: string\n  excludeDates?: Date[]\n  id?: string\n  labelDescription?: string\n  success?: string | boolean\n  helper?: string\n  size?: 'small' | 'medium' | 'large'\n  readOnly?: boolean\n  tooltip?: string\n  showMonthYearPicker?: boolean\n  placeholder?: string\n  startDate?: Date | null\n  endDate?: Date | null\n  /**\n   * Display the component as an input + a calendar popup (\"text\") or only as a calendar (\"calendar\")\n   */\n  input?: 'calendar' | 'text'\n  selectsRange?: IsRange\n  onChange?: IsRange extends true\n    ? (\n        date: Date[] | [Date | null, Date | null],\n        event?: React.SyntheticEvent,\n      ) => void\n    : (date: Date | null, event?: React.SyntheticEvent) => void\n}\n\n/**\n * DateInput can be used to select a specific date\n */\nexport const DateInput = <IsRange extends undefined | boolean>({\n  autoFocus = false,\n  disabled = false,\n  error,\n  format,\n  label,\n  labelDescription,\n  locale = 'en-US',\n  maxDate,\n  minDate,\n  startDate,\n  endDate,\n  name,\n  onBlur,\n  onChange,\n  placeholder,\n  onFocus,\n  required = false,\n  excludeDates,\n  value,\n  className,\n  id,\n  success,\n  helper,\n  size = 'large',\n  readOnly = false,\n  tooltip,\n  selectsRange = false,\n  showMonthYearPicker = false,\n  input = 'text',\n  'data-testid': dataTestId,\n}: DateInputProps<IsRange>) => {\n  const defaultMonthToShow = useMemo(() => {\n    if (value) return new Date(value).getMonth() + 1\n    if (startDate && selectsRange) return startDate.getMonth() + 1\n    if (endDate && selectsRange) return endDate.getMonth() + 1\n\n    return new Date().getMonth() + 1\n  }, [endDate, selectsRange, startDate, value])\n\n  const defaultYearToShow = useMemo(() => {\n    if (value) return new Date(value).getFullYear()\n    if (startDate && selectsRange) return startDate.getFullYear()\n    if (endDate && selectsRange) return endDate.getFullYear()\n\n    return new Date().getFullYear()\n  }, [endDate, selectsRange, startDate, value])\n\n  const [computedValue, setValue] = useState(\n    value && !selectsRange ? new Date(value) : null,\n  )\n  const [computedRange, setRange] = useState({\n    start: startDate ?? null,\n    end: endDate ?? null,\n  })\n  const [isPopupVisible, setVisible] = useState(false)\n  const [monthToShow, setMonthToShow] = useState(defaultMonthToShow)\n  const [yearToShow, setYearToShow] = useState(defaultYearToShow)\n  const [inputValue, setInputValue] = useState(\n    formatValue(\n      computedValue,\n      computedRange,\n      showMonthYearPicker,\n      selectsRange,\n      format,\n    ),\n  )\n  const refInput = useRef<HTMLInputElement>(null)\n  const MONTHS = getMonths(locale)\n  const DAYS = getDays(locale)\n  const MONTHS_ARR = getLocalizedMonths(locale)\n\n  const valueContext = useMemo(\n    () =>\n      ({\n        showMonthYearPicker,\n        disabled,\n        readOnly,\n        value: computedValue,\n        range: computedRange,\n        setRange,\n        setValue,\n        monthToShow,\n        yearToShow,\n        setMonthToShow,\n        setYearToShow,\n        excludeDates,\n        maxDate,\n        minDate,\n        MONTHS,\n        MONTHS_ARR,\n        DAYS,\n        onChange,\n        selectsRange,\n        format,\n        setInputValue,\n        setVisible,\n      }) as ContextProps,\n    [\n      showMonthYearPicker,\n      disabled,\n      readOnly,\n      selectsRange,\n      computedValue,\n      computedRange,\n      monthToShow,\n      yearToShow,\n      excludeDates,\n      maxDate,\n      minDate,\n      MONTHS,\n      MONTHS_ARR,\n      DAYS,\n      onChange,\n      format,\n      setInputValue,\n      setVisible,\n    ],\n  )\n\n  useEffect(() => {\n    if (value && !selectsRange) {\n      setValue(new Date(value))\n      setInputValue(\n        formatValue(\n          new Date(value),\n          null,\n          showMonthYearPicker,\n          selectsRange,\n          format,\n        ),\n      )\n    }\n    if (selectsRange) {\n      setRange({\n        start: startDate ?? computedRange.start,\n        end: endDate ?? computedRange.end,\n      })\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [endDate, startDate, value])\n\n  const manageOnChange = (event: ChangeEvent<HTMLInputElement>) => {\n    const newValue = event.currentTarget.value\n\n    if (selectsRange) {\n      const [startDateInput, endDateInput] = newValue.split(' - ').map(val => {\n        if (showMonthYearPicker) {\n          // Force YYYY/MM (since MM/YYYY not recognised as a date in typescript)\n          const res = val.split(/\\D+/).map(aa => Number.parseInt(aa, 10))\n\n          return new Date(Math.max(...res), Math.min(...res) - 1)\n        }\n\n        return new Date(val)\n      })\n\n      const computedNewRange: [Date | null, Date | null] = [\n        startDateInput instanceof Date &&\n        !Number.isNaN(startDateInput.getTime())\n          ? startDateInput\n          : null,\n        endDateInput instanceof Date && !Number.isNaN(endDateInput.getTime())\n          ? endDateInput\n          : null,\n      ]\n\n      setRange({ start: computedNewRange[0], end: computedNewRange[1] })\n      setInputValue(newValue)\n\n      if (computedNewRange[0]) {\n        setMonthToShow(computedNewRange[0].getMonth() + 1)\n        setYearToShow(computedNewRange[0].getFullYear())\n      }\n      // TypeScript fails to automatically get the correct type of onChange here\n      ;(\n        onChange as (\n          date: Date[] | [Date | null, Date | null],\n          event: React.SyntheticEvent | undefined,\n        ) => void\n      )?.(computedNewRange, event)\n    } else {\n      const computedDate = new Date(newValue)\n      setInputValue(newValue)\n\n      if (Date.parse(newValue)) {\n        setValue(computedDate)\n        setMonthToShow(computedDate.getMonth() + 1)\n        setYearToShow(computedDate.getFullYear())\n\n        // TypeScript fails to automatically get the correct type of onChange here\n        ;(\n          onChange as (date: Date | null, event?: React.SyntheticEvent) => void\n        )?.(computedDate, event)\n      }\n    }\n  }\n\n  return (\n    <DateInputContext.Provider value={valueContext}>\n      <Container\n        id={id}\n        className={className}\n        data-testid={dataTestId}\n        onBlur={onBlur}\n        onFocus={onFocus}\n        onClick={() => {\n          if (!isPopupVisible) setVisible(true)\n        }}\n      >\n        {input === 'text' ? (\n          <CalendarPopup\n            visible={isPopupVisible}\n            setVisible={setVisible}\n            refInput={refInput}\n            content={<CalendarContent />}\n          >\n            <TextInputV2\n              label={label}\n              placeholder={placeholder}\n              value={inputValue}\n              required={required}\n              error={error}\n              success={success}\n              readOnly={readOnly}\n              disabled={disabled}\n              size={size}\n              autoFocus={autoFocus}\n              helper={helper}\n              labelDescription={labelDescription}\n              name={name}\n              suffix={\n                <CalendarRangeIcon\n                  size=\"medium\"\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                />\n              }\n              ref={refInput}\n              tooltip={tooltip}\n              autoComplete=\"false\"\n              onChange={manageOnChange}\n            />\n          </CalendarPopup>\n        ) : (\n          <Stack gap={0.5}>\n            {labelDescription ? (\n              <Stack direction=\"row\" gap=\"1\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  prominence=\"strong\"\n                  sentiment=\"neutral\"\n                >\n                  {label}\n                </Text>\n                {labelDescription}\n              </Stack>\n            ) : (\n              <Text\n                as=\"label\"\n                variant=\"bodyStrong\"\n                prominence=\"strong\"\n                sentiment=\"neutral\"\n              >\n                {label}\n              </Text>\n            )}\n\n            <StyledCard disabled={disabled}>\n              <CalendarContent />\n            </StyledCard>\n          </Stack>\n        )}\n      </Container>\n    </DateInputContext.Provider>\n  )\n}\n"]} */",
24
28
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
25
29
  });
30
+ const StyledCard = /* @__PURE__ */ _styled(Card, process.env.NODE_ENV === "production" ? {
31
+ target: "e1bm75lk0"
32
+ } : {
33
+ target: "e1bm75lk0",
34
+ label: "StyledCard"
35
+ })(({
36
+ theme
37
+ }) => styleCalendarContainer(theme), ' width:16.5rem;&[data-disabled="true"]{cursor:not-allowed;}' + (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/DateInput/index.tsx"],"names":[],"mappings":"AAkB+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/DateInput/index.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { CalendarRangeIcon } from '@ultraviolet/icons'\nimport type { Locale } from 'date-fns'\nimport type { ChangeEvent, FocusEvent } from 'react'\nimport { useEffect, useMemo, useRef, useState } from 'react'\nimport { Card } from '../Card'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { TextInputV2 } from '../TextInputV2'\nimport { type ContextProps, DateInputContext } from './Context'\nimport { CalendarContent } from './components/CalendarContent'\nimport { CalendarPopup } from './components/Popup'\nimport { formatValue, styleCalendarContainer } from './helpers'\nimport { getDays, getLocalizedMonths, getMonths } from './helpersLocale'\n\nconst Container = styled.div`\nwidth: 100%;`\n\nconst StyledCard = styled(Card)`\n  ${({ theme }) => styleCalendarContainer(theme)}\n  width: 16.5rem;\n\n  &[data-disabled=\"true\"] {\n      cursor: not-allowed;\n    }\n`\n\ntype DateInputProps<IsRange extends undefined | boolean = false> = {\n  autoFocus?: boolean\n  locale?: string | Locale\n  disabled?: boolean\n  maxDate?: Date | null\n  minDate?: Date | null\n  name?: string\n  onBlur?: (event: FocusEvent<HTMLInputElement>) => void\n  onFocus?: (event: FocusEvent<HTMLInputElement>) => void\n  error?: string\n  required?: boolean\n  format?: (value?: Date) => string | undefined\n  /**\n   * Label of the field\n   */\n  label?: string\n  value?: Date | string | null\n  className?: string\n  'data-testid'?: string\n  excludeDates?: Date[]\n  id?: string\n  labelDescription?: string\n  success?: string | boolean\n  helper?: string\n  size?: 'small' | 'medium' | 'large'\n  readOnly?: boolean\n  tooltip?: string\n  showMonthYearPicker?: boolean\n  placeholder?: string\n  startDate?: Date | null\n  endDate?: Date | null\n  /**\n   * Display the component as an input + a calendar popup (\"text\") or only as a calendar (\"calendar\")\n   */\n  input?: 'calendar' | 'text'\n  selectsRange?: IsRange\n  onChange?: IsRange extends true\n    ? (\n        date: Date[] | [Date | null, Date | null],\n        event?: React.SyntheticEvent,\n      ) => void\n    : (date: Date | null, event?: React.SyntheticEvent) => void\n}\n\n/**\n * DateInput can be used to select a specific date\n */\nexport const DateInput = <IsRange extends undefined | boolean>({\n  autoFocus = false,\n  disabled = false,\n  error,\n  format,\n  label,\n  labelDescription,\n  locale = 'en-US',\n  maxDate,\n  minDate,\n  startDate,\n  endDate,\n  name,\n  onBlur,\n  onChange,\n  placeholder,\n  onFocus,\n  required = false,\n  excludeDates,\n  value,\n  className,\n  id,\n  success,\n  helper,\n  size = 'large',\n  readOnly = false,\n  tooltip,\n  selectsRange = false,\n  showMonthYearPicker = false,\n  input = 'text',\n  'data-testid': dataTestId,\n}: DateInputProps<IsRange>) => {\n  const defaultMonthToShow = useMemo(() => {\n    if (value) return new Date(value).getMonth() + 1\n    if (startDate && selectsRange) return startDate.getMonth() + 1\n    if (endDate && selectsRange) return endDate.getMonth() + 1\n\n    return new Date().getMonth() + 1\n  }, [endDate, selectsRange, startDate, value])\n\n  const defaultYearToShow = useMemo(() => {\n    if (value) return new Date(value).getFullYear()\n    if (startDate && selectsRange) return startDate.getFullYear()\n    if (endDate && selectsRange) return endDate.getFullYear()\n\n    return new Date().getFullYear()\n  }, [endDate, selectsRange, startDate, value])\n\n  const [computedValue, setValue] = useState(\n    value && !selectsRange ? new Date(value) : null,\n  )\n  const [computedRange, setRange] = useState({\n    start: startDate ?? null,\n    end: endDate ?? null,\n  })\n  const [isPopupVisible, setVisible] = useState(false)\n  const [monthToShow, setMonthToShow] = useState(defaultMonthToShow)\n  const [yearToShow, setYearToShow] = useState(defaultYearToShow)\n  const [inputValue, setInputValue] = useState(\n    formatValue(\n      computedValue,\n      computedRange,\n      showMonthYearPicker,\n      selectsRange,\n      format,\n    ),\n  )\n  const refInput = useRef<HTMLInputElement>(null)\n  const MONTHS = getMonths(locale)\n  const DAYS = getDays(locale)\n  const MONTHS_ARR = getLocalizedMonths(locale)\n\n  const valueContext = useMemo(\n    () =>\n      ({\n        showMonthYearPicker,\n        disabled,\n        readOnly,\n        value: computedValue,\n        range: computedRange,\n        setRange,\n        setValue,\n        monthToShow,\n        yearToShow,\n        setMonthToShow,\n        setYearToShow,\n        excludeDates,\n        maxDate,\n        minDate,\n        MONTHS,\n        MONTHS_ARR,\n        DAYS,\n        onChange,\n        selectsRange,\n        format,\n        setInputValue,\n        setVisible,\n      }) as ContextProps,\n    [\n      showMonthYearPicker,\n      disabled,\n      readOnly,\n      selectsRange,\n      computedValue,\n      computedRange,\n      monthToShow,\n      yearToShow,\n      excludeDates,\n      maxDate,\n      minDate,\n      MONTHS,\n      MONTHS_ARR,\n      DAYS,\n      onChange,\n      format,\n      setInputValue,\n      setVisible,\n    ],\n  )\n\n  useEffect(() => {\n    if (value && !selectsRange) {\n      setValue(new Date(value))\n      setInputValue(\n        formatValue(\n          new Date(value),\n          null,\n          showMonthYearPicker,\n          selectsRange,\n          format,\n        ),\n      )\n    }\n    if (selectsRange) {\n      setRange({\n        start: startDate ?? computedRange.start,\n        end: endDate ?? computedRange.end,\n      })\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [endDate, startDate, value])\n\n  const manageOnChange = (event: ChangeEvent<HTMLInputElement>) => {\n    const newValue = event.currentTarget.value\n\n    if (selectsRange) {\n      const [startDateInput, endDateInput] = newValue.split(' - ').map(val => {\n        if (showMonthYearPicker) {\n          // Force YYYY/MM (since MM/YYYY not recognised as a date in typescript)\n          const res = val.split(/\\D+/).map(aa => Number.parseInt(aa, 10))\n\n          return new Date(Math.max(...res), Math.min(...res) - 1)\n        }\n\n        return new Date(val)\n      })\n\n      const computedNewRange: [Date | null, Date | null] = [\n        startDateInput instanceof Date &&\n        !Number.isNaN(startDateInput.getTime())\n          ? startDateInput\n          : null,\n        endDateInput instanceof Date && !Number.isNaN(endDateInput.getTime())\n          ? endDateInput\n          : null,\n      ]\n\n      setRange({ start: computedNewRange[0], end: computedNewRange[1] })\n      setInputValue(newValue)\n\n      if (computedNewRange[0]) {\n        setMonthToShow(computedNewRange[0].getMonth() + 1)\n        setYearToShow(computedNewRange[0].getFullYear())\n      }\n      // TypeScript fails to automatically get the correct type of onChange here\n      ;(\n        onChange as (\n          date: Date[] | [Date | null, Date | null],\n          event: React.SyntheticEvent | undefined,\n        ) => void\n      )?.(computedNewRange, event)\n    } else {\n      const computedDate = new Date(newValue)\n      setInputValue(newValue)\n\n      if (Date.parse(newValue)) {\n        setValue(computedDate)\n        setMonthToShow(computedDate.getMonth() + 1)\n        setYearToShow(computedDate.getFullYear())\n\n        // TypeScript fails to automatically get the correct type of onChange here\n        ;(\n          onChange as (date: Date | null, event?: React.SyntheticEvent) => void\n        )?.(computedDate, event)\n      }\n    }\n  }\n\n  return (\n    <DateInputContext.Provider value={valueContext}>\n      <Container\n        id={id}\n        className={className}\n        data-testid={dataTestId}\n        onBlur={onBlur}\n        onFocus={onFocus}\n        onClick={() => {\n          if (!isPopupVisible) setVisible(true)\n        }}\n      >\n        {input === 'text' ? (\n          <CalendarPopup\n            visible={isPopupVisible}\n            setVisible={setVisible}\n            refInput={refInput}\n            content={<CalendarContent />}\n          >\n            <TextInputV2\n              label={label}\n              placeholder={placeholder}\n              value={inputValue}\n              required={required}\n              error={error}\n              success={success}\n              readOnly={readOnly}\n              disabled={disabled}\n              size={size}\n              autoFocus={autoFocus}\n              helper={helper}\n              labelDescription={labelDescription}\n              name={name}\n              suffix={\n                <CalendarRangeIcon\n                  size=\"medium\"\n                  sentiment=\"neutral\"\n                  disabled={disabled}\n                />\n              }\n              ref={refInput}\n              tooltip={tooltip}\n              autoComplete=\"false\"\n              onChange={manageOnChange}\n            />\n          </CalendarPopup>\n        ) : (\n          <Stack gap={0.5}>\n            {labelDescription ? (\n              <Stack direction=\"row\" gap=\"1\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  prominence=\"strong\"\n                  sentiment=\"neutral\"\n                >\n                  {label}\n                </Text>\n                {labelDescription}\n              </Stack>\n            ) : (\n              <Text\n                as=\"label\"\n                variant=\"bodyStrong\"\n                prominence=\"strong\"\n                sentiment=\"neutral\"\n              >\n                {label}\n              </Text>\n            )}\n\n            <StyledCard disabled={disabled}>\n              <CalendarContent />\n            </StyledCard>\n          </Stack>\n        )}\n      </Container>\n    </DateInputContext.Provider>\n  )\n}\n"]} */"));
26
38
  const DateInput = ({
27
39
  autoFocus = false,
28
40
  disabled = false,
@@ -52,6 +64,7 @@ const DateInput = ({
52
64
  tooltip,
53
65
  selectsRange = false,
54
66
  showMonthYearPicker = false,
67
+ input = "text",
55
68
  "data-testid": dataTestId
56
69
  }) => {
57
70
  const defaultMonthToShow = useMemo(() => {
@@ -82,6 +95,7 @@ const DateInput = ({
82
95
  const valueContext = useMemo(() => ({
83
96
  showMonthYearPicker,
84
97
  disabled,
98
+ readOnly,
85
99
  value: computedValue,
86
100
  range: computedRange,
87
101
  setRange,
@@ -101,7 +115,7 @@ const DateInput = ({
101
115
  format,
102
116
  setInputValue,
103
117
  setVisible
104
- }), [showMonthYearPicker, disabled, selectsRange, computedValue, computedRange, monthToShow, yearToShow, excludeDates, maxDate, minDate, MONTHS, MONTHS_ARR, DAYS, onChange, format, setInputValue, setVisible]);
118
+ }), [showMonthYearPicker, disabled, readOnly, selectsRange, computedValue, computedRange, monthToShow, yearToShow, excludeDates, maxDate, minDate, MONTHS, MONTHS_ARR, DAYS, onChange, format, setInputValue, setVisible]);
105
119
  useEffect(() => {
106
120
  if (value && !selectsRange) {
107
121
  setValue(new Date(value));
@@ -148,7 +162,13 @@ const DateInput = ({
148
162
  };
149
163
  return /* @__PURE__ */ jsx(DateInputContext.Provider, { value: valueContext, children: /* @__PURE__ */ jsx(Container, { id, className, "data-testid": dataTestId, onBlur, onFocus, onClick: () => {
150
164
  if (!isPopupVisible) setVisible(true);
151
- }, children: /* @__PURE__ */ jsx(CalendarPopup, { visible: isPopupVisible, setVisible, refInput, children: /* @__PURE__ */ jsx(TextInputV2, { label, placeholder, value: inputValue, required, error, success, readOnly, disabled, size, autoFocus, helper, labelDescription, name, suffix: /* @__PURE__ */ jsx(CalendarRangeIcon, { size: "large", sentiment: "neutral", disabled }), ref: refInput, tooltip, autoComplete: "false", onChange: manageOnChange }) }) }) });
165
+ }, children: input === "text" ? /* @__PURE__ */ jsx(CalendarPopup, { visible: isPopupVisible, setVisible, refInput, content: /* @__PURE__ */ jsx(CalendarContent, {}), children: /* @__PURE__ */ jsx(TextInputV2, { label, placeholder, value: inputValue, required, error, success, readOnly, disabled, size, autoFocus, helper, labelDescription, name, suffix: /* @__PURE__ */ jsx(CalendarRangeIcon, { size: "medium", sentiment: "neutral", disabled }), ref: refInput, tooltip, autoComplete: "false", onChange: manageOnChange }) }) : /* @__PURE__ */ jsxs(Stack, { gap: 0.5, children: [
166
+ labelDescription ? /* @__PURE__ */ jsxs(Stack, { direction: "row", gap: "1", children: [
167
+ /* @__PURE__ */ jsx(Text, { as: "label", variant: "bodyStrong", prominence: "strong", sentiment: "neutral", children: label }),
168
+ labelDescription
169
+ ] }) : /* @__PURE__ */ jsx(Text, { as: "label", variant: "bodyStrong", prominence: "strong", sentiment: "neutral", children: label }),
170
+ /* @__PURE__ */ jsx(StyledCard, { disabled, children: /* @__PURE__ */ jsx(CalendarContent, {}) })
171
+ ] }) }) });
152
172
  };
153
173
  export {
154
174
  DateInput
@@ -4,6 +4,7 @@ const jsxRuntime = require("@emotion/react/jsx-runtime");
4
4
  const _styled = require("@emotion/styled/base");
5
5
  const React = require("react");
6
6
  const index = require("../Button/index.cjs");
7
+ const constants = require("../Button/constants.cjs");
7
8
  const index$1 = require("../Stack/index.cjs");
8
9
  const index$2 = require("../Text/index.cjs");
9
10
  const GlobalAlertLink = require("./GlobalAlertLink.cjs");
@@ -16,7 +17,7 @@ const CloseButton = /* @__PURE__ */ _styled__default.default(index.Button, proce
16
17
  label: "CloseButton"
17
18
  })("background:none;position:absolute;right:", ({
18
19
  theme
19
- }) => theme.sizing[index.SIZE_HEIGHT.large], ";&:hover,&:focus,&:active{background:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFRa0MiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvR2xvYmFsQWxlcnQvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlUmVkdWNlciB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQnV0dG9uLCBTSVpFX0hFSUdIVCB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcbmltcG9ydCB7IEdsb2JhbEFsZXJ0TGluayB9IGZyb20gJy4vR2xvYmFsQWxlcnRMaW5rJ1xuXG5jb25zdCBDbG9zZUJ1dHRvbiA9IHN0eWxlZChCdXR0b24pYFxuICBiYWNrZ3JvdW5kOiBub25lO1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHJpZ2h0OiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNpemluZ1tTSVpFX0hFSUdIVC5sYXJnZV19O1xuXG4gICY6aG92ZXIsXG4gICY6Zm9jdXMsXG4gICY6YWN0aXZlIHtcbiAgICBiYWNrZ3JvdW5kOiBub25lO1xuICB9XG5gXG5cbmNvbnN0IENvbnRhaW5lciA9IHN0eWxlZChTdGFjaylgXG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nWyc3MDAnXX07XG4gIHBhZGRpbmc6IDAgJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcblxuICAmW2RhdGEtdmFyaWFudD0naW5mbyddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5pbmZvLmJhY2tncm91bmRTdHJvbmd9O1xuICB9XG5cbiAgJltkYXRhLXZhcmlhbnQ9J2RhbmdlciddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5kYW5nZXIuYmFja2dyb3VuZFN0cm9uZ307XG4gIH1cblxuICAmW2RhdGEtdmFyaWFudD0ncHJvbW90aW9uYWwnXSB7XG4gICAgYmFja2dyb3VuZDogJHsoeyB0aGVtZSB9KSA9PlxuICAgICAgdGhlbWUuY29sb3JzLm90aGVyLmdyYWRpZW50cy5iYWNrZ3JvdW5kLmxpbmVhci5hcXVhfTtcbiAgfVxuYFxuXG50eXBlIEdsb2JhbEFsZXJ0UHJvcHMgPSB7XG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgdmFyaWFudD86ICdpbmZvJyB8ICdkYW5nZXInIHwgJ3Byb21vdGlvbmFsJ1xuICBvbkNsb3NlPzogKCkgPT4gdm9pZFxuICBjbG9zYWJsZT86IGJvb2xlYW5cbiAgY2xhc3NOYW1lPzogc3RyaW5nXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbiAgYnV0dG9uVGV4dD86IHN0cmluZ1xuICBvbkNsaWNrQnV0dG9uPzogKCkgPT4gdm9pZFxufVxuXG4vKipcbiAqIEdsb2JhbEFsZXJ0IGlzIGEgY29tcG9uZW50IHRoYXQgaXMgdXNlZCB0byBkaXNwbGF5IGEgZ2xvYmFsIGFsZXJ0IG1lc3NhZ2UuXG4gKiBJdCBoYXMgaXRzIG93biBpbnRlcm5hbCBzdGF0ZSBhbmQgY2FuIGJlIGNsb3NlZCBieSBjbGlja2luZyBvbiB0aGUgY2xvc2UgYnV0dG9uLlxuICovXG5leHBvcnQgY29uc3QgR2xvYmFsQWxlcnQgPSAoe1xuICBjaGlsZHJlbixcbiAgdmFyaWFudCA9ICdpbmZvJyxcbiAgb25DbG9zZSxcbiAgY2xvc2FibGUgPSB0cnVlLFxuICBidXR0b25UZXh0LFxuICBvbkNsaWNrQnV0dG9uLFxuICBjbGFzc05hbWUsXG4gICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG59OiBHbG9iYWxBbGVydFByb3BzKSA9PiB7XG4gIGNvbnN0IFtvcGVuZWQsIHRvZ2dsZU9wZW5lZF0gPSB1c2VSZWR1Y2VyKHZhbHVlID0+ICF2YWx1ZSwgdHJ1ZSlcblxuICBpZiAoIW9wZW5lZCkgcmV0dXJuIG51bGxcblxuICByZXR1cm4gKFxuICAgIDxDb250YWluZXJcbiAgICAgIGp1c3RpZnlDb250ZW50PVwiY2VudGVyXCJcbiAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgIGRhdGEtdmFyaWFudD17dmFyaWFudH1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgY2xhc3NOYW1lPXtjbGFzc05hbWV9XG4gICAgPlxuICAgICAgPFN0YWNrXG4gICAgICAgIGdhcD17Mn1cbiAgICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgID5cbiAgICAgICAgPFRleHQgdmFyaWFudD1cImJvZHlTbWFsbFwiIGFzPVwicFwiIHNlbnRpbWVudD1cIndoaXRlXCI+XG4gICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICA8L1RleHQ+XG4gICAgICAgIHtvbkNsaWNrQnV0dG9uICYmIGJ1dHRvblRleHQgPyAoXG4gICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgb25DbGljaz17b25DbGlja0J1dHRvbn1cbiAgICAgICAgICAgIHZhcmlhbnQ9XCJmaWxsZWRcIlxuICAgICAgICAgICAgc2VudGltZW50PVwid2hpdGVcIlxuICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7YnV0dG9uVGV4dH1cbiAgICAgICAgICA8L0J1dHRvbj5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICA8L1N0YWNrPlxuICAgICAge2Nsb3NhYmxlID8gKFxuICAgICAgICA8Q2xvc2VCdXR0b25cbiAgICAgICAgICB2YXJpYW50PVwiZmlsbGVkXCJcbiAgICAgICAgICBzaXplPVwieHNtYWxsXCJcbiAgICAgICAgICBpY29uPVwiY2xvc2VcIlxuICAgICAgICAgIHNlbnRpbWVudD1cInByaW1hcnlcIlxuICAgICAgICAgIG9uQ2xpY2s9eygpID0+IHtcbiAgICAgICAgICAgIHRvZ2dsZU9wZW5lZCgpXG4gICAgICAgICAgICBvbkNsb3NlPy4oKVxuICAgICAgICAgIH19XG4gICAgICAgIC8+XG4gICAgICApIDogbnVsbH1cbiAgICA8L0NvbnRhaW5lcj5cbiAgKVxufVxuXG5HbG9iYWxBbGVydC5MaW5rID0gR2xvYmFsQWxlcnRMaW5rXG4iXX0= */"));
20
+ }) => theme.sizing[constants.SIZE_HEIGHT.large], ";&:hover,&:focus,&:active{background:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFTa0MiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvR2xvYmFsQWxlcnQvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlUmVkdWNlciB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQnV0dG9uIH0gZnJvbSAnLi4vQnV0dG9uJ1xuaW1wb3J0IHsgU0laRV9IRUlHSFQgfSBmcm9tICcuLi9CdXR0b24vY29uc3RhbnRzJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9UZXh0J1xuaW1wb3J0IHsgR2xvYmFsQWxlcnRMaW5rIH0gZnJvbSAnLi9HbG9iYWxBbGVydExpbmsnXG5cbmNvbnN0IENsb3NlQnV0dG9uID0gc3R5bGVkKEJ1dHRvbilgXG4gIGJhY2tncm91bmQ6IG5vbmU7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgcmlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nW1NJWkVfSEVJR0hULmxhcmdlXX07XG5cbiAgJjpob3ZlcixcbiAgJjpmb2N1cyxcbiAgJjphY3RpdmUge1xuICAgIGJhY2tncm91bmQ6IG5vbmU7XG4gIH1cbmBcblxuY29uc3QgQ29udGFpbmVyID0gc3R5bGVkKFN0YWNrKWBcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zaXppbmdbJzcwMCddfTtcbiAgcGFkZGluZzogMCAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycyJ119O1xuXG4gICZbZGF0YS12YXJpYW50PSdpbmZvJ10ge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLmluZm8uYmFja2dyb3VuZFN0cm9uZ307XG4gIH1cblxuICAmW2RhdGEtdmFyaWFudD0nZGFuZ2VyJ10ge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLmRhbmdlci5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgfVxuXG4gICZbZGF0YS12YXJpYW50PSdwcm9tb3Rpb25hbCddIHtcbiAgICBiYWNrZ3JvdW5kOiAkeyh7IHRoZW1lIH0pID0+XG4gICAgICB0aGVtZS5jb2xvcnMub3RoZXIuZ3JhZGllbnRzLmJhY2tncm91bmQubGluZWFyLmFxdWF9O1xuICB9XG5gXG5cbnR5cGUgR2xvYmFsQWxlcnRQcm9wcyA9IHtcbiAgY2hpbGRyZW46IFJlYWN0Tm9kZVxuICB2YXJpYW50PzogJ2luZm8nIHwgJ2RhbmdlcicgfCAncHJvbW90aW9uYWwnXG4gIG9uQ2xvc2U/OiAoKSA9PiB2b2lkXG4gIGNsb3NhYmxlPzogYm9vbGVhblxuICBjbGFzc05hbWU/OiBzdHJpbmdcbiAgJ2RhdGEtdGVzdGlkJz86IHN0cmluZ1xuICBidXR0b25UZXh0Pzogc3RyaW5nXG4gIG9uQ2xpY2tCdXR0b24/OiAoKSA9PiB2b2lkXG59XG5cbi8qKlxuICogR2xvYmFsQWxlcnQgaXMgYSBjb21wb25lbnQgdGhhdCBpcyB1c2VkIHRvIGRpc3BsYXkgYSBnbG9iYWwgYWxlcnQgbWVzc2FnZS5cbiAqIEl0IGhhcyBpdHMgb3duIGludGVybmFsIHN0YXRlIGFuZCBjYW4gYmUgY2xvc2VkIGJ5IGNsaWNraW5nIG9uIHRoZSBjbG9zZSBidXR0b24uXG4gKi9cbmV4cG9ydCBjb25zdCBHbG9iYWxBbGVydCA9ICh7XG4gIGNoaWxkcmVuLFxuICB2YXJpYW50ID0gJ2luZm8nLFxuICBvbkNsb3NlLFxuICBjbG9zYWJsZSA9IHRydWUsXG4gIGJ1dHRvblRleHQsXG4gIG9uQ2xpY2tCdXR0b24sXG4gIGNsYXNzTmFtZSxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbn06IEdsb2JhbEFsZXJ0UHJvcHMpID0+IHtcbiAgY29uc3QgW29wZW5lZCwgdG9nZ2xlT3BlbmVkXSA9IHVzZVJlZHVjZXIodmFsdWUgPT4gIXZhbHVlLCB0cnVlKVxuXG4gIGlmICghb3BlbmVkKSByZXR1cm4gbnVsbFxuXG4gIHJldHVybiAoXG4gICAgPENvbnRhaW5lclxuICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgICAgYWxpZ25JdGVtcz1cImNlbnRlclwiXG4gICAgICBkaXJlY3Rpb249XCJyb3dcIlxuICAgICAgZGF0YS12YXJpYW50PXt2YXJpYW50fVxuICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICA+XG4gICAgICA8U3RhY2tcbiAgICAgICAgZ2FwPXsyfVxuICAgICAgICBkaXJlY3Rpb249XCJyb3dcIlxuICAgICAgICBqdXN0aWZ5Q29udGVudD1cImNlbnRlclwiXG4gICAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgPlxuICAgICAgICA8VGV4dCB2YXJpYW50PVwiYm9keVNtYWxsXCIgYXM9XCJwXCIgc2VudGltZW50PVwid2hpdGVcIj5cbiAgICAgICAgICB7Y2hpbGRyZW59XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAge29uQ2xpY2tCdXR0b24gJiYgYnV0dG9uVGV4dCA/IChcbiAgICAgICAgICA8QnV0dG9uXG4gICAgICAgICAgICBvbkNsaWNrPXtvbkNsaWNrQnV0dG9ufVxuICAgICAgICAgICAgdmFyaWFudD1cImZpbGxlZFwiXG4gICAgICAgICAgICBzZW50aW1lbnQ9XCJ3aGl0ZVwiXG4gICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtidXR0b25UZXh0fVxuICAgICAgICAgIDwvQnV0dG9uPlxuICAgICAgICApIDogbnVsbH1cbiAgICAgIDwvU3RhY2s+XG4gICAgICB7Y2xvc2FibGUgPyAoXG4gICAgICAgIDxDbG9zZUJ1dHRvblxuICAgICAgICAgIHZhcmlhbnQ9XCJmaWxsZWRcIlxuICAgICAgICAgIHNpemU9XCJ4c21hbGxcIlxuICAgICAgICAgIGljb249XCJjbG9zZVwiXG4gICAgICAgICAgc2VudGltZW50PVwicHJpbWFyeVwiXG4gICAgICAgICAgb25DbGljaz17KCkgPT4ge1xuICAgICAgICAgICAgdG9nZ2xlT3BlbmVkKClcbiAgICAgICAgICAgIG9uQ2xvc2U/LigpXG4gICAgICAgICAgfX1cbiAgICAgICAgLz5cbiAgICAgICkgOiBudWxsfVxuICAgIDwvQ29udGFpbmVyPlxuICApXG59XG5cbkdsb2JhbEFsZXJ0LkxpbmsgPSBHbG9iYWxBbGVydExpbmtcbiJdfQ== */"));
20
21
  const Container = /* @__PURE__ */ _styled__default.default(index$1.Stack, process.env.NODE_ENV === "production" ? {
21
22
  target: "ewvyccp0"
22
23
  } : {
@@ -32,7 +33,7 @@ const Container = /* @__PURE__ */ _styled__default.default(index$1.Stack, proces
32
33
  theme
33
34
  }) => theme.colors.danger.backgroundStrong, ";}&[data-variant='promotional']{background:", ({
34
35
  theme
35
- }) => theme.colors.other.gradients.background.linear.aqua, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFvQitCIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZVJlZHVjZXIgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJ1dHRvbiwgU0laRV9IRUlHSFQgfSBmcm9tICcuLi9CdXR0b24nXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5pbXBvcnQgeyBHbG9iYWxBbGVydExpbmsgfSBmcm9tICcuL0dsb2JhbEFsZXJ0TGluaydcblxuY29uc3QgQ2xvc2VCdXR0b24gPSBzdHlsZWQoQnV0dG9uKWBcbiAgYmFja2dyb3VuZDogbm9uZTtcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICByaWdodDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zaXppbmdbU0laRV9IRUlHSFQubGFyZ2VdfTtcblxuICAmOmhvdmVyLFxuICAmOmZvY3VzLFxuICAmOmFjdGl2ZSB7XG4gICAgYmFja2dyb3VuZDogbm9uZTtcbiAgfVxuYFxuXG5jb25zdCBDb250YWluZXIgPSBzdHlsZWQoU3RhY2spYFxuICB3aWR0aDogMTAwJTtcbiAgaGVpZ2h0OiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNpemluZ1snNzAwJ119O1xuICBwYWRkaW5nOiAwICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG5cbiAgJltkYXRhLXZhcmlhbnQ9J2luZm8nXSB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMuaW5mby5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgfVxuXG4gICZbZGF0YS12YXJpYW50PSdkYW5nZXInXSB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMuZGFuZ2VyLmJhY2tncm91bmRTdHJvbmd9O1xuICB9XG5cbiAgJltkYXRhLXZhcmlhbnQ9J3Byb21vdGlvbmFsJ10ge1xuICAgIGJhY2tncm91bmQ6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICAgIHRoZW1lLmNvbG9ycy5vdGhlci5ncmFkaWVudHMuYmFja2dyb3VuZC5saW5lYXIuYXF1YX07XG4gIH1cbmBcblxudHlwZSBHbG9iYWxBbGVydFByb3BzID0ge1xuICBjaGlsZHJlbjogUmVhY3ROb2RlXG4gIHZhcmlhbnQ/OiAnaW5mbycgfCAnZGFuZ2VyJyB8ICdwcm9tb3Rpb25hbCdcbiAgb25DbG9zZT86ICgpID0+IHZvaWRcbiAgY2xvc2FibGU/OiBib29sZWFuXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICAnZGF0YS10ZXN0aWQnPzogc3RyaW5nXG4gIGJ1dHRvblRleHQ/OiBzdHJpbmdcbiAgb25DbGlja0J1dHRvbj86ICgpID0+IHZvaWRcbn1cblxuLyoqXG4gKiBHbG9iYWxBbGVydCBpcyBhIGNvbXBvbmVudCB0aGF0IGlzIHVzZWQgdG8gZGlzcGxheSBhIGdsb2JhbCBhbGVydCBtZXNzYWdlLlxuICogSXQgaGFzIGl0cyBvd24gaW50ZXJuYWwgc3RhdGUgYW5kIGNhbiBiZSBjbG9zZWQgYnkgY2xpY2tpbmcgb24gdGhlIGNsb3NlIGJ1dHRvbi5cbiAqL1xuZXhwb3J0IGNvbnN0IEdsb2JhbEFsZXJ0ID0gKHtcbiAgY2hpbGRyZW4sXG4gIHZhcmlhbnQgPSAnaW5mbycsXG4gIG9uQ2xvc2UsXG4gIGNsb3NhYmxlID0gdHJ1ZSxcbiAgYnV0dG9uVGV4dCxcbiAgb25DbGlja0J1dHRvbixcbiAgY2xhc3NOYW1lLFxuICAnZGF0YS10ZXN0aWQnOiBkYXRhVGVzdElkLFxufTogR2xvYmFsQWxlcnRQcm9wcykgPT4ge1xuICBjb25zdCBbb3BlbmVkLCB0b2dnbGVPcGVuZWRdID0gdXNlUmVkdWNlcih2YWx1ZSA9PiAhdmFsdWUsIHRydWUpXG5cbiAgaWYgKCFvcGVuZWQpIHJldHVybiBudWxsXG5cbiAgcmV0dXJuIChcbiAgICA8Q29udGFpbmVyXG4gICAgICBqdXN0aWZ5Q29udGVudD1cImNlbnRlclwiXG4gICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgIGRpcmVjdGlvbj1cInJvd1wiXG4gICAgICBkYXRhLXZhcmlhbnQ9e3ZhcmlhbnR9XG4gICAgICBkYXRhLXRlc3RpZD17ZGF0YVRlc3RJZH1cbiAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgID5cbiAgICAgIDxTdGFja1xuICAgICAgICBnYXA9ezJ9XG4gICAgICAgIGRpcmVjdGlvbj1cInJvd1wiXG4gICAgICAgIGp1c3RpZnlDb250ZW50PVwiY2VudGVyXCJcbiAgICAgICAgYWxpZ25JdGVtcz1cImNlbnRlclwiXG4gICAgICA+XG4gICAgICAgIDxUZXh0IHZhcmlhbnQ9XCJib2R5U21hbGxcIiBhcz1cInBcIiBzZW50aW1lbnQ9XCJ3aGl0ZVwiPlxuICAgICAgICAgIHtjaGlsZHJlbn1cbiAgICAgICAgPC9UZXh0PlxuICAgICAgICB7b25DbGlja0J1dHRvbiAmJiBidXR0b25UZXh0ID8gKFxuICAgICAgICAgIDxCdXR0b25cbiAgICAgICAgICAgIG9uQ2xpY2s9e29uQ2xpY2tCdXR0b259XG4gICAgICAgICAgICB2YXJpYW50PVwiZmlsbGVkXCJcbiAgICAgICAgICAgIHNlbnRpbWVudD1cIndoaXRlXCJcbiAgICAgICAgICAgIHNpemU9XCJzbWFsbFwiXG4gICAgICAgICAgPlxuICAgICAgICAgICAge2J1dHRvblRleHR9XG4gICAgICAgICAgPC9CdXR0b24+XG4gICAgICAgICkgOiBudWxsfVxuICAgICAgPC9TdGFjaz5cbiAgICAgIHtjbG9zYWJsZSA/IChcbiAgICAgICAgPENsb3NlQnV0dG9uXG4gICAgICAgICAgdmFyaWFudD1cImZpbGxlZFwiXG4gICAgICAgICAgc2l6ZT1cInhzbWFsbFwiXG4gICAgICAgICAgaWNvbj1cImNsb3NlXCJcbiAgICAgICAgICBzZW50aW1lbnQ9XCJwcmltYXJ5XCJcbiAgICAgICAgICBvbkNsaWNrPXsoKSA9PiB7XG4gICAgICAgICAgICB0b2dnbGVPcGVuZWQoKVxuICAgICAgICAgICAgb25DbG9zZT8uKClcbiAgICAgICAgICB9fVxuICAgICAgICAvPlxuICAgICAgKSA6IG51bGx9XG4gICAgPC9Db250YWluZXI+XG4gIClcbn1cblxuR2xvYmFsQWxlcnQuTGluayA9IEdsb2JhbEFsZXJ0TGlua1xuIl19 */"));
36
+ }) => theme.colors.other.gradients.background.linear.aqua, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFxQitCIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZVJlZHVjZXIgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IFNJWkVfSEVJR0hUIH0gZnJvbSAnLi4vQnV0dG9uL2NvbnN0YW50cydcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcbmltcG9ydCB7IEdsb2JhbEFsZXJ0TGluayB9IGZyb20gJy4vR2xvYmFsQWxlcnRMaW5rJ1xuXG5jb25zdCBDbG9zZUJ1dHRvbiA9IHN0eWxlZChCdXR0b24pYFxuICBiYWNrZ3JvdW5kOiBub25lO1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHJpZ2h0OiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNpemluZ1tTSVpFX0hFSUdIVC5sYXJnZV19O1xuXG4gICY6aG92ZXIsXG4gICY6Zm9jdXMsXG4gICY6YWN0aXZlIHtcbiAgICBiYWNrZ3JvdW5kOiBub25lO1xuICB9XG5gXG5cbmNvbnN0IENvbnRhaW5lciA9IHN0eWxlZChTdGFjaylgXG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nWyc3MDAnXX07XG4gIHBhZGRpbmc6IDAgJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcblxuICAmW2RhdGEtdmFyaWFudD0naW5mbyddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5pbmZvLmJhY2tncm91bmRTdHJvbmd9O1xuICB9XG5cbiAgJltkYXRhLXZhcmlhbnQ9J2RhbmdlciddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5kYW5nZXIuYmFja2dyb3VuZFN0cm9uZ307XG4gIH1cblxuICAmW2RhdGEtdmFyaWFudD0ncHJvbW90aW9uYWwnXSB7XG4gICAgYmFja2dyb3VuZDogJHsoeyB0aGVtZSB9KSA9PlxuICAgICAgdGhlbWUuY29sb3JzLm90aGVyLmdyYWRpZW50cy5iYWNrZ3JvdW5kLmxpbmVhci5hcXVhfTtcbiAgfVxuYFxuXG50eXBlIEdsb2JhbEFsZXJ0UHJvcHMgPSB7XG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgdmFyaWFudD86ICdpbmZvJyB8ICdkYW5nZXInIHwgJ3Byb21vdGlvbmFsJ1xuICBvbkNsb3NlPzogKCkgPT4gdm9pZFxuICBjbG9zYWJsZT86IGJvb2xlYW5cbiAgY2xhc3NOYW1lPzogc3RyaW5nXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbiAgYnV0dG9uVGV4dD86IHN0cmluZ1xuICBvbkNsaWNrQnV0dG9uPzogKCkgPT4gdm9pZFxufVxuXG4vKipcbiAqIEdsb2JhbEFsZXJ0IGlzIGEgY29tcG9uZW50IHRoYXQgaXMgdXNlZCB0byBkaXNwbGF5IGEgZ2xvYmFsIGFsZXJ0IG1lc3NhZ2UuXG4gKiBJdCBoYXMgaXRzIG93biBpbnRlcm5hbCBzdGF0ZSBhbmQgY2FuIGJlIGNsb3NlZCBieSBjbGlja2luZyBvbiB0aGUgY2xvc2UgYnV0dG9uLlxuICovXG5leHBvcnQgY29uc3QgR2xvYmFsQWxlcnQgPSAoe1xuICBjaGlsZHJlbixcbiAgdmFyaWFudCA9ICdpbmZvJyxcbiAgb25DbG9zZSxcbiAgY2xvc2FibGUgPSB0cnVlLFxuICBidXR0b25UZXh0LFxuICBvbkNsaWNrQnV0dG9uLFxuICBjbGFzc05hbWUsXG4gICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG59OiBHbG9iYWxBbGVydFByb3BzKSA9PiB7XG4gIGNvbnN0IFtvcGVuZWQsIHRvZ2dsZU9wZW5lZF0gPSB1c2VSZWR1Y2VyKHZhbHVlID0+ICF2YWx1ZSwgdHJ1ZSlcblxuICBpZiAoIW9wZW5lZCkgcmV0dXJuIG51bGxcblxuICByZXR1cm4gKFxuICAgIDxDb250YWluZXJcbiAgICAgIGp1c3RpZnlDb250ZW50PVwiY2VudGVyXCJcbiAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgIGRhdGEtdmFyaWFudD17dmFyaWFudH1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgY2xhc3NOYW1lPXtjbGFzc05hbWV9XG4gICAgPlxuICAgICAgPFN0YWNrXG4gICAgICAgIGdhcD17Mn1cbiAgICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgID5cbiAgICAgICAgPFRleHQgdmFyaWFudD1cImJvZHlTbWFsbFwiIGFzPVwicFwiIHNlbnRpbWVudD1cIndoaXRlXCI+XG4gICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICA8L1RleHQ+XG4gICAgICAgIHtvbkNsaWNrQnV0dG9uICYmIGJ1dHRvblRleHQgPyAoXG4gICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgb25DbGljaz17b25DbGlja0J1dHRvbn1cbiAgICAgICAgICAgIHZhcmlhbnQ9XCJmaWxsZWRcIlxuICAgICAgICAgICAgc2VudGltZW50PVwid2hpdGVcIlxuICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7YnV0dG9uVGV4dH1cbiAgICAgICAgICA8L0J1dHRvbj5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICA8L1N0YWNrPlxuICAgICAge2Nsb3NhYmxlID8gKFxuICAgICAgICA8Q2xvc2VCdXR0b25cbiAgICAgICAgICB2YXJpYW50PVwiZmlsbGVkXCJcbiAgICAgICAgICBzaXplPVwieHNtYWxsXCJcbiAgICAgICAgICBpY29uPVwiY2xvc2VcIlxuICAgICAgICAgIHNlbnRpbWVudD1cInByaW1hcnlcIlxuICAgICAgICAgIG9uQ2xpY2s9eygpID0+IHtcbiAgICAgICAgICAgIHRvZ2dsZU9wZW5lZCgpXG4gICAgICAgICAgICBvbkNsb3NlPy4oKVxuICAgICAgICAgIH19XG4gICAgICAgIC8+XG4gICAgICApIDogbnVsbH1cbiAgICA8L0NvbnRhaW5lcj5cbiAgKVxufVxuXG5HbG9iYWxBbGVydC5MaW5rID0gR2xvYmFsQWxlcnRMaW5rXG4iXX0= */"));
36
37
  const GlobalAlert = ({
37
38
  children,
38
39
  variant = "info",
@@ -1,7 +1,8 @@
1
1
  import { jsxs, jsx } from "@emotion/react/jsx-runtime";
2
2
  import _styled from "@emotion/styled/base";
3
3
  import { useReducer } from "react";
4
- import { Button, SIZE_HEIGHT } from "../Button/index.js";
4
+ import { Button } from "../Button/index.js";
5
+ import { SIZE_HEIGHT } from "../Button/constants.js";
5
6
  import { Stack } from "../Stack/index.js";
6
7
  import { Text } from "../Text/index.js";
7
8
  import { GlobalAlertLink } from "./GlobalAlertLink.js";
@@ -12,7 +13,7 @@ const CloseButton = /* @__PURE__ */ _styled(Button, process.env.NODE_ENV === "pr
12
13
  label: "CloseButton"
13
14
  })("background:none;position:absolute;right:", ({
14
15
  theme
15
- }) => theme.sizing[SIZE_HEIGHT.large], ";&:hover,&:focus,&:active{background:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFRa0MiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvR2xvYmFsQWxlcnQvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlUmVkdWNlciB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQnV0dG9uLCBTSVpFX0hFSUdIVCB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcbmltcG9ydCB7IEdsb2JhbEFsZXJ0TGluayB9IGZyb20gJy4vR2xvYmFsQWxlcnRMaW5rJ1xuXG5jb25zdCBDbG9zZUJ1dHRvbiA9IHN0eWxlZChCdXR0b24pYFxuICBiYWNrZ3JvdW5kOiBub25lO1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHJpZ2h0OiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNpemluZ1tTSVpFX0hFSUdIVC5sYXJnZV19O1xuXG4gICY6aG92ZXIsXG4gICY6Zm9jdXMsXG4gICY6YWN0aXZlIHtcbiAgICBiYWNrZ3JvdW5kOiBub25lO1xuICB9XG5gXG5cbmNvbnN0IENvbnRhaW5lciA9IHN0eWxlZChTdGFjaylgXG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nWyc3MDAnXX07XG4gIHBhZGRpbmc6IDAgJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcblxuICAmW2RhdGEtdmFyaWFudD0naW5mbyddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5pbmZvLmJhY2tncm91bmRTdHJvbmd9O1xuICB9XG5cbiAgJltkYXRhLXZhcmlhbnQ9J2RhbmdlciddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5kYW5nZXIuYmFja2dyb3VuZFN0cm9uZ307XG4gIH1cblxuICAmW2RhdGEtdmFyaWFudD0ncHJvbW90aW9uYWwnXSB7XG4gICAgYmFja2dyb3VuZDogJHsoeyB0aGVtZSB9KSA9PlxuICAgICAgdGhlbWUuY29sb3JzLm90aGVyLmdyYWRpZW50cy5iYWNrZ3JvdW5kLmxpbmVhci5hcXVhfTtcbiAgfVxuYFxuXG50eXBlIEdsb2JhbEFsZXJ0UHJvcHMgPSB7XG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgdmFyaWFudD86ICdpbmZvJyB8ICdkYW5nZXInIHwgJ3Byb21vdGlvbmFsJ1xuICBvbkNsb3NlPzogKCkgPT4gdm9pZFxuICBjbG9zYWJsZT86IGJvb2xlYW5cbiAgY2xhc3NOYW1lPzogc3RyaW5nXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbiAgYnV0dG9uVGV4dD86IHN0cmluZ1xuICBvbkNsaWNrQnV0dG9uPzogKCkgPT4gdm9pZFxufVxuXG4vKipcbiAqIEdsb2JhbEFsZXJ0IGlzIGEgY29tcG9uZW50IHRoYXQgaXMgdXNlZCB0byBkaXNwbGF5IGEgZ2xvYmFsIGFsZXJ0IG1lc3NhZ2UuXG4gKiBJdCBoYXMgaXRzIG93biBpbnRlcm5hbCBzdGF0ZSBhbmQgY2FuIGJlIGNsb3NlZCBieSBjbGlja2luZyBvbiB0aGUgY2xvc2UgYnV0dG9uLlxuICovXG5leHBvcnQgY29uc3QgR2xvYmFsQWxlcnQgPSAoe1xuICBjaGlsZHJlbixcbiAgdmFyaWFudCA9ICdpbmZvJyxcbiAgb25DbG9zZSxcbiAgY2xvc2FibGUgPSB0cnVlLFxuICBidXR0b25UZXh0LFxuICBvbkNsaWNrQnV0dG9uLFxuICBjbGFzc05hbWUsXG4gICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG59OiBHbG9iYWxBbGVydFByb3BzKSA9PiB7XG4gIGNvbnN0IFtvcGVuZWQsIHRvZ2dsZU9wZW5lZF0gPSB1c2VSZWR1Y2VyKHZhbHVlID0+ICF2YWx1ZSwgdHJ1ZSlcblxuICBpZiAoIW9wZW5lZCkgcmV0dXJuIG51bGxcblxuICByZXR1cm4gKFxuICAgIDxDb250YWluZXJcbiAgICAgIGp1c3RpZnlDb250ZW50PVwiY2VudGVyXCJcbiAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgIGRhdGEtdmFyaWFudD17dmFyaWFudH1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgY2xhc3NOYW1lPXtjbGFzc05hbWV9XG4gICAgPlxuICAgICAgPFN0YWNrXG4gICAgICAgIGdhcD17Mn1cbiAgICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgID5cbiAgICAgICAgPFRleHQgdmFyaWFudD1cImJvZHlTbWFsbFwiIGFzPVwicFwiIHNlbnRpbWVudD1cIndoaXRlXCI+XG4gICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICA8L1RleHQ+XG4gICAgICAgIHtvbkNsaWNrQnV0dG9uICYmIGJ1dHRvblRleHQgPyAoXG4gICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgb25DbGljaz17b25DbGlja0J1dHRvbn1cbiAgICAgICAgICAgIHZhcmlhbnQ9XCJmaWxsZWRcIlxuICAgICAgICAgICAgc2VudGltZW50PVwid2hpdGVcIlxuICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7YnV0dG9uVGV4dH1cbiAgICAgICAgICA8L0J1dHRvbj5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICA8L1N0YWNrPlxuICAgICAge2Nsb3NhYmxlID8gKFxuICAgICAgICA8Q2xvc2VCdXR0b25cbiAgICAgICAgICB2YXJpYW50PVwiZmlsbGVkXCJcbiAgICAgICAgICBzaXplPVwieHNtYWxsXCJcbiAgICAgICAgICBpY29uPVwiY2xvc2VcIlxuICAgICAgICAgIHNlbnRpbWVudD1cInByaW1hcnlcIlxuICAgICAgICAgIG9uQ2xpY2s9eygpID0+IHtcbiAgICAgICAgICAgIHRvZ2dsZU9wZW5lZCgpXG4gICAgICAgICAgICBvbkNsb3NlPy4oKVxuICAgICAgICAgIH19XG4gICAgICAgIC8+XG4gICAgICApIDogbnVsbH1cbiAgICA8L0NvbnRhaW5lcj5cbiAgKVxufVxuXG5HbG9iYWxBbGVydC5MaW5rID0gR2xvYmFsQWxlcnRMaW5rXG4iXX0= */"));
16
+ }) => theme.sizing[SIZE_HEIGHT.large], ";&:hover,&:focus,&:active{background:none;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFTa0MiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvdWkvc3JjL2NvbXBvbmVudHMvR2xvYmFsQWxlcnQvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgdHlwZSB7IFJlYWN0Tm9kZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlUmVkdWNlciB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgQnV0dG9uIH0gZnJvbSAnLi4vQnV0dG9uJ1xuaW1wb3J0IHsgU0laRV9IRUlHSFQgfSBmcm9tICcuLi9CdXR0b24vY29uc3RhbnRzJ1xuaW1wb3J0IHsgU3RhY2sgfSBmcm9tICcuLi9TdGFjaydcbmltcG9ydCB7IFRleHQgfSBmcm9tICcuLi9UZXh0J1xuaW1wb3J0IHsgR2xvYmFsQWxlcnRMaW5rIH0gZnJvbSAnLi9HbG9iYWxBbGVydExpbmsnXG5cbmNvbnN0IENsb3NlQnV0dG9uID0gc3R5bGVkKEJ1dHRvbilgXG4gIGJhY2tncm91bmQ6IG5vbmU7XG4gIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgcmlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nW1NJWkVfSEVJR0hULmxhcmdlXX07XG5cbiAgJjpob3ZlcixcbiAgJjpmb2N1cyxcbiAgJjphY3RpdmUge1xuICAgIGJhY2tncm91bmQ6IG5vbmU7XG4gIH1cbmBcblxuY29uc3QgQ29udGFpbmVyID0gc3R5bGVkKFN0YWNrKWBcbiAgd2lkdGg6IDEwMCU7XG4gIGhlaWdodDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zaXppbmdbJzcwMCddfTtcbiAgcGFkZGluZzogMCAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNwYWNlWycyJ119O1xuXG4gICZbZGF0YS12YXJpYW50PSdpbmZvJ10ge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLmluZm8uYmFja2dyb3VuZFN0cm9uZ307XG4gIH1cblxuICAmW2RhdGEtdmFyaWFudD0nZGFuZ2VyJ10ge1xuICAgIGJhY2tncm91bmQtY29sb3I6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuY29sb3JzLmRhbmdlci5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgfVxuXG4gICZbZGF0YS12YXJpYW50PSdwcm9tb3Rpb25hbCddIHtcbiAgICBiYWNrZ3JvdW5kOiAkeyh7IHRoZW1lIH0pID0+XG4gICAgICB0aGVtZS5jb2xvcnMub3RoZXIuZ3JhZGllbnRzLmJhY2tncm91bmQubGluZWFyLmFxdWF9O1xuICB9XG5gXG5cbnR5cGUgR2xvYmFsQWxlcnRQcm9wcyA9IHtcbiAgY2hpbGRyZW46IFJlYWN0Tm9kZVxuICB2YXJpYW50PzogJ2luZm8nIHwgJ2RhbmdlcicgfCAncHJvbW90aW9uYWwnXG4gIG9uQ2xvc2U/OiAoKSA9PiB2b2lkXG4gIGNsb3NhYmxlPzogYm9vbGVhblxuICBjbGFzc05hbWU/OiBzdHJpbmdcbiAgJ2RhdGEtdGVzdGlkJz86IHN0cmluZ1xuICBidXR0b25UZXh0Pzogc3RyaW5nXG4gIG9uQ2xpY2tCdXR0b24/OiAoKSA9PiB2b2lkXG59XG5cbi8qKlxuICogR2xvYmFsQWxlcnQgaXMgYSBjb21wb25lbnQgdGhhdCBpcyB1c2VkIHRvIGRpc3BsYXkgYSBnbG9iYWwgYWxlcnQgbWVzc2FnZS5cbiAqIEl0IGhhcyBpdHMgb3duIGludGVybmFsIHN0YXRlIGFuZCBjYW4gYmUgY2xvc2VkIGJ5IGNsaWNraW5nIG9uIHRoZSBjbG9zZSBidXR0b24uXG4gKi9cbmV4cG9ydCBjb25zdCBHbG9iYWxBbGVydCA9ICh7XG4gIGNoaWxkcmVuLFxuICB2YXJpYW50ID0gJ2luZm8nLFxuICBvbkNsb3NlLFxuICBjbG9zYWJsZSA9IHRydWUsXG4gIGJ1dHRvblRleHQsXG4gIG9uQ2xpY2tCdXR0b24sXG4gIGNsYXNzTmFtZSxcbiAgJ2RhdGEtdGVzdGlkJzogZGF0YVRlc3RJZCxcbn06IEdsb2JhbEFsZXJ0UHJvcHMpID0+IHtcbiAgY29uc3QgW29wZW5lZCwgdG9nZ2xlT3BlbmVkXSA9IHVzZVJlZHVjZXIodmFsdWUgPT4gIXZhbHVlLCB0cnVlKVxuXG4gIGlmICghb3BlbmVkKSByZXR1cm4gbnVsbFxuXG4gIHJldHVybiAoXG4gICAgPENvbnRhaW5lclxuICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgICAgYWxpZ25JdGVtcz1cImNlbnRlclwiXG4gICAgICBkaXJlY3Rpb249XCJyb3dcIlxuICAgICAgZGF0YS12YXJpYW50PXt2YXJpYW50fVxuICAgICAgZGF0YS10ZXN0aWQ9e2RhdGFUZXN0SWR9XG4gICAgICBjbGFzc05hbWU9e2NsYXNzTmFtZX1cbiAgICA+XG4gICAgICA8U3RhY2tcbiAgICAgICAgZ2FwPXsyfVxuICAgICAgICBkaXJlY3Rpb249XCJyb3dcIlxuICAgICAgICBqdXN0aWZ5Q29udGVudD1cImNlbnRlclwiXG4gICAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgPlxuICAgICAgICA8VGV4dCB2YXJpYW50PVwiYm9keVNtYWxsXCIgYXM9XCJwXCIgc2VudGltZW50PVwid2hpdGVcIj5cbiAgICAgICAgICB7Y2hpbGRyZW59XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAge29uQ2xpY2tCdXR0b24gJiYgYnV0dG9uVGV4dCA/IChcbiAgICAgICAgICA8QnV0dG9uXG4gICAgICAgICAgICBvbkNsaWNrPXtvbkNsaWNrQnV0dG9ufVxuICAgICAgICAgICAgdmFyaWFudD1cImZpbGxlZFwiXG4gICAgICAgICAgICBzZW50aW1lbnQ9XCJ3aGl0ZVwiXG4gICAgICAgICAgICBzaXplPVwic21hbGxcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtidXR0b25UZXh0fVxuICAgICAgICAgIDwvQnV0dG9uPlxuICAgICAgICApIDogbnVsbH1cbiAgICAgIDwvU3RhY2s+XG4gICAgICB7Y2xvc2FibGUgPyAoXG4gICAgICAgIDxDbG9zZUJ1dHRvblxuICAgICAgICAgIHZhcmlhbnQ9XCJmaWxsZWRcIlxuICAgICAgICAgIHNpemU9XCJ4c21hbGxcIlxuICAgICAgICAgIGljb249XCJjbG9zZVwiXG4gICAgICAgICAgc2VudGltZW50PVwicHJpbWFyeVwiXG4gICAgICAgICAgb25DbGljaz17KCkgPT4ge1xuICAgICAgICAgICAgdG9nZ2xlT3BlbmVkKClcbiAgICAgICAgICAgIG9uQ2xvc2U/LigpXG4gICAgICAgICAgfX1cbiAgICAgICAgLz5cbiAgICAgICkgOiBudWxsfVxuICAgIDwvQ29udGFpbmVyPlxuICApXG59XG5cbkdsb2JhbEFsZXJ0LkxpbmsgPSBHbG9iYWxBbGVydExpbmtcbiJdfQ== */"));
16
17
  const Container = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "production" ? {
17
18
  target: "ewvyccp0"
18
19
  } : {
@@ -28,7 +29,7 @@ const Container = /* @__PURE__ */ _styled(Stack, process.env.NODE_ENV === "produ
28
29
  theme
29
30
  }) => theme.colors.danger.backgroundStrong, ";}&[data-variant='promotional']{background:", ({
30
31
  theme
31
- }) => theme.colors.other.gradients.background.linear.aqua, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFvQitCIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZVJlZHVjZXIgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJ1dHRvbiwgU0laRV9IRUlHSFQgfSBmcm9tICcuLi9CdXR0b24nXG5pbXBvcnQgeyBTdGFjayB9IGZyb20gJy4uL1N0YWNrJ1xuaW1wb3J0IHsgVGV4dCB9IGZyb20gJy4uL1RleHQnXG5pbXBvcnQgeyBHbG9iYWxBbGVydExpbmsgfSBmcm9tICcuL0dsb2JhbEFsZXJ0TGluaydcblxuY29uc3QgQ2xvc2VCdXR0b24gPSBzdHlsZWQoQnV0dG9uKWBcbiAgYmFja2dyb3VuZDogbm9uZTtcbiAgcG9zaXRpb246IGFic29sdXRlO1xuICByaWdodDogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zaXppbmdbU0laRV9IRUlHSFQubGFyZ2VdfTtcblxuICAmOmhvdmVyLFxuICAmOmZvY3VzLFxuICAmOmFjdGl2ZSB7XG4gICAgYmFja2dyb3VuZDogbm9uZTtcbiAgfVxuYFxuXG5jb25zdCBDb250YWluZXIgPSBzdHlsZWQoU3RhY2spYFxuICB3aWR0aDogMTAwJTtcbiAgaGVpZ2h0OiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNpemluZ1snNzAwJ119O1xuICBwYWRkaW5nOiAwICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzInXX07XG5cbiAgJltkYXRhLXZhcmlhbnQ9J2luZm8nXSB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMuaW5mby5iYWNrZ3JvdW5kU3Ryb25nfTtcbiAgfVxuXG4gICZbZGF0YS12YXJpYW50PSdkYW5nZXInXSB7XG4gICAgYmFja2dyb3VuZC1jb2xvcjogJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5jb2xvcnMuZGFuZ2VyLmJhY2tncm91bmRTdHJvbmd9O1xuICB9XG5cbiAgJltkYXRhLXZhcmlhbnQ9J3Byb21vdGlvbmFsJ10ge1xuICAgIGJhY2tncm91bmQ6ICR7KHsgdGhlbWUgfSkgPT5cbiAgICAgIHRoZW1lLmNvbG9ycy5vdGhlci5ncmFkaWVudHMuYmFja2dyb3VuZC5saW5lYXIuYXF1YX07XG4gIH1cbmBcblxudHlwZSBHbG9iYWxBbGVydFByb3BzID0ge1xuICBjaGlsZHJlbjogUmVhY3ROb2RlXG4gIHZhcmlhbnQ/OiAnaW5mbycgfCAnZGFuZ2VyJyB8ICdwcm9tb3Rpb25hbCdcbiAgb25DbG9zZT86ICgpID0+IHZvaWRcbiAgY2xvc2FibGU/OiBib29sZWFuXG4gIGNsYXNzTmFtZT86IHN0cmluZ1xuICAnZGF0YS10ZXN0aWQnPzogc3RyaW5nXG4gIGJ1dHRvblRleHQ/OiBzdHJpbmdcbiAgb25DbGlja0J1dHRvbj86ICgpID0+IHZvaWRcbn1cblxuLyoqXG4gKiBHbG9iYWxBbGVydCBpcyBhIGNvbXBvbmVudCB0aGF0IGlzIHVzZWQgdG8gZGlzcGxheSBhIGdsb2JhbCBhbGVydCBtZXNzYWdlLlxuICogSXQgaGFzIGl0cyBvd24gaW50ZXJuYWwgc3RhdGUgYW5kIGNhbiBiZSBjbG9zZWQgYnkgY2xpY2tpbmcgb24gdGhlIGNsb3NlIGJ1dHRvbi5cbiAqL1xuZXhwb3J0IGNvbnN0IEdsb2JhbEFsZXJ0ID0gKHtcbiAgY2hpbGRyZW4sXG4gIHZhcmlhbnQgPSAnaW5mbycsXG4gIG9uQ2xvc2UsXG4gIGNsb3NhYmxlID0gdHJ1ZSxcbiAgYnV0dG9uVGV4dCxcbiAgb25DbGlja0J1dHRvbixcbiAgY2xhc3NOYW1lLFxuICAnZGF0YS10ZXN0aWQnOiBkYXRhVGVzdElkLFxufTogR2xvYmFsQWxlcnRQcm9wcykgPT4ge1xuICBjb25zdCBbb3BlbmVkLCB0b2dnbGVPcGVuZWRdID0gdXNlUmVkdWNlcih2YWx1ZSA9PiAhdmFsdWUsIHRydWUpXG5cbiAgaWYgKCFvcGVuZWQpIHJldHVybiBudWxsXG5cbiAgcmV0dXJuIChcbiAgICA8Q29udGFpbmVyXG4gICAgICBqdXN0aWZ5Q29udGVudD1cImNlbnRlclwiXG4gICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgIGRpcmVjdGlvbj1cInJvd1wiXG4gICAgICBkYXRhLXZhcmlhbnQ9e3ZhcmlhbnR9XG4gICAgICBkYXRhLXRlc3RpZD17ZGF0YVRlc3RJZH1cbiAgICAgIGNsYXNzTmFtZT17Y2xhc3NOYW1lfVxuICAgID5cbiAgICAgIDxTdGFja1xuICAgICAgICBnYXA9ezJ9XG4gICAgICAgIGRpcmVjdGlvbj1cInJvd1wiXG4gICAgICAgIGp1c3RpZnlDb250ZW50PVwiY2VudGVyXCJcbiAgICAgICAgYWxpZ25JdGVtcz1cImNlbnRlclwiXG4gICAgICA+XG4gICAgICAgIDxUZXh0IHZhcmlhbnQ9XCJib2R5U21hbGxcIiBhcz1cInBcIiBzZW50aW1lbnQ9XCJ3aGl0ZVwiPlxuICAgICAgICAgIHtjaGlsZHJlbn1cbiAgICAgICAgPC9UZXh0PlxuICAgICAgICB7b25DbGlja0J1dHRvbiAmJiBidXR0b25UZXh0ID8gKFxuICAgICAgICAgIDxCdXR0b25cbiAgICAgICAgICAgIG9uQ2xpY2s9e29uQ2xpY2tCdXR0b259XG4gICAgICAgICAgICB2YXJpYW50PVwiZmlsbGVkXCJcbiAgICAgICAgICAgIHNlbnRpbWVudD1cIndoaXRlXCJcbiAgICAgICAgICAgIHNpemU9XCJzbWFsbFwiXG4gICAgICAgICAgPlxuICAgICAgICAgICAge2J1dHRvblRleHR9XG4gICAgICAgICAgPC9CdXR0b24+XG4gICAgICAgICkgOiBudWxsfVxuICAgICAgPC9TdGFjaz5cbiAgICAgIHtjbG9zYWJsZSA/IChcbiAgICAgICAgPENsb3NlQnV0dG9uXG4gICAgICAgICAgdmFyaWFudD1cImZpbGxlZFwiXG4gICAgICAgICAgc2l6ZT1cInhzbWFsbFwiXG4gICAgICAgICAgaWNvbj1cImNsb3NlXCJcbiAgICAgICAgICBzZW50aW1lbnQ9XCJwcmltYXJ5XCJcbiAgICAgICAgICBvbkNsaWNrPXsoKSA9PiB7XG4gICAgICAgICAgICB0b2dnbGVPcGVuZWQoKVxuICAgICAgICAgICAgb25DbG9zZT8uKClcbiAgICAgICAgICB9fVxuICAgICAgICAvPlxuICAgICAgKSA6IG51bGx9XG4gICAgPC9Db250YWluZXI+XG4gIClcbn1cblxuR2xvYmFsQWxlcnQuTGluayA9IEdsb2JhbEFsZXJ0TGlua1xuIl19 */"));
32
+ }) => theme.colors.other.gradients.background.linear.aqua, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFxQitCIiwiZmlsZSI6Ii9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3VpL3NyYy9jb21wb25lbnRzL0dsb2JhbEFsZXJ0L2luZGV4LnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJ1xuaW1wb3J0IHR5cGUgeyBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZVJlZHVjZXIgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IEJ1dHRvbiB9IGZyb20gJy4uL0J1dHRvbidcbmltcG9ydCB7IFNJWkVfSEVJR0hUIH0gZnJvbSAnLi4vQnV0dG9uL2NvbnN0YW50cydcbmltcG9ydCB7IFN0YWNrIH0gZnJvbSAnLi4vU3RhY2snXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vVGV4dCdcbmltcG9ydCB7IEdsb2JhbEFsZXJ0TGluayB9IGZyb20gJy4vR2xvYmFsQWxlcnRMaW5rJ1xuXG5jb25zdCBDbG9zZUJ1dHRvbiA9IHN0eWxlZChCdXR0b24pYFxuICBiYWNrZ3JvdW5kOiBub25lO1xuICBwb3NpdGlvbjogYWJzb2x1dGU7XG4gIHJpZ2h0OiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLnNpemluZ1tTSVpFX0hFSUdIVC5sYXJnZV19O1xuXG4gICY6aG92ZXIsXG4gICY6Zm9jdXMsXG4gICY6YWN0aXZlIHtcbiAgICBiYWNrZ3JvdW5kOiBub25lO1xuICB9XG5gXG5cbmNvbnN0IENvbnRhaW5lciA9IHN0eWxlZChTdGFjaylgXG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc2l6aW5nWyc3MDAnXX07XG4gIHBhZGRpbmc6IDAgJHsoeyB0aGVtZSB9KSA9PiB0aGVtZS5zcGFjZVsnMiddfTtcblxuICAmW2RhdGEtdmFyaWFudD0naW5mbyddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5pbmZvLmJhY2tncm91bmRTdHJvbmd9O1xuICB9XG5cbiAgJltkYXRhLXZhcmlhbnQ9J2RhbmdlciddIHtcbiAgICBiYWNrZ3JvdW5kLWNvbG9yOiAkeyh7IHRoZW1lIH0pID0+IHRoZW1lLmNvbG9ycy5kYW5nZXIuYmFja2dyb3VuZFN0cm9uZ307XG4gIH1cblxuICAmW2RhdGEtdmFyaWFudD0ncHJvbW90aW9uYWwnXSB7XG4gICAgYmFja2dyb3VuZDogJHsoeyB0aGVtZSB9KSA9PlxuICAgICAgdGhlbWUuY29sb3JzLm90aGVyLmdyYWRpZW50cy5iYWNrZ3JvdW5kLmxpbmVhci5hcXVhfTtcbiAgfVxuYFxuXG50eXBlIEdsb2JhbEFsZXJ0UHJvcHMgPSB7XG4gIGNoaWxkcmVuOiBSZWFjdE5vZGVcbiAgdmFyaWFudD86ICdpbmZvJyB8ICdkYW5nZXInIHwgJ3Byb21vdGlvbmFsJ1xuICBvbkNsb3NlPzogKCkgPT4gdm9pZFxuICBjbG9zYWJsZT86IGJvb2xlYW5cbiAgY2xhc3NOYW1lPzogc3RyaW5nXG4gICdkYXRhLXRlc3RpZCc/OiBzdHJpbmdcbiAgYnV0dG9uVGV4dD86IHN0cmluZ1xuICBvbkNsaWNrQnV0dG9uPzogKCkgPT4gdm9pZFxufVxuXG4vKipcbiAqIEdsb2JhbEFsZXJ0IGlzIGEgY29tcG9uZW50IHRoYXQgaXMgdXNlZCB0byBkaXNwbGF5IGEgZ2xvYmFsIGFsZXJ0IG1lc3NhZ2UuXG4gKiBJdCBoYXMgaXRzIG93biBpbnRlcm5hbCBzdGF0ZSBhbmQgY2FuIGJlIGNsb3NlZCBieSBjbGlja2luZyBvbiB0aGUgY2xvc2UgYnV0dG9uLlxuICovXG5leHBvcnQgY29uc3QgR2xvYmFsQWxlcnQgPSAoe1xuICBjaGlsZHJlbixcbiAgdmFyaWFudCA9ICdpbmZvJyxcbiAgb25DbG9zZSxcbiAgY2xvc2FibGUgPSB0cnVlLFxuICBidXR0b25UZXh0LFxuICBvbkNsaWNrQnV0dG9uLFxuICBjbGFzc05hbWUsXG4gICdkYXRhLXRlc3RpZCc6IGRhdGFUZXN0SWQsXG59OiBHbG9iYWxBbGVydFByb3BzKSA9PiB7XG4gIGNvbnN0IFtvcGVuZWQsIHRvZ2dsZU9wZW5lZF0gPSB1c2VSZWR1Y2VyKHZhbHVlID0+ICF2YWx1ZSwgdHJ1ZSlcblxuICBpZiAoIW9wZW5lZCkgcmV0dXJuIG51bGxcblxuICByZXR1cm4gKFxuICAgIDxDb250YWluZXJcbiAgICAgIGp1c3RpZnlDb250ZW50PVwiY2VudGVyXCJcbiAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgIGRhdGEtdmFyaWFudD17dmFyaWFudH1cbiAgICAgIGRhdGEtdGVzdGlkPXtkYXRhVGVzdElkfVxuICAgICAgY2xhc3NOYW1lPXtjbGFzc05hbWV9XG4gICAgPlxuICAgICAgPFN0YWNrXG4gICAgICAgIGdhcD17Mn1cbiAgICAgICAgZGlyZWN0aW9uPVwicm93XCJcbiAgICAgICAganVzdGlmeUNvbnRlbnQ9XCJjZW50ZXJcIlxuICAgICAgICBhbGlnbkl0ZW1zPVwiY2VudGVyXCJcbiAgICAgID5cbiAgICAgICAgPFRleHQgdmFyaWFudD1cImJvZHlTbWFsbFwiIGFzPVwicFwiIHNlbnRpbWVudD1cIndoaXRlXCI+XG4gICAgICAgICAge2NoaWxkcmVufVxuICAgICAgICA8L1RleHQ+XG4gICAgICAgIHtvbkNsaWNrQnV0dG9uICYmIGJ1dHRvblRleHQgPyAoXG4gICAgICAgICAgPEJ1dHRvblxuICAgICAgICAgICAgb25DbGljaz17b25DbGlja0J1dHRvbn1cbiAgICAgICAgICAgIHZhcmlhbnQ9XCJmaWxsZWRcIlxuICAgICAgICAgICAgc2VudGltZW50PVwid2hpdGVcIlxuICAgICAgICAgICAgc2l6ZT1cInNtYWxsXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7YnV0dG9uVGV4dH1cbiAgICAgICAgICA8L0J1dHRvbj5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICA8L1N0YWNrPlxuICAgICAge2Nsb3NhYmxlID8gKFxuICAgICAgICA8Q2xvc2VCdXR0b25cbiAgICAgICAgICB2YXJpYW50PVwiZmlsbGVkXCJcbiAgICAgICAgICBzaXplPVwieHNtYWxsXCJcbiAgICAgICAgICBpY29uPVwiY2xvc2VcIlxuICAgICAgICAgIHNlbnRpbWVudD1cInByaW1hcnlcIlxuICAgICAgICAgIG9uQ2xpY2s9eygpID0+IHtcbiAgICAgICAgICAgIHRvZ2dsZU9wZW5lZCgpXG4gICAgICAgICAgICBvbkNsb3NlPy4oKVxuICAgICAgICAgIH19XG4gICAgICAgIC8+XG4gICAgICApIDogbnVsbH1cbiAgICA8L0NvbnRhaW5lcj5cbiAgKVxufVxuXG5HbG9iYWxBbGVydC5MaW5rID0gR2xvYmFsQWxlcnRMaW5rXG4iXX0= */"));
32
33
  const GlobalAlert = ({
33
34
  children,
34
35
  variant = "info",
@@ -5,11 +5,12 @@ const _styled = require("@emotion/styled/base");
5
5
  const react = require("@emotion/react");
6
6
  const Icon = require("@ultraviolet/icons");
7
7
  const React = require("react");
8
- const index$1 = require("../Button/index.cjs");
8
+ const index$3 = require("../Button/index.cjs");
9
+ const constants = require("../Button/constants.cjs");
9
10
  const index$4 = require("../Row/index.cjs");
10
11
  const index = require("../Stack/index.cjs");
11
- const index$2 = require("../Text/index.cjs");
12
- const index$3 = require("../Tooltip/index.cjs");
12
+ const index$1 = require("../Text/index.cjs");
13
+ const index$2 = require("../Tooltip/index.cjs");
13
14
  const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
14
15
  const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
15
16
  function _EMOTION_STRINGIFIED_CSS_ERROR__() {
@@ -26,7 +27,7 @@ const StyledTextAreaWrapper = /* @__PURE__ */ _styled__default.default("div", pr
26
27
  styles: "position:relative;display:flex"
27
28
  } : {
28
29
  name: "8k1832",
29
- styles: "position:relative;display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AAwBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
30
+ styles: "position:relative;display:flex/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx"],"names":[],"mappings":"AAyBwC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */",
30
31
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
31
32
  });
32
33
  const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(index.Stack, process.env.NODE_ENV === "production" ? {
@@ -38,7 +39,7 @@ const StyledTextAreaAbsoluteStack = /* @__PURE__ */ _styled__default.default(ind
38
39
  theme
39
40
  }) => theme.space["1.5"], ";right:", ({
40
41
  theme
41
- }) => theme.space["1"], ";" + (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/TextArea/index.tsx"],"names":[],"mappings":"AA6BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
42
+ }) => theme.space["1"], ";" + (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/TextArea/index.tsx"],"names":[],"mappings":"AA8BiD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
42
43
  const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", process.env.NODE_ENV === "production" ? {
43
44
  shouldForwardProp: (prop) => !["hasSentimentIcon", "isClearable"].includes(prop),
44
45
  target: "enu776d0"
@@ -64,7 +65,7 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
64
65
  hasSentimentIcon
65
66
  }) => (
66
67
  /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */
67
- `calc(${theme.space[isClearable && hasSentimentIcon ? "4" : "3"]} + ${isClearable ? `${index$1.SIZE_HEIGHT.xsmall}px` : "0px"} + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : "0px"})`
68
+ `calc(${theme.space[isClearable && hasSentimentIcon ? "4" : "3"]} + ${isClearable ? `${constants.SIZE_HEIGHT.xsmall}px` : "0px"} + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : "0px"})`
68
69
  ), ";&[data-success='true']{border-color:", ({
69
70
  theme
70
71
  }) => theme.colors.success.border, ";}&[data-error='true']{border-color:", ({
@@ -87,7 +88,7 @@ const StyledTextArea = /* @__PURE__ */ _styled__default.default("textarea", proc
87
88
  theme
88
89
  }) => theme.colors.primary.border, ";box-shadow:", ({
89
90
  theme
90
- }) => theme.shadows.focusPrimary, ";}}" + (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/TextArea/index.tsx"],"names":[],"mappings":"AA0CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button, SIZE_HEIGHT as ButtonSizeHeight } from '../Button'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
91
+ }) => theme.shadows.focusPrimary, ";}}" + (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/TextArea/index.tsx"],"names":[],"mappings":"AA2CuB","file":"/home/runner/work/ultraviolet/ultraviolet/packages/ui/src/components/TextArea/index.tsx","sourcesContent":["import { useTheme } from '@emotion/react'\nimport styled from '@emotion/styled'\nimport {\n  AlertCircleIcon,\n  AsteriskIcon,\n  CheckCircleIcon,\n} from '@ultraviolet/icons'\nimport type { DOMAttributes, ReactNode } from 'react'\nimport {\n  forwardRef,\n  useEffect,\n  useId,\n  useImperativeHandle,\n  useMemo,\n  useRef,\n} from 'react'\nimport { Button } from '../Button'\nimport { SIZE_HEIGHT as ButtonSizeHeight } from '../Button/constants'\nimport { Row } from '../Row'\nimport { Stack } from '../Stack'\nimport { Text } from '../Text'\nimport { Tooltip } from '../Tooltip'\n\nconst STATE_ICON_SIZE = 16\n\nconst StyledTextAreaWrapper = styled.div`\n  position: relative;\n  display: flex;\n`\n\nconst StyledTextAreaAbsoluteStack = styled(Stack)`\n  position: absolute;\n  top: ${({ theme }) => theme.space['1.5']};\n  right: ${({ theme }) => theme.space['1']};\n`\n\ntype StyledTextAreaProps = {\n  hasSentimentIcon: boolean\n  isClearable: boolean\n}\nconst StyledTextArea = styled('textarea', {\n  shouldForwardProp: prop =>\n    !['hasSentimentIcon', 'isClearable'].includes(prop),\n})<StyledTextAreaProps>`\n  width: 100%;\n  resize: vertical;\n  background: ${({ theme }) => theme.colors.neutral.background};\n  border: 1px solid ${({ theme }) => theme.colors.neutral.border};\n  color: ${({ theme }) => theme.colors.neutral.text};\n  &::placeholder {\n    color: ${({ theme }) => theme.colors.neutral.textWeak};\n  }\n  border-radius: ${({ theme }) => theme.radii.default};\n  padding: ${({ theme }) =>\n    `${theme.space['1.5']} ${theme.space['1']} ${theme.space['1.5']} ${theme.space['2']}`};\n  padding-right: ${({ theme, isClearable, hasSentimentIcon }) =>\n    /* including 1 optional if both element is visible + 1 because content is absolute 1space unit from right */\n    `calc(${theme.space[isClearable && hasSentimentIcon ? '4' : '3']} + ${\n      isClearable ? `${ButtonSizeHeight.xsmall}px` : '0px'\n    } + ${hasSentimentIcon ? `${STATE_ICON_SIZE}px` : '0px'})`};\n\n  &[data-success='true'] {\n    border-color: ${({ theme }) => theme.colors.success.border};\n  }\n\n  &[data-error='true'] {\n    border-color: ${({ theme }) => theme.colors.danger.border};\n  }\n\n  &[data-readonly='true'] {\n    background: ${({ theme }) => theme.colors.neutral.backgroundWeak};\n    border-color: ${({ theme }) => theme.colors.neutral.border};\n  }\n\n  &:disabled {\n    background: ${({ theme }) => theme.colors.neutral.backgroundDisabled};\n    border-color: ${({ theme }) => theme.colors.neutral.borderDisabled};\n    color: ${({ theme }) => theme.colors.neutral.textDisabled};\n\n    &::placeholder {\n      color: ${({ theme }) => theme.colors.neutral.textWeakDisabled};\n    }\n  }\n\n  &:not(:disabled) {\n    &:hover {\n      border-color: ${({ theme }) => theme.colors.primary.border};\n    }\n\n    &:focus {\n      outline: none;\n      border-color: ${({ theme }) => theme.colors.primary.border};\n      box-shadow: ${({ theme }) => theme.shadows.focusPrimary};\n    }\n  }\n`\n\ntype LabelProps =\n  | {\n      label: string\n      'aria-label'?: never\n    }\n  | {\n      label?: never\n      'aria-label': string\n    }\n\ntype TextAreaProps = {\n  id?: string\n  className?: string\n  tabIndex?: number\n  autoFocus?: boolean\n  value?: string\n  onChange: (newValue: string) => void\n  placeholder?: string\n  /**\n   * Override others properties : readyOnly, success, error.\n   */\n  disabled?: boolean\n  /**\n   * Override others properties : success, error.\n   * Ignored if following props are provided : disabled.\n   */\n  readOnly?: boolean\n  /**\n   * Override others properties : error, helper.\n   * Ignored if following props are provided : disabled, readyOnly.\n   */\n  success?: string\n  /**\n   * Override others properties : helper.\n   * Ignored if following props are provided : disabled, readyOnly, success.\n   */\n  error?: string\n  /**\n   * Ignored if following props are provided : readyOnly, success.\n   */\n  helper?: ReactNode\n  /**\n   * Number of rows to display. If 'auto', the textarea will grow with the content and won't be resizable\n   */\n  rows?: number | 'auto'\n  /**\n   * Text area will grow with the content with a maximum number of rows.\n   */\n  maxRows?: number\n  minLength?: number\n  maxLength?: number\n  tooltip?: string\n  required?: boolean\n  'data-testid'?: string\n  name?: string\n  onFocus?: DOMAttributes<HTMLTextAreaElement>['onFocus']\n  onBlur?: DOMAttributes<HTMLTextAreaElement>['onBlur']\n  onKeyDown?: DOMAttributes<HTMLTextAreaElement>['onKeyDown']\n  clearable?: boolean\n  labelDescription?: ReactNode\n} & LabelProps\n\n/**\n * This component offers an extended textarea HTML\n */\nexport const TextArea = forwardRef<HTMLTextAreaElement, TextAreaProps>(\n  (\n    {\n      id,\n      className,\n      tabIndex,\n      value,\n      onChange,\n      placeholder,\n      rows = 3,\n      maxRows,\n      disabled = false,\n      readOnly = false,\n      success,\n      error,\n      helper,\n      minLength,\n      maxLength,\n      tooltip,\n      label,\n      autoFocus,\n      required = false,\n      'data-testid': dataTestId,\n      name,\n      onFocus,\n      onBlur,\n      onKeyDown,\n      clearable = false,\n      labelDescription,\n      'aria-label': ariaLabel,\n    },\n    ref,\n  ) => {\n    const localId = useId()\n    const theme = useTheme()\n    const textAreaRef = useRef<HTMLTextAreaElement>(null)\n    useImperativeHandle(ref, () => textAreaRef.current as HTMLTextAreaElement)\n\n    useEffect(() => {\n      const textArea = textAreaRef.current\n      const padding = theme.space['1.5']\n\n      if (textArea && rows === 'auto' && !maxRows) {\n        textArea.style.height = 'auto'\n        textArea.style.resize = 'none'\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n      } else if (textArea && maxRows) {\n        const lineHeight = Number.parseFloat(\n          getComputedStyle(textArea).lineHeight,\n        )\n\n        textArea.style.height = 'auto'\n        const maxHeight = maxRows * lineHeight\n\n        textArea.style.height = `${textArea.scrollHeight + 2}px`\n        textArea.style.maxHeight = `calc(${maxHeight}px + 2*${padding})`\n\n        if (typeof rows === 'number') {\n          const minHeight = rows * lineHeight\n          textArea.style.minHeight = `calc(${minHeight}px + 2*${padding})`\n        }\n      }\n    }, [value, rows, theme, maxRows])\n\n    const sentiment = useMemo(() => {\n      if (error) {\n        return 'danger'\n      }\n\n      if (success) {\n        return 'success'\n      }\n\n      return 'neutral'\n    }, [error, success])\n    const notice = success || error || helper\n\n    const computedClearable = clearable && !!value\n\n    return (\n      <Stack gap=\"0.5\" className={className}>\n        {label || labelDescription ? (\n          <Stack direction=\"row\" gap=\"1\" alignItems=\"center\">\n            {label ? (\n              <Stack direction=\"row\" gap=\"0.5\" alignItems=\"start\">\n                <Text\n                  as=\"label\"\n                  variant=\"bodyStrong\"\n                  sentiment=\"neutral\"\n                  htmlFor={id ?? localId}\n                  prominence=\"strong\"\n                >\n                  {label}\n                </Text>\n                {required ? <AsteriskIcon sentiment=\"danger\" size={8} /> : null}\n              </Stack>\n            ) : null}\n            {labelDescription ?? null}\n          </Stack>\n        ) : null}\n        <Tooltip text={tooltip}>\n          <StyledTextAreaWrapper>\n            <StyledTextArea\n              aria-invalid={!!error}\n              id={id ?? localId}\n              tabIndex={tabIndex}\n              autoFocus={autoFocus}\n              disabled={disabled}\n              rows={rows !== 'auto' ? rows : 1}\n              ref={textAreaRef}\n              value={value}\n              onChange={event => {\n                onChange(event.currentTarget.value)\n              }}\n              hasSentimentIcon={!!success || !!error}\n              data-readonly={readOnly}\n              data-success={!!success}\n              data-error={!!error}\n              isClearable={!!computedClearable}\n              minLength={minLength}\n              maxLength={maxLength}\n              placeholder={placeholder}\n              data-testid={dataTestId}\n              name={name}\n              onFocus={onFocus}\n              onBlur={onBlur}\n              onKeyDown={onKeyDown}\n              aria-label={ariaLabel}\n            />\n            <StyledTextAreaAbsoluteStack\n              direction=\"row\"\n              alignItems=\"center\"\n              gap=\"1\"\n            >\n              {computedClearable ? (\n                <Button\n                  aria-label=\"clear value\"\n                  variant=\"ghost\"\n                  size=\"xsmall\"\n                  icon=\"close\"\n                  onClick={() => {\n                    onChange('')\n                  }}\n                  sentiment=\"neutral\"\n                />\n              ) : null}\n              {success ? (\n                <CheckCircleIcon sentiment=\"success\" size={STATE_ICON_SIZE} />\n              ) : null}\n              {error ? <AlertCircleIcon sentiment=\"danger\" /> : null}\n            </StyledTextAreaAbsoluteStack>\n          </StyledTextAreaWrapper>\n        </Tooltip>\n\n        {notice || maxLength ? (\n          <Row templateColumns=\"minmax(0, 1fr) min-content\" gap=\"1\">\n            <div>\n              {error || success || typeof helper === 'string' ? (\n                <Text\n                  as=\"p\"\n                  variant=\"caption\"\n                  sentiment={sentiment}\n                  prominence={!error && !success ? 'weak' : 'default'}\n                  disabled={disabled}\n                >\n                  {error || success || helper}\n                </Text>\n              ) : null}\n              {!error && !success && typeof helper !== 'string' && helper\n                ? helper\n                : null}\n            </div>\n            {maxLength ? (\n              <Text\n                as=\"div\"\n                sentiment=\"neutral\"\n                prominence=\"weak\"\n                variant=\"caption\"\n              >\n                {value?.length ?? 0}/{maxLength}\n              </Text>\n            ) : null}\n          </Row>\n        ) : null}\n      </Stack>\n    )\n  },\n)\n"]} */"));
91
92
  const TextArea = React.forwardRef(({
92
93
  id,
93
94
  className,
@@ -154,17 +155,17 @@ const TextArea = React.forwardRef(({
154
155
  return /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { gap: "0.5", className, children: [
155
156
  label || labelDescription ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "1", alignItems: "center", children: [
156
157
  label ? /* @__PURE__ */ jsxRuntime.jsxs(index.Stack, { direction: "row", gap: "0.5", alignItems: "start", children: [
157
- /* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { as: "label", variant: "bodyStrong", sentiment: "neutral", htmlFor: id ?? localId, prominence: "strong", children: label }),
158
+ /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "label", variant: "bodyStrong", sentiment: "neutral", htmlFor: id ?? localId, prominence: "strong", children: label }),
158
159
  required ? /* @__PURE__ */ jsxRuntime.jsx(Icon.AsteriskIcon, { sentiment: "danger", size: 8 }) : null
159
160
  ] }) : null,
160
161
  labelDescription ?? null
161
162
  ] }) : null,
162
- /* @__PURE__ */ jsxRuntime.jsx(index$3.Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsxs(StyledTextAreaWrapper, { children: [
163
+ /* @__PURE__ */ jsxRuntime.jsx(index$2.Tooltip, { text: tooltip, children: /* @__PURE__ */ jsxRuntime.jsxs(StyledTextAreaWrapper, { children: [
163
164
  /* @__PURE__ */ jsxRuntime.jsx(StyledTextArea, { "aria-invalid": !!error, id: id ?? localId, tabIndex, autoFocus, disabled, rows: rows !== "auto" ? rows : 1, ref: textAreaRef, value, onChange: (event) => {
164
165
  onChange(event.currentTarget.value);
165
166
  }, hasSentimentIcon: !!success || !!error, "data-readonly": readOnly, "data-success": !!success, "data-error": !!error, isClearable: !!computedClearable, minLength, maxLength, placeholder, "data-testid": dataTestId, name, onFocus, onBlur, onKeyDown, "aria-label": ariaLabel }),
166
167
  /* @__PURE__ */ jsxRuntime.jsxs(StyledTextAreaAbsoluteStack, { direction: "row", alignItems: "center", gap: "1", children: [
167
- computedClearable ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Button, { "aria-label": "clear value", variant: "ghost", size: "xsmall", icon: "close", onClick: () => {
168
+ computedClearable ? /* @__PURE__ */ jsxRuntime.jsx(index$3.Button, { "aria-label": "clear value", variant: "ghost", size: "xsmall", icon: "close", onClick: () => {
168
169
  onChange("");
169
170
  }, sentiment: "neutral" }) : null,
170
171
  success ? /* @__PURE__ */ jsxRuntime.jsx(Icon.CheckCircleIcon, { sentiment: "success", size: STATE_ICON_SIZE }) : null,
@@ -173,10 +174,10 @@ const TextArea = React.forwardRef(({
173
174
  ] }) }),
174
175
  notice || maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$4.Row, { templateColumns: "minmax(0, 1fr) min-content", gap: "1", children: [
175
176
  /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
176
- error || success || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$2.Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
177
+ error || success || typeof helper === "string" ? /* @__PURE__ */ jsxRuntime.jsx(index$1.Text, { as: "p", variant: "caption", sentiment, prominence: !error && !success ? "weak" : "default", disabled, children: error || success || helper }) : null,
177
178
  !error && !success && typeof helper !== "string" && helper ? helper : null
178
179
  ] }),
179
- maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$2.Text, { as: "div", sentiment: "neutral", prominence: "weak", variant: "caption", children: [
180
+ maxLength ? /* @__PURE__ */ jsxRuntime.jsxs(index$1.Text, { as: "div", sentiment: "neutral", prominence: "weak", variant: "caption", children: [
180
181
  value?.length ?? 0,
181
182
  "/",
182
183
  maxLength