td-stylekit 26.9.1 → 26.10.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -1,3 +1,24 @@
1
+ ## [26.10.2](https://github.com/treasure-data/td-stylekit/compare/v26.10.1...v26.10.2) (2022-05-24)
2
+
3
+
4
+ ### Bug Fixes
5
+
6
+ * Update tippyjs to handle memory leak issue in td-crystal ([#1278](https://github.com/treasure-data/td-stylekit/issues/1278)) ([ee9aa30](https://github.com/treasure-data/td-stylekit/commit/ee9aa30))
7
+
8
+ ## [26.10.1](https://github.com/treasure-data/td-stylekit/compare/v26.10.0...v26.10.1) (2022-05-23)
9
+
10
+
11
+ ### Bug Fixes
12
+
13
+ * **CON-10029:** Render ColorPicker in tooltip popover ([#1277](https://github.com/treasure-data/td-stylekit/issues/1277)) ([0c935a4](https://github.com/treasure-data/td-stylekit/commit/0c935a4))
14
+
15
+ # [26.10.0](https://github.com/treasure-data/td-stylekit/compare/v26.9.1...v26.10.0) (2022-05-17)
16
+
17
+
18
+ ### Features
19
+
20
+ * **CON-10008:** Add hideColumn prop to DataGrid.Column ([#1276](https://github.com/treasure-data/td-stylekit/issues/1276)) ([444d9d5](https://github.com/treasure-data/td-stylekit/commit/444d9d5))
21
+
1
22
  ## [26.9.1](https://github.com/treasure-data/td-stylekit/compare/v26.9.0...v26.9.1) (2022-05-13)
2
23
 
3
24
 
@@ -23,6 +23,8 @@ var _Button = _interopRequireDefault(require("../Button"));
23
23
 
24
24
  var _Grid = _interopRequireWildcard(require("../Grid"));
25
25
 
26
+ var _TooltipPopover = _interopRequireDefault(require("../TooltipPopover"));
27
+
26
28
  var _utils = require("./utils");
27
29
 
28
30
  var _hooks = require("./hooks");
@@ -72,9 +74,6 @@ var PickerWrapper = ( /*#__PURE__*/0, _base["default"])('div', process.env.NODE_
72
74
  borderStyle: 'solid',
73
75
  borderWidth: theme.borderWidth[1],
74
76
  paddingBottom: theme.space[5],
75
- position: 'absolute',
76
- right: -(PICKER_WIDTH + 9),
77
- top: 0,
78
77
  width: PICKER_WIDTH,
79
78
  '& .react-colorful': {
80
79
  height: 'auto',
@@ -103,7 +102,7 @@ var PickerWrapper = ( /*#__PURE__*/0, _base["default"])('div', process.env.NODE_
103
102
  borderRadius: '50%' // Make them round
104
103
 
105
104
  }), _ref2;
106
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAgBsB","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */");
105
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAiBsB","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */");
107
106
  var Label = ( /*#__PURE__*/0, _base["default"])(_FormLabel["default"], process.env.NODE_ENV === "production" ? {
108
107
  target: "eu4c7ix4"
109
108
  } : {
@@ -118,7 +117,7 @@ var Label = ( /*#__PURE__*/0, _base["default"])(_FormLabel["default"], process.e
118
117
  height: theme.space[6],
119
118
  lineHeight: theme.space[6]
120
119
  };
121
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AA4Dc","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */");
120
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AA0Dc","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */");
122
121
  var Input = ( /*#__PURE__*/0, _base["default"])(_Input["default"], process.env.NODE_ENV === "production" ? {
123
122
  target: "eu4c7ix3"
124
123
  } : {
@@ -133,7 +132,7 @@ var Input = ( /*#__PURE__*/0, _base["default"])(_Input["default"], process.env.N
133
132
  marginRight: theme.space[4],
134
133
  textTransform: 'uppercase'
135
134
  };
136
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAoEc","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */");
135
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAkEc","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */");
137
136
  var Palette = ( /*#__PURE__*/0, _base["default"])('div', process.env.NODE_ENV === "production" ? {
138
137
  target: "eu4c7ix2"
139
138
  } : {
@@ -149,7 +148,7 @@ var Palette = ( /*#__PURE__*/0, _base["default"])('div', process.env.NODE_ENV ==
149
148
  marginRight: theme.space[3],
150
149
  backgroundColor: color
151
150
  };
152
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AA4EgB","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */");
151
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AA0EgB","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */");
153
152
  var Button = ( /*#__PURE__*/0, _base["default"])(_Button["default"], process.env.NODE_ENV === "production" ? {
154
153
  target: "eu4c7ix1"
155
154
  } : {
@@ -164,7 +163,7 @@ var Button = ( /*#__PURE__*/0, _base["default"])(_Button["default"], process.env
164
163
  paddingRight: theme.space[4],
165
164
  borderRadius: theme.radius[0]
166
165
  };
167
- }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAoFe","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */");
166
+ }, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAkFe","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */");
168
167
  var Item = ( /*#__PURE__*/0, _base["default"])(_Grid.GridItem, process.env.NODE_ENV === "production" ? {
169
168
  target: "eu4c7ix0"
170
169
  } : {
@@ -176,7 +175,7 @@ var Item = ( /*#__PURE__*/0, _base["default"])(_Grid.GridItem, process.env.NODE_
176
175
  } : {
177
176
  name: "zjik7",
178
177
  styles: "display:flex",
179
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AA4Fa","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */",
178
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AA0Fa","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */",
180
179
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
181
180
  });
182
181
 
@@ -186,7 +185,7 @@ var _ref9 = process.env.NODE_ENV === "production" ? {
186
185
  } : {
187
186
  name: "16f6a97-ColorPicker",
188
187
  styles: "position:relative;display:inline-flex;label:ColorPicker;",
189
- map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAsKM","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useClickOutsideToggle, useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  position: 'absolute',\n  right: -(PICKER_WIDTH + 9),\n  top: 0,\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Handle open/closed state of color picker popup\n  const [state, wrapperRef, handleToggle] = useClickOutsideToggle()\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={handleToggle}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      {state === 'open' ? (\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      ) : null}\n    </div>\n  )\n}\n"]} */",
188
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/ColorPicker/ColorPicker.tsx"],"names":[],"mappings":"AAmKM","file":"../../../src/ColorPicker/ColorPicker.tsx","sourcesContent":["import { useCallback, useMemo, useRef, useState } from 'react'\nimport { RgbaColorPicker } from 'react-colorful'\nimport styled from '@emotion/styled'\nimport debounce from 'lodash.debounce'\nimport SkInput from '../Input'\nimport FormLabel from '../FormLabel'\nimport SkButton from '../Button'\nimport Grid, { GridItem } from '../Grid'\nimport TooltipPopover from '../TooltipPopover'\nimport { rgbaToHexa, hexaToRgba, opacityFromHexa } from './utils'\nimport { useOpacityValueSync } from './hooks'\nimport type { RGBA } from './types'\n\n// Values from UX\nconst PICKER_WIDTH = 196\nconst SATURATION_HEIGHT = 129\n\nconst PickerWrapper = styled('div')(({ theme }) => ({\n  background: theme.palette.neutral[0],\n  borderColor: theme.palette.neutral[4],\n  borderRadius: theme.radius[2],\n  borderStyle: 'solid',\n  borderWidth: theme.borderWidth[1],\n  paddingBottom: theme.space[5],\n  width: PICKER_WIDTH,\n  '& .react-colorful': {\n    height: 'auto',\n    width: '100%'\n  },\n  '& .react-colorful__saturation': {\n    border: 'none',\n    borderTopLeftRadius: theme.radius[2],\n    borderTopRightRadius: theme.radius[2],\n    height: SATURATION_HEIGHT\n  },\n  [`& .react-colorful__hue,\n    & .react-colorful__alpha`]: {\n    borderRadius: theme.space[3], // Same as height\n    height: theme.space[3],\n    marginLeft: theme.space[4],\n    marginRight: theme.space[4]\n  },\n  '& .react-colorful__hue': {\n    marginBottom: theme.space[3],\n    marginTop: theme.space[4]\n  },\n  '& .react-colorful__alpha': {\n    marginBottom: theme.space[4]\n  },\n  [`& .react-colorful__saturation-pointer,\n    & .react-colorful__hue-pointer,\n    & .react-colorful__alpha-pointer`]: {\n    width: theme.space[3],\n    height: theme.space[3],\n    borderRadius: '50%' // Make them round\n  }\n}))\n\nconst Label = styled(FormLabel)(({ theme }) => ({\n  fontSize: theme.fontSize[1],\n  marginLeft: theme.space[4],\n  flexShrink: 0,\n  height: theme.space[6],\n  lineHeight: theme.space[6]\n}))\n\nconst Input = styled(SkInput)(({ theme }) => ({\n  flexShrink: 1,\n  fontSize: theme.fontSize[1],\n  height: theme.space[6],\n  marginRight: theme.space[4],\n  textTransform: 'uppercase'\n}))\n\nconst Palette = styled('div')<{ color: string }>(({ theme, color }) => ({\n  height: theme.space[6],\n  width: theme.space[6],\n  borderRadius: theme.radius[0],\n  marginRight: theme.space[3],\n  backgroundColor: color\n}))\n\nconst Button = styled(SkButton)(({ theme }) => ({\n  paddingTop: theme.space[2],\n  paddingBottom: theme.space[2],\n  paddingLeft: theme.space[4],\n  paddingRight: theme.space[4],\n  borderRadius: theme.radius[0]\n}))\n\nconst Item = styled(GridItem)({\n  display: 'flex'\n})\n\ntype ColorPickerProps = {\n  ['data-instrumentation']: string\n  onChange: (hex: string) => void\n  value: string\n  label: string\n}\n\nexport default function ColorPicker(props: ColorPickerProps) {\n  const { onChange, 'data-instrumentation': dataInstru, value, label } = props\n  const [isOpen, setIsOpen] = useState(false)\n  const wrapperRef = useRef(null)\n  const [opacity, setOpacity] = useState<string>(opacityFromHexa(value))\n\n  // Need to slightly delay the call to setState because moving the hue\n  // slider too fast can cause the picker internal state and our state\n  // to become unaligned and cause a render loop (for some reason).\n  // TODO: We may be able to replace this with useDeferredValue when we\n  // upgrade to React 18\n  const debouncedOnChange = useMemo(() => debounce(onChange, 1), [onChange])\n\n  const handleRgbaChange = useCallback(\n    (rgba: RGBA) => {\n      const hex = rgbaToHexa(rgba)\n      debouncedOnChange(hex)\n    },\n    [debouncedOnChange]\n  )\n\n  const handleHexaChange = useCallback(\n    ({ value: inString }) => {\n      let val = inString\n      if (typeof val === 'string') {\n        if (!val.startsWith('#') && val.length < 9) {\n          val = '#' + val\n        }\n        if (val !== value) {\n          onChange(val)\n        }\n      }\n    },\n    [onChange, value]\n  )\n\n  const handleOpacityChange = useCallback(({ value: inString }) => {\n    const val = inString\n    if (typeof val === 'string') {\n      setOpacity(val)\n    }\n  }, [])\n\n  // Contains logic to call onChange when opacity changes, update opacity\n  // when the value from props changes, and to reformat the opacity\n  // when the input is blurred. This is tricky because of the circular\n  // nature of opacity updates and the input color updates, especially\n  // when typing into the opacity field. You don't want to reformat the\n  // field while typing because it moves the cursor to the end.\n  const { handleOpacityBlur, handleOpacityFocus } = useOpacityValueSync(\n    value,\n    opacity,\n    onChange,\n    setOpacity\n  )\n\n  // Expands the input value if it is a condensed hex\n  const paletteColor = rgbaToHexa(hexaToRgba(value))\n\n  return (\n    <div\n      ref={wrapperRef}\n      css={{\n        position: 'relative',\n        display: 'inline-flex'\n      }}\n    >\n      <Button\n        data-instrumentation={`${dataInstru} colorpicker-button`}\n        onClick={() => setIsOpen(!isOpen)}\n        plain\n      >\n        <Palette title={paletteColor} color={paletteColor} />\n        {label}\n      </Button>\n      <TooltipPopover\n        show={isOpen}\n        target={wrapperRef}\n        placement=\"right-start\"\n        interactive\n        escClose\n        hideOnClick\n        onHide={() => setIsOpen(false)}\n      >\n        <PickerWrapper>\n          <RgbaColorPicker\n            color={hexaToRgba(value)}\n            data-instrumentation={`${dataInstru} colorpicker-colorpicker`}\n            onChange={handleRgbaChange}\n          />\n          <Grid columns=\"auto auto\" gap=\".5rem .25rem\">\n            <Item>\n              <Label htmlFor=\"hex-color\">Hex</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-hexinput`}\n                maxLength={9} // #rrggbbaa\n                name=\"hex-color\"\n                onChange={handleHexaChange}\n                type=\"text\"\n                value={value}\n              />\n            </Item>\n\n            <Item>\n              <Label htmlFor=\"opacity\">Opacity</Label>\n            </Item>\n            <Item>\n              <Input\n                data-instrumentation={`${dataInstru} colorpicker-opacityinput`}\n                maxLength={5} // 99.9%\n                name=\"opacity\"\n                onChange={handleOpacityChange}\n                type=\"text\"\n                value={opacity}\n                onFocus={handleOpacityFocus}\n                onBlur={handleOpacityBlur}\n              />\n            </Item>\n          </Grid>\n        </PickerWrapper>\n      </TooltipPopover>\n    </div>\n  )\n}\n"]} */",
190
189
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
191
190
  };
192
191
 
@@ -196,10 +195,17 @@ function ColorPicker(props) {
196
195
  value = props.value,
197
196
  label = props.label;
198
197
 
199
- var _useState = (0, _react.useState)((0, _utils.opacityFromHexa)(value)),
198
+ var _useState = (0, _react.useState)(false),
200
199
  _useState2 = _slicedToArray(_useState, 2),
201
- opacity = _useState2[0],
202
- setOpacity = _useState2[1]; // Need to slightly delay the call to setState because moving the hue
200
+ isOpen = _useState2[0],
201
+ setIsOpen = _useState2[1];
202
+
203
+ var wrapperRef = (0, _react.useRef)(null);
204
+
205
+ var _useState3 = (0, _react.useState)((0, _utils.opacityFromHexa)(value)),
206
+ _useState4 = _slicedToArray(_useState3, 2),
207
+ opacity = _useState4[0],
208
+ setOpacity = _useState4[1]; // Need to slightly delay the call to setState because moving the hue
203
209
  // slider too fast can cause the picker internal state and our state
204
210
  // to become unaligned and cause a render loop (for some reason).
205
211
  // TODO: We may be able to replace this with useDeferredValue when we
@@ -243,14 +249,7 @@ function ColorPicker(props) {
243
249
 
244
250
  var _useOpacityValueSync = (0, _hooks.useOpacityValueSync)(value, opacity, onChange, setOpacity),
245
251
  handleOpacityBlur = _useOpacityValueSync.handleOpacityBlur,
246
- handleOpacityFocus = _useOpacityValueSync.handleOpacityFocus; // Handle open/closed state of color picker popup
247
-
248
-
249
- var _useClickOutsideToggl = (0, _hooks.useClickOutsideToggle)(),
250
- _useClickOutsideToggl2 = _slicedToArray(_useClickOutsideToggl, 3),
251
- state = _useClickOutsideToggl2[0],
252
- wrapperRef = _useClickOutsideToggl2[1],
253
- handleToggle = _useClickOutsideToggl2[2]; // Expands the input value if it is a condensed hex
252
+ handleOpacityFocus = _useOpacityValueSync.handleOpacityFocus; // Expands the input value if it is a condensed hex
254
253
 
255
254
 
256
255
  var paletteColor = (0, _utils.rgbaToHexa)((0, _utils.hexaToRgba)(value));
@@ -261,7 +260,9 @@ function ColorPicker(props) {
261
260
  children: [(0, _jsxRuntime.jsxs)(Button, {
262
261
  "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "button"),
263
262
  "data-instrumentation": "".concat(dataInstru, " colorpicker-button"),
264
- onClick: handleToggle,
263
+ onClick: function onClick() {
264
+ return setIsOpen(!isOpen);
265
+ },
265
266
  plain: true,
266
267
  children: [(0, _jsxRuntime.jsx)(Palette, {
267
268
  "data-gs-c": gsC(paletteColor),
@@ -269,63 +270,75 @@ function ColorPicker(props) {
269
270
  title: paletteColor,
270
271
  color: paletteColor
271
272
  }), label]
272
- }), state === 'open' ? (0, _jsxRuntime.jsxs)(PickerWrapper, {
273
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper"),
274
- children: [(0, _jsxRuntime.jsx)(_reactColorful.RgbaColorPicker, {
275
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "rgba-color-picker"),
276
- color: (0, _utils.hexaToRgba)(value),
277
- "data-instrumentation": "".concat(dataInstru, " colorpicker-colorpicker"),
278
- onChange: handleRgbaChange
279
- }), (0, _jsxRuntime.jsxs)(_Grid["default"], {
280
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid"),
281
- columns: "auto auto",
282
- gap: ".5rem .25rem",
283
- children: [(0, _jsxRuntime.jsx)(Item, {
284
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item"),
285
- children: (0, _jsxRuntime.jsx)(Label, {
286
- "data-gs-c": gsC("Hex"),
287
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item", "label"),
288
- htmlFor: "hex-color",
289
- children: "Hex"
290
- })
291
- }), (0, _jsxRuntime.jsx)(Item, {
292
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item"),
293
- children: (0, _jsxRuntime.jsx)(Input, {
294
- "data-gs-c": gsC("hex-color", value),
295
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item", "input"),
296
- "data-instrumentation": "".concat(dataInstru, " colorpicker-hexinput"),
297
- maxLength: 9 // #rrggbbaa
298
- ,
299
- name: "hex-color",
300
- onChange: handleHexaChange,
301
- type: "text",
302
- value: value
303
- })
304
- }), (0, _jsxRuntime.jsx)(Item, {
305
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item"),
306
- children: (0, _jsxRuntime.jsx)(Label, {
307
- "data-gs-c": gsC("Opacity"),
308
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item", "label"),
309
- htmlFor: "opacity",
310
- children: "Opacity"
311
- })
312
- }), (0, _jsxRuntime.jsx)(Item, {
313
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item"),
314
- children: (0, _jsxRuntime.jsx)(Input, {
315
- "data-gs-c": gsC("opacity", opacity),
316
- "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "picker-wrapper", "grid", "item", "input"),
317
- "data-instrumentation": "".concat(dataInstru, " colorpicker-opacityinput"),
318
- maxLength: 5 // 99.9%
319
- ,
320
- name: "opacity",
321
- onChange: handleOpacityChange,
322
- type: "text",
323
- value: opacity,
324
- onFocus: handleOpacityFocus,
325
- onBlur: handleOpacityBlur
326
- })
273
+ }), (0, _jsxRuntime.jsx)(_TooltipPopover["default"], {
274
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover"),
275
+ show: isOpen,
276
+ target: wrapperRef,
277
+ placement: "right-start",
278
+ interactive: true,
279
+ escClose: true,
280
+ hideOnClick: true,
281
+ onHide: function onHide() {
282
+ return setIsOpen(false);
283
+ },
284
+ children: (0, _jsxRuntime.jsxs)(PickerWrapper, {
285
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper"),
286
+ children: [(0, _jsxRuntime.jsx)(_reactColorful.RgbaColorPicker, {
287
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "rgba-color-picker"),
288
+ color: (0, _utils.hexaToRgba)(value),
289
+ "data-instrumentation": "".concat(dataInstru, " colorpicker-colorpicker"),
290
+ onChange: handleRgbaChange
291
+ }), (0, _jsxRuntime.jsxs)(_Grid["default"], {
292
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid"),
293
+ columns: "auto auto",
294
+ gap: ".5rem .25rem",
295
+ children: [(0, _jsxRuntime.jsx)(Item, {
296
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item"),
297
+ children: (0, _jsxRuntime.jsx)(Label, {
298
+ "data-gs-c": gsC("Hex"),
299
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item", "label"),
300
+ htmlFor: "hex-color",
301
+ children: "Hex"
302
+ })
303
+ }), (0, _jsxRuntime.jsx)(Item, {
304
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item"),
305
+ children: (0, _jsxRuntime.jsx)(Input, {
306
+ "data-gs-c": gsC("hex-color", value),
307
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item", "input"),
308
+ "data-instrumentation": "".concat(dataInstru, " colorpicker-hexinput"),
309
+ maxLength: 9 // #rrggbbaa
310
+ ,
311
+ name: "hex-color",
312
+ onChange: handleHexaChange,
313
+ type: "text",
314
+ value: value
315
+ })
316
+ }), (0, _jsxRuntime.jsx)(Item, {
317
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item"),
318
+ children: (0, _jsxRuntime.jsx)(Label, {
319
+ "data-gs-c": gsC("Opacity"),
320
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item", "label"),
321
+ htmlFor: "opacity",
322
+ children: "Opacity"
323
+ })
324
+ }), (0, _jsxRuntime.jsx)(Item, {
325
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item"),
326
+ children: (0, _jsxRuntime.jsx)(Input, {
327
+ "data-gs-c": gsC("opacity", opacity),
328
+ "data-gs": gs("src", "colorpicker", "colorpicker.tsx", "color-picker", "div", "tooltip-popover", "picker-wrapper", "grid", "item", "input"),
329
+ "data-instrumentation": "".concat(dataInstru, " colorpicker-opacityinput"),
330
+ maxLength: 5 // 99.9%
331
+ ,
332
+ name: "opacity",
333
+ onChange: handleOpacityChange,
334
+ type: "text",
335
+ value: opacity,
336
+ onFocus: handleOpacityFocus,
337
+ onBlur: handleOpacityBlur
338
+ })
339
+ })]
327
340
  })]
328
- })]
329
- }) : null]
341
+ })
342
+ })]
330
343
  });
331
344
  }
@@ -1,13 +1,5 @@
1
- import { RefObject } from 'react';
2
- declare type State = 'open' | 'closed';
3
- export declare function useClickOutsideToggle(): [
4
- State,
5
- RefObject<HTMLDivElement>,
6
- () => void
7
- ];
8
1
  export declare function useOpacityValueSync(value: string, opacity: string, onChange: (value: string) => void, setOpacity: (opacity: string) => void): {
9
2
  handleOpacityFocus: () => void;
10
3
  handleOpacityBlur: () => void;
11
4
  };
12
- export {};
13
5
  //# sourceMappingURL=hooks.d.ts.map
@@ -3,52 +3,15 @@
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- exports.useClickOutsideToggle = useClickOutsideToggle;
7
6
  exports.useOpacityValueSync = useOpacityValueSync;
8
7
 
9
8
  var _react = require("react");
10
9
 
11
- var _useOnclickoutside = _interopRequireDefault(require("use-onclickoutside"));
12
-
13
10
  var _utils = require("./utils");
14
11
 
15
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
16
-
17
- function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); }
18
-
19
- function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
20
-
21
- function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
22
-
23
- function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }
24
-
25
- function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return; var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
26
-
27
- function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
28
-
29
12
  if (typeof window !== "undefined" && !window.gs) window.gs = function () {};
30
13
  if (typeof window !== "undefined" && !window.gsC) window.gsC = function () {};
31
14
 
32
- // Returns an open/closed state, a ref to attach a clickoutide event to,
33
- // and a function that toggles the state. Pass the function to a button,
34
- // and the ref to a div above the button and the component you want to toggle
35
- function useClickOutsideToggle() {
36
- var wrapperRef = (0, _react.useRef)(null);
37
-
38
- var _useState = (0, _react.useState)('closed'),
39
- _useState2 = _slicedToArray(_useState, 2),
40
- state = _useState2[0],
41
- setState = _useState2[1];
42
-
43
- var handleToggle = (0, _react.useCallback)(function () {
44
- setState(state === 'open' ? 'closed' : 'open');
45
- }, [state]);
46
- (0, _useOnclickoutside["default"])(wrapperRef, function () {
47
- return setState('closed');
48
- });
49
- return [state, wrapperRef, handleToggle];
50
- }
51
-
52
15
  function useOpacityValueSync(value, opacity, onChange, setOpacity) {
53
16
  var oldValue = (0, _react.useRef)(value);
54
17
  var oldOpacity = (0, _react.useRef)(opacity);
@@ -99,7 +99,7 @@ declare type HandleFilterArguments = {
99
99
  *
100
100
  */
101
101
  declare class DataGrid extends Component<DataGridProps, DataGridState> {
102
- static Column: import("lodash").CurriedFunction2<import("./components/BasicColumn").BasicColumnProps, BasicColumnMappedProps, import("@emotion/react/jsx-runtime").JSX.Element>;
102
+ static Column: import("lodash").CurriedFunction2<import("./components/BasicColumn").BasicColumnProps, BasicColumnMappedProps, import("@emotion/react/jsx-runtime").JSX.Element | null>;
103
103
  static SortDirection: import("react-virtualized/dist/es/Table").SortDirectionStatic;
104
104
  static Elements: typeof Elements;
105
105
  static InputCell: any;
@@ -550,7 +550,9 @@ var DataGrid = /*#__PURE__*/function (_Component) {
550
550
  var safariStyle = {
551
551
  transform: "translateZ(0)"
552
552
  };
553
- var mappedChildren = this.mapChildren(); // TODO: Better type signature when accessing children props?
553
+ var mappedChildren = this.mapChildren().filter(function (child) {
554
+ return child !== null;
555
+ }); // TODO: Better type signature when accessing children props?
554
556
 
555
557
  var spanData = mappedChildren.map(function (child) {
556
558
  return {
@@ -610,7 +612,7 @@ var DataGrid = /*#__PURE__*/function (_Component) {
610
612
  css: /*#__PURE__*/(0, _react.css)({
611
613
  width: width,
612
614
  height: height
613
- }, process.env.NODE_ENV === "production" ? "" : ";label:render;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DataGrid/DataGrid.tsx"],"names":[],"mappings":"AAylBQ","file":"../../../src/DataGrid/DataGrid.tsx","sourcesContent":["import React, { Component } from 'react'\nimport ReactDOM from 'react-dom'\nimport {\n  InfiniteLoader,\n  Table,\n  SortDirection,\n  TableRowProps\n} from 'react-virtualized'\nimport type {\n  SortDirectionType,\n  TableProps,\n  IndexRange,\n  Index,\n  RowMouseEventHandlerParams\n} from 'react-virtualized'\nimport * as Elements from './elements'\nimport * as Renderers from './renderers'\nimport {\n  BasicColumn,\n  FocusableRow,\n  InputCell,\n  CollapsibleCell,\n  ScrollableTable\n} from './components'\nimport { CloseWrapper } from '../utils'\nimport curry from 'lodash.curry'\nimport includes from 'lodash.includes'\nimport noop from 'lodash.noop'\nimport range from 'lodash.range'\nimport throttle from 'lodash.throttle'\nimport uniq from 'lodash.uniq'\nimport * as dataGridUtils from './utils'\nimport type { BasicColumnMappedProps } from './components/BasicColumn'\nimport type { LocaleStrings } from './constants'\n\nexport type onFilterArguments = {\n  filters: { [dataKey: string]: any }\n  sortBy?: string\n  sortDirection?: SortDirectionType\n}\n\ntype activeContextMenuState = {\n  top: number\n  left: number\n  index: number\n  rowData: any\n}\n\ntype contextMenuRendererArguments = {\n  top?: number\n  left?: number\n  index?: number\n  rowData?: any\n  selectedRows: Array<number>\n  onClose: () => void\n}\n\nexport type SelectableType = 'single' | 'multi'\n\ntype ColumnFunction = (args: BasicColumnMappedProps) => React.ReactNode\n\nexport type DataGridProps = {\n  /** Functions that create react virtualized Columns. Default column is exposed as DataGrid.Column. */\n  // Can take a single column or an array of columns\n  children: ColumnFunction | Array<ColumnFunction>\n  /** Instrumentation */\n  'data-instrumentation'?: string\n  /** Called during the onContextMenu event of each row. */\n  contextMenuRenderer?: (\n    contextMenuRendererArguments: contextMenuRendererArguments\n  ) => React.ReactNode\n  /** Called when a column is filtered or sorted. */\n  onFilter: (onFilterArguments: onFilterArguments) => void\n  /** Called when the row selection changes. Receives array of indices. */\n  onSelect: (indexes: number[]) => void\n  /** Called when the row is double clicked. Receives index. */\n  onRowDoubleClick?: (info: RowMouseEventHandlerParams) => void\n  /** Infinite loading: Indicates more rows need to be loaded. */\n  hasMoreRows?: boolean\n  /** Infinite loading: Called when more rows need to be loaded. */\n  loadMoreRows?: (props: IndexRange) => Promise<any>\n  /** Count of rows that have already been loaded. */\n  rowCount: number\n  /** Required by react-virtualized table. */\n  rowGetter: ({ index: number }) => any\n  /** Whether rows can be selected (and how many). */\n  selectable?: SelectableType | false\n  /** Array of indices that should be selected. Only used if `selectable` is set. */\n  selectedRows: Array<number>\n  /** The column you want to sort */\n  sortBy?: string\n  /** The direction sorting should be done */\n  sortDirection?: SortDirectionType\n  /** A map of columns to filter data */\n  filters?: { [dataKey: string]: any }\n  /** Set the width of scrollable content */\n  scrollWidth?: number\n  width: number\n  height: number\n  /** Locale string: ja, de, es, ko, en, etc. */\n  locale?: string\n  /** Locale strings to override en for language in locale */\n  localeStrings?: LocaleStrings\n  /** allow events to propagate to a cell on row select */\n  allowRowEventPassthrough?: boolean\n} & Partial<TableProps>\n\ntype DataGridState = {\n  activeContextMenu: activeContextMenuState | null\n  disableKeyboardShortcuts: boolean\n  focusedRow: number | null\n  selecting: boolean\n  lastSelectedRow: number | null\n}\n\ntype HandleFilterArguments = {\n  sortBy?: string\n  sortDirection?: SortDirectionType\n  filters?: any\n}\n\n/**\n * Component for displaying data in a grid with support for things like\n * filtering, sorting, and context menus.\n *\n * Depends on react-virtualized/styles.css.\n *\n * For react-virtualized Table docs see: https://github.com/bvaughn/react-virtualized/blob/master/docs/Table.md\n *\n */\nclass DataGrid extends Component<DataGridProps, DataGridState> {\n  // expose sub-components\n  static Column = curry(BasicColumn)\n  static SortDirection = SortDirection\n  static Elements = Elements\n  static InputCell = InputCell\n  static CollapsibleCell = CollapsibleCell\n  static Renderers = Renderers\n\n  static SINGLE_SELECT: SelectableType = 'single'\n  static MULTI_SELECT: SelectableType = 'multi'\n\n  static utils = dataGridUtils\n\n  static defaultProps = {\n    loadMoreRows: noop,\n    onSelect: noop,\n    onFilter: noop,\n    'data-instrumentation': 'datagrid',\n    sortBy: '',\n    sortDirection: SortDirection.ASC,\n    filters: {},\n    selectedRows: [],\n    allowRowEventPassthrough: false\n  }\n\n  static displayName = 'DataGrid'\n\n  state: DataGridState = {\n    activeContextMenu: null,\n    disableKeyboardShortcuts: false,\n    focusedRow: null,\n    lastSelectedRow: null,\n    selecting: false\n  }\n\n  infiniteTable = React.createRef<InfiniteLoader>()\n\n  columnTransformer: (\n    columns: React.ReactElement<any>[],\n    rowData: any,\n    rowIndex: number\n  ) => React.ReactNode[] | null | undefined\n\n  select = ({\n    index,\n    contiguous = false\n  }: {\n    index: number\n    contiguous: boolean\n  }) => {\n    if (index === null) {\n      return\n    }\n    this.setState(state => {\n      const { lastSelectedRow } = state\n      const { selectedRows } = this.props\n      let selected = [...selectedRows, index]\n      if (this.props.selectable === DataGrid.SINGLE_SELECT) {\n        selected = [index]\n      } else if (contiguous && typeof lastSelectedRow === 'number') {\n        const start = Math.min(lastSelectedRow, index)\n        const end = Math.max(lastSelectedRow, index)\n        selected = uniq([...selectedRows, ...range(start, end + 1)])\n      }\n      this.props.onSelect(selected)\n      return { lastSelectedRow: index }\n    })\n  }\n\n  selectAll = () => {\n    const { rowCount } = this.props\n    this.setSelectedRows(Array.from({ length: rowCount }).map((_, idx) => idx))\n  }\n\n  deselect = ({\n    index,\n    contiguous = false\n  }: {\n    index: number\n    contiguous: boolean\n  }) => {\n    if (index === null) {\n      return\n    }\n    this.setState(state => {\n      const { lastSelectedRow } = state\n      const { selectedRows } = this.props\n      let selected = selectedRows.filter(idx => idx !== index)\n      if (contiguous && typeof lastSelectedRow === 'number') {\n        const start = Math.min(lastSelectedRow, index)\n        const end = Math.max(lastSelectedRow, index)\n        const toDeselect = range(start, end + 1)\n        selected = selectedRows.filter(idx => !includes(toDeselect, idx))\n      }\n      if (\n        this.props.selectable === DataGrid.SINGLE_SELECT &&\n        !selected.length\n      ) {\n        return null\n      }\n      this.props.onSelect(selected)\n      return { lastSelectedRow: index }\n    })\n  }\n\n  deselectAll = () => {\n    this.setSelectedRows([])\n    this.setState({\n      lastSelectedRow: null\n    })\n  }\n\n  toggleSelect = ({\n    index,\n    contiguous\n  }: {\n    index: number\n    contiguous: boolean\n  }) => {\n    if (index === null) {\n      return\n    }\n    if (!this.props.selectedRows.includes(index)) {\n      this.select({ index, contiguous })\n    } else {\n      this.deselect({ index, contiguous })\n    }\n  }\n\n  setSelectedRows = (rows: Array<number>) => {\n    this.props.onSelect(rows)\n  }\n\n  handleFilter = ({\n    sortBy,\n    sortDirection,\n    filters\n  }: HandleFilterArguments) => {\n    const {\n      sortBy: oldSortBy,\n      sortDirection: oldSortDirection,\n      filters: oldFilters\n    } = this.props\n    const newData = {\n      sortBy: sortBy || oldSortBy,\n      sortDirection: sortDirection || oldSortDirection,\n      filters: {\n        ...oldFilters,\n        ...filters\n      }\n    }\n    this.props.onFilter(newData)\n  }\n\n  handleKeyDown = (event: React.KeyboardEvent<HTMLDivElement>) => {\n    if (this.state.disableKeyboardShortcuts) {\n      return\n    }\n    switch (event.key) {\n      case 'a':\n        if (\n          (event.metaKey || event.ctrlKey) &&\n          this.props.selectable === DataGrid.MULTI_SELECT\n        ) {\n          event.preventDefault()\n          if (this.props.rowCount === this.props.selectedRows.length) {\n            this.deselectAll()\n          } else {\n            this.selectAll()\n          }\n        }\n        break\n      case 'ArrowUp':\n        this.setState(state => ({\n          focusedRow: Math.max((state.focusedRow || 0) - 1, 0)\n        }))\n        event.preventDefault()\n        break\n      case 'ArrowDown':\n        this.setState(state => ({\n          focusedRow:\n            state.focusedRow != null\n              ? Math.min(state.focusedRow + 1, this.props.rowCount - 1)\n              : 0\n        }))\n        event.preventDefault()\n        break\n      case 'Enter':\n      case ' ':\n        if (this.props.selectable) {\n          this.toggleSelect({\n            index: this.state.focusedRow || 0,\n            contiguous: event.shiftKey\n          })\n        }\n        event.preventDefault()\n        break\n      case 'Shift':\n        this.setState({\n          selecting: true\n        })\n        break\n    }\n  }\n\n  handleKeyUp = (event: React.KeyboardEvent<HTMLDivElement>) => {\n    if (this.state.disableKeyboardShortcuts) {\n      return\n    }\n    switch (event.key) {\n      case 'Shift':\n        this.setState({\n          selecting: false\n        })\n        break\n    }\n  }\n\n  // throttling this so a double click behaves like a single click\n  handleRowClick = throttle(\n    ({ event, index }: { event: React.MouseEvent<any>; index: number }) => {\n      if (!this.props.allowRowEventPassthrough) {\n        event.stopPropagation()\n      }\n\n      this.toggleSelect({ index, contiguous: event.shiftKey })\n    },\n    250,\n    { leading: true, trailing: false }\n  )\n\n  mapChildren = () => {\n    const IsArray = (children): children is Array<ColumnFunction> =>\n      Array.isArray(children)\n\n    const children: Array<ColumnFunction> = IsArray(this.props.children)\n      ? this.props.children\n      : [this.props.children]\n\n    const validChildren = children.filter(child => typeof child === 'function')\n\n    return validChildren.map((child, index) =>\n      child({\n        locale: this.props.locale || 'en',\n        localeStrings: this.props.localeStrings || {},\n        columnIndex: index,\n        columnCount: validChildren.length,\n        getFilterValue: dataKey =>\n          this.props.filters && this.props.filters[dataKey],\n        onHeaderPopoverOpen: open =>\n          this.setState({ disableKeyboardShortcuts: open }),\n        onSort: sortArgs => {\n          this.setSelectedRows([])\n          this.handleFilter({ ...sortArgs })\n        },\n        onFilter: ({ dataKey, value, operator }) => {\n          this.setSelectedRows([])\n          this.handleFilter({\n            filters: {\n              [dataKey]: operator\n                ? {\n                    value,\n                    operator\n                  }\n                : value\n            }\n          })\n        }\n      })\n    )\n  }\n\n  isRowLoaded = ({ index }: Index) => {\n    const { hasMoreRows, rowCount } = this.props\n    return !hasMoreRows || index < rowCount\n  }\n\n  isContextMenuVisible = (index: number) =>\n    this.state.activeContextMenu && this.state.activeContextMenu.index === index\n\n  closeContextMenu = () => {\n    this.setState({\n      activeContextMenu: null\n    })\n  }\n\n  contextMenuRenderer = (index: number) => {\n    if (\n      this.props.contextMenuRenderer &&\n      this.isContextMenuVisible(index) &&\n      document.body !== null\n    ) {\n      const contextMenuCloseWrapper = (\n        <CloseWrapper onClose={this.closeContextMenu}>\n          {this.props.contextMenuRenderer({\n            ...this.state.activeContextMenu,\n            selectedRows: this.props.selectedRows,\n            onClose: this.closeContextMenu\n          })}\n        </CloseWrapper>\n      )\n      return ReactDOM.createPortal(contextMenuCloseWrapper, document.body)\n    }\n    return null\n  }\n\n  headerRowRenderer = ({\n    columns,\n    style\n  }: {\n    columns: React.ReactNode[]\n    style: {}\n  }) => (\n    <Elements.Row role=\"row\" header zebra style={style}>\n      {columns}\n    </Elements.Row>\n  )\n\n  rowRenderer = ({\n    key,\n    style,\n    index,\n    rowData,\n    columns,\n    className,\n    onRowMouseOver,\n    onRowMouseOut\n  }: TableRowProps) => (\n    <FocusableRow\n      className={className}\n      aria-rowindex={index + 1}\n      zebra={index % 2 === 1}\n      selected={this.props.selectedRows.includes(index)}\n      selecting={this.state.selecting}\n      focused={index === this.state.focusedRow}\n      key={key}\n      style={style}\n      onMouseLeave={event => {\n        onRowMouseOut && onRowMouseOut({ rowData, index, event })\n      }}\n      onMouseEnter={event => {\n        onRowMouseOver && onRowMouseOver({ rowData, index, event })\n      }}\n      onFocus={() => this.setState({ focusedRow: index })}\n      data-instrumentation={`${this.props['data-instrumentation']}-row${\n        index + 1\n      }`}\n      onClick={\n        this.props.selectable && !this.state.activeContextMenu\n          ? event => this.handleRowClick({ event, index: index })\n          : undefined\n      }\n      onContextMenu={\n        this.props.contextMenuRenderer\n          ? event => {\n              this.setState({\n                activeContextMenu: {\n                  left: event.clientX,\n                  top: event.clientY,\n                  index: index,\n                  rowData: rowData\n                }\n              })\n              if (\n                !this.props.selectedRows.includes(index) &&\n                this.props.selectable\n              ) {\n                this.handleRowClick({ event, index: index })\n              }\n              event.preventDefault()\n            }\n          : undefined\n      }\n      onDoubleClick={event =>\n        this.props.onRowDoubleClick &&\n        this.props.onRowDoubleClick({ index, event, rowData })\n      }\n    >\n      {this.columnTransformer\n        ? this.columnTransformer(columns, rowData, index)\n        : columns}\n      {this.contextMenuRenderer(index)}\n    </FocusableRow>\n  )\n\n  /**\n   *  @typescript-eslint/no-unused-vars is added because even though the props are not used in this component\n   * they are getting destructured into ...props and passed further down the tree\n   */\n  render() {\n    const {\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      children,\n      hasMoreRows,\n      height,\n      loadMoreRows,\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      onFilter,\n      rowCount,\n      sortBy = '',\n      sortDirection,\n      width,\n      ...props\n    } = this.props\n\n    // This fixes an issue where the header of the DataGrid doesn't render on page load in Safari\n    const safariStyle = {\n      transform: `translateZ(0)`\n    }\n\n    const mappedChildren = this.mapChildren()\n    // TODO: Better type signature when accessing children props?\n    const spanData = mappedChildren.map((child: React.ReactElement) => ({\n      colspan: child.props.colspan ? child.props.colspan : () => 1,\n      width: child.props.width,\n      flexGrow: child.props.flexGrow\n    }))\n    this.columnTransformer = (\n      columns: React.ReactElement<any>[],\n      rowData,\n      rowIndex\n    ) => {\n      const spans = spanData.map(item => item.colspan(rowData))\n      const spannedColumns: React.ReactElement[] = []\n      let index = 0\n      while (index < columns.length) {\n        const span = spans[index]\n        const column = columns[index]\n        if (span > 1) {\n          const followingColumns = spanData.slice(index, span)\n          // find new width by adding current column width plus columns that are spanned over\n\n          const width = followingColumns.reduce(\n            (accumulator, { width }: { width: number }) => accumulator + width,\n            0\n          )\n\n          const flexGrow = followingColumns.reduce(\n            (accumulator, { flexGrow }: { flexGrow: number }) =>\n              accumulator + flexGrow,\n            0\n          )\n          spannedColumns.push(\n            React.cloneElement(column, {\n              'data-instrumentation': `${this.props['data-instrumentation']}-row${rowIndex}-col${index}`,\n              style: {\n                ...Elements.cellResetStyle,\n                flex: `0 1 ${width}px`,\n                flexGrow\n              }\n            })\n          )\n          index += span // skip columns that have been spanned over\n        } else {\n          spannedColumns.push(\n            React.cloneElement(column, {\n              'data-instrumentation': `${this.props['data-instrumentation']}-row${rowIndex}-col${index}`\n            })\n          )\n          index++\n        }\n      }\n      return spannedColumns\n    }\n\n    const TableComponent =\n      this.props.scrollWidth !== undefined ? ScrollableTable : Table\n\n    return (\n      <Elements.TableContainer\n        css={{ width, height }}\n        onKeyDown={this.handleKeyDown}\n        onKeyUp={this.handleKeyUp}\n        data-instrumentation={`${this.props['data-instrumentation']}-grid`}\n      >\n        <InfiniteLoader\n          isRowLoaded={this.isRowLoaded}\n          loadMoreRows={loadMoreRows || (() => Promise.resolve(false))}\n          rowCount={hasMoreRows ? rowCount + 1 : rowCount}\n          threshold={5}\n          ref={this.infiniteTable}\n        >\n          {({ onRowsRendered, registerChild }) => (\n            <TableComponent\n              style={safariStyle}\n              gridStyle={{ outline: 'none' }}\n              headerHeight={36}\n              rowHeight={30}\n              sortBy={sortBy}\n              sortDirection={sortDirection}\n              ref={registerChild}\n              noRowsRendered={() => <div>No results</div>}\n              onRowsRendered={onRowsRendered}\n              onScroll={\n                this.state.activeContextMenu\n                  ? () => this.closeContextMenu()\n                  : undefined\n              }\n              rowCount={hasMoreRows ? rowCount + 1 : rowCount}\n              headerRowRenderer={this.headerRowRenderer}\n              rowRenderer={this.rowRenderer}\n              // account for border width of container\n              width={width - 2}\n              height={height - 2}\n              {...props}\n            >\n              {mappedChildren}\n            </TableComponent>\n          )}\n        </InfiniteLoader>\n      </Elements.TableContainer>\n    )\n  }\n}\n\nexport default DataGrid\n"]} */"),
615
+ }, process.env.NODE_ENV === "production" ? "" : ";label:render;", process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/DataGrid/DataGrid.tsx"],"names":[],"mappings":"AAylBQ","file":"../../../src/DataGrid/DataGrid.tsx","sourcesContent":["import React, { Component } from 'react'\nimport ReactDOM from 'react-dom'\nimport {\n  InfiniteLoader,\n  Table,\n  SortDirection,\n  TableRowProps\n} from 'react-virtualized'\nimport type {\n  SortDirectionType,\n  TableProps,\n  IndexRange,\n  Index,\n  RowMouseEventHandlerParams\n} from 'react-virtualized'\nimport * as Elements from './elements'\nimport * as Renderers from './renderers'\nimport {\n  BasicColumn,\n  FocusableRow,\n  InputCell,\n  CollapsibleCell,\n  ScrollableTable\n} from './components'\nimport { CloseWrapper } from '../utils'\nimport curry from 'lodash.curry'\nimport includes from 'lodash.includes'\nimport noop from 'lodash.noop'\nimport range from 'lodash.range'\nimport throttle from 'lodash.throttle'\nimport uniq from 'lodash.uniq'\nimport * as dataGridUtils from './utils'\nimport type { BasicColumnMappedProps } from './components/BasicColumn'\nimport type { LocaleStrings } from './constants'\n\nexport type onFilterArguments = {\n  filters: { [dataKey: string]: any }\n  sortBy?: string\n  sortDirection?: SortDirectionType\n}\n\ntype activeContextMenuState = {\n  top: number\n  left: number\n  index: number\n  rowData: any\n}\n\ntype contextMenuRendererArguments = {\n  top?: number\n  left?: number\n  index?: number\n  rowData?: any\n  selectedRows: Array<number>\n  onClose: () => void\n}\n\nexport type SelectableType = 'single' | 'multi'\n\ntype ColumnFunction = (args: BasicColumnMappedProps) => React.ReactNode\n\nexport type DataGridProps = {\n  /** Functions that create react virtualized Columns. Default column is exposed as DataGrid.Column. */\n  // Can take a single column or an array of columns\n  children: ColumnFunction | Array<ColumnFunction>\n  /** Instrumentation */\n  'data-instrumentation'?: string\n  /** Called during the onContextMenu event of each row. */\n  contextMenuRenderer?: (\n    contextMenuRendererArguments: contextMenuRendererArguments\n  ) => React.ReactNode\n  /** Called when a column is filtered or sorted. */\n  onFilter: (onFilterArguments: onFilterArguments) => void\n  /** Called when the row selection changes. Receives array of indices. */\n  onSelect: (indexes: number[]) => void\n  /** Called when the row is double clicked. Receives index. */\n  onRowDoubleClick?: (info: RowMouseEventHandlerParams) => void\n  /** Infinite loading: Indicates more rows need to be loaded. */\n  hasMoreRows?: boolean\n  /** Infinite loading: Called when more rows need to be loaded. */\n  loadMoreRows?: (props: IndexRange) => Promise<any>\n  /** Count of rows that have already been loaded. */\n  rowCount: number\n  /** Required by react-virtualized table. */\n  rowGetter: ({ index: number }) => any\n  /** Whether rows can be selected (and how many). */\n  selectable?: SelectableType | false\n  /** Array of indices that should be selected. Only used if `selectable` is set. */\n  selectedRows: Array<number>\n  /** The column you want to sort */\n  sortBy?: string\n  /** The direction sorting should be done */\n  sortDirection?: SortDirectionType\n  /** A map of columns to filter data */\n  filters?: { [dataKey: string]: any }\n  /** Set the width of scrollable content */\n  scrollWidth?: number\n  width: number\n  height: number\n  /** Locale string: ja, de, es, ko, en, etc. */\n  locale?: string\n  /** Locale strings to override en for language in locale */\n  localeStrings?: LocaleStrings\n  /** allow events to propagate to a cell on row select */\n  allowRowEventPassthrough?: boolean\n} & Partial<TableProps>\n\ntype DataGridState = {\n  activeContextMenu: activeContextMenuState | null\n  disableKeyboardShortcuts: boolean\n  focusedRow: number | null\n  selecting: boolean\n  lastSelectedRow: number | null\n}\n\ntype HandleFilterArguments = {\n  sortBy?: string\n  sortDirection?: SortDirectionType\n  filters?: any\n}\n\n/**\n * Component for displaying data in a grid with support for things like\n * filtering, sorting, and context menus.\n *\n * Depends on react-virtualized/styles.css.\n *\n * For react-virtualized Table docs see: https://github.com/bvaughn/react-virtualized/blob/master/docs/Table.md\n *\n */\nclass DataGrid extends Component<DataGridProps, DataGridState> {\n  // expose sub-components\n  static Column = curry(BasicColumn)\n  static SortDirection = SortDirection\n  static Elements = Elements\n  static InputCell = InputCell\n  static CollapsibleCell = CollapsibleCell\n  static Renderers = Renderers\n\n  static SINGLE_SELECT: SelectableType = 'single'\n  static MULTI_SELECT: SelectableType = 'multi'\n\n  static utils = dataGridUtils\n\n  static defaultProps = {\n    loadMoreRows: noop,\n    onSelect: noop,\n    onFilter: noop,\n    'data-instrumentation': 'datagrid',\n    sortBy: '',\n    sortDirection: SortDirection.ASC,\n    filters: {},\n    selectedRows: [],\n    allowRowEventPassthrough: false\n  }\n\n  static displayName = 'DataGrid'\n\n  state: DataGridState = {\n    activeContextMenu: null,\n    disableKeyboardShortcuts: false,\n    focusedRow: null,\n    lastSelectedRow: null,\n    selecting: false\n  }\n\n  infiniteTable = React.createRef<InfiniteLoader>()\n\n  columnTransformer: (\n    columns: React.ReactElement<any>[],\n    rowData: any,\n    rowIndex: number\n  ) => React.ReactNode[] | null | undefined\n\n  select = ({\n    index,\n    contiguous = false\n  }: {\n    index: number\n    contiguous: boolean\n  }) => {\n    if (index === null) {\n      return\n    }\n    this.setState(state => {\n      const { lastSelectedRow } = state\n      const { selectedRows } = this.props\n      let selected = [...selectedRows, index]\n      if (this.props.selectable === DataGrid.SINGLE_SELECT) {\n        selected = [index]\n      } else if (contiguous && typeof lastSelectedRow === 'number') {\n        const start = Math.min(lastSelectedRow, index)\n        const end = Math.max(lastSelectedRow, index)\n        selected = uniq([...selectedRows, ...range(start, end + 1)])\n      }\n      this.props.onSelect(selected)\n      return { lastSelectedRow: index }\n    })\n  }\n\n  selectAll = () => {\n    const { rowCount } = this.props\n    this.setSelectedRows(Array.from({ length: rowCount }).map((_, idx) => idx))\n  }\n\n  deselect = ({\n    index,\n    contiguous = false\n  }: {\n    index: number\n    contiguous: boolean\n  }) => {\n    if (index === null) {\n      return\n    }\n    this.setState(state => {\n      const { lastSelectedRow } = state\n      const { selectedRows } = this.props\n      let selected = selectedRows.filter(idx => idx !== index)\n      if (contiguous && typeof lastSelectedRow === 'number') {\n        const start = Math.min(lastSelectedRow, index)\n        const end = Math.max(lastSelectedRow, index)\n        const toDeselect = range(start, end + 1)\n        selected = selectedRows.filter(idx => !includes(toDeselect, idx))\n      }\n      if (\n        this.props.selectable === DataGrid.SINGLE_SELECT &&\n        !selected.length\n      ) {\n        return null\n      }\n      this.props.onSelect(selected)\n      return { lastSelectedRow: index }\n    })\n  }\n\n  deselectAll = () => {\n    this.setSelectedRows([])\n    this.setState({\n      lastSelectedRow: null\n    })\n  }\n\n  toggleSelect = ({\n    index,\n    contiguous\n  }: {\n    index: number\n    contiguous: boolean\n  }) => {\n    if (index === null) {\n      return\n    }\n    if (!this.props.selectedRows.includes(index)) {\n      this.select({ index, contiguous })\n    } else {\n      this.deselect({ index, contiguous })\n    }\n  }\n\n  setSelectedRows = (rows: Array<number>) => {\n    this.props.onSelect(rows)\n  }\n\n  handleFilter = ({\n    sortBy,\n    sortDirection,\n    filters\n  }: HandleFilterArguments) => {\n    const {\n      sortBy: oldSortBy,\n      sortDirection: oldSortDirection,\n      filters: oldFilters\n    } = this.props\n    const newData = {\n      sortBy: sortBy || oldSortBy,\n      sortDirection: sortDirection || oldSortDirection,\n      filters: {\n        ...oldFilters,\n        ...filters\n      }\n    }\n    this.props.onFilter(newData)\n  }\n\n  handleKeyDown = (event: React.KeyboardEvent<HTMLDivElement>) => {\n    if (this.state.disableKeyboardShortcuts) {\n      return\n    }\n    switch (event.key) {\n      case 'a':\n        if (\n          (event.metaKey || event.ctrlKey) &&\n          this.props.selectable === DataGrid.MULTI_SELECT\n        ) {\n          event.preventDefault()\n          if (this.props.rowCount === this.props.selectedRows.length) {\n            this.deselectAll()\n          } else {\n            this.selectAll()\n          }\n        }\n        break\n      case 'ArrowUp':\n        this.setState(state => ({\n          focusedRow: Math.max((state.focusedRow || 0) - 1, 0)\n        }))\n        event.preventDefault()\n        break\n      case 'ArrowDown':\n        this.setState(state => ({\n          focusedRow:\n            state.focusedRow != null\n              ? Math.min(state.focusedRow + 1, this.props.rowCount - 1)\n              : 0\n        }))\n        event.preventDefault()\n        break\n      case 'Enter':\n      case ' ':\n        if (this.props.selectable) {\n          this.toggleSelect({\n            index: this.state.focusedRow || 0,\n            contiguous: event.shiftKey\n          })\n        }\n        event.preventDefault()\n        break\n      case 'Shift':\n        this.setState({\n          selecting: true\n        })\n        break\n    }\n  }\n\n  handleKeyUp = (event: React.KeyboardEvent<HTMLDivElement>) => {\n    if (this.state.disableKeyboardShortcuts) {\n      return\n    }\n    switch (event.key) {\n      case 'Shift':\n        this.setState({\n          selecting: false\n        })\n        break\n    }\n  }\n\n  // throttling this so a double click behaves like a single click\n  handleRowClick = throttle(\n    ({ event, index }: { event: React.MouseEvent<any>; index: number }) => {\n      if (!this.props.allowRowEventPassthrough) {\n        event.stopPropagation()\n      }\n\n      this.toggleSelect({ index, contiguous: event.shiftKey })\n    },\n    250,\n    { leading: true, trailing: false }\n  )\n\n  mapChildren = () => {\n    const IsArray = (children): children is Array<ColumnFunction> =>\n      Array.isArray(children)\n\n    const children: Array<ColumnFunction> = IsArray(this.props.children)\n      ? this.props.children\n      : [this.props.children]\n\n    const validChildren = children.filter(child => typeof child === 'function')\n\n    return validChildren.map((child, index) =>\n      child({\n        locale: this.props.locale || 'en',\n        localeStrings: this.props.localeStrings || {},\n        columnIndex: index,\n        columnCount: validChildren.length,\n        getFilterValue: dataKey =>\n          this.props.filters && this.props.filters[dataKey],\n        onHeaderPopoverOpen: open =>\n          this.setState({ disableKeyboardShortcuts: open }),\n        onSort: sortArgs => {\n          this.setSelectedRows([])\n          this.handleFilter({ ...sortArgs })\n        },\n        onFilter: ({ dataKey, value, operator }) => {\n          this.setSelectedRows([])\n          this.handleFilter({\n            filters: {\n              [dataKey]: operator\n                ? {\n                    value,\n                    operator\n                  }\n                : value\n            }\n          })\n        }\n      })\n    )\n  }\n\n  isRowLoaded = ({ index }: Index) => {\n    const { hasMoreRows, rowCount } = this.props\n    return !hasMoreRows || index < rowCount\n  }\n\n  isContextMenuVisible = (index: number) =>\n    this.state.activeContextMenu && this.state.activeContextMenu.index === index\n\n  closeContextMenu = () => {\n    this.setState({\n      activeContextMenu: null\n    })\n  }\n\n  contextMenuRenderer = (index: number) => {\n    if (\n      this.props.contextMenuRenderer &&\n      this.isContextMenuVisible(index) &&\n      document.body !== null\n    ) {\n      const contextMenuCloseWrapper = (\n        <CloseWrapper onClose={this.closeContextMenu}>\n          {this.props.contextMenuRenderer({\n            ...this.state.activeContextMenu,\n            selectedRows: this.props.selectedRows,\n            onClose: this.closeContextMenu\n          })}\n        </CloseWrapper>\n      )\n      return ReactDOM.createPortal(contextMenuCloseWrapper, document.body)\n    }\n    return null\n  }\n\n  headerRowRenderer = ({\n    columns,\n    style\n  }: {\n    columns: React.ReactNode[]\n    style: {}\n  }) => (\n    <Elements.Row role=\"row\" header zebra style={style}>\n      {columns}\n    </Elements.Row>\n  )\n\n  rowRenderer = ({\n    key,\n    style,\n    index,\n    rowData,\n    columns,\n    className,\n    onRowMouseOver,\n    onRowMouseOut\n  }: TableRowProps) => (\n    <FocusableRow\n      className={className}\n      aria-rowindex={index + 1}\n      zebra={index % 2 === 1}\n      selected={this.props.selectedRows.includes(index)}\n      selecting={this.state.selecting}\n      focused={index === this.state.focusedRow}\n      key={key}\n      style={style}\n      onMouseLeave={event => {\n        onRowMouseOut && onRowMouseOut({ rowData, index, event })\n      }}\n      onMouseEnter={event => {\n        onRowMouseOver && onRowMouseOver({ rowData, index, event })\n      }}\n      onFocus={() => this.setState({ focusedRow: index })}\n      data-instrumentation={`${this.props['data-instrumentation']}-row${\n        index + 1\n      }`}\n      onClick={\n        this.props.selectable && !this.state.activeContextMenu\n          ? event => this.handleRowClick({ event, index: index })\n          : undefined\n      }\n      onContextMenu={\n        this.props.contextMenuRenderer\n          ? event => {\n              this.setState({\n                activeContextMenu: {\n                  left: event.clientX,\n                  top: event.clientY,\n                  index: index,\n                  rowData: rowData\n                }\n              })\n              if (\n                !this.props.selectedRows.includes(index) &&\n                this.props.selectable\n              ) {\n                this.handleRowClick({ event, index: index })\n              }\n              event.preventDefault()\n            }\n          : undefined\n      }\n      onDoubleClick={event =>\n        this.props.onRowDoubleClick &&\n        this.props.onRowDoubleClick({ index, event, rowData })\n      }\n    >\n      {this.columnTransformer\n        ? this.columnTransformer(columns, rowData, index)\n        : columns}\n      {this.contextMenuRenderer(index)}\n    </FocusableRow>\n  )\n\n  /**\n   *  @typescript-eslint/no-unused-vars is added because even though the props are not used in this component\n   * they are getting destructured into ...props and passed further down the tree\n   */\n  render() {\n    const {\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      children,\n      hasMoreRows,\n      height,\n      loadMoreRows,\n      // eslint-disable-next-line @typescript-eslint/no-unused-vars\n      onFilter,\n      rowCount,\n      sortBy = '',\n      sortDirection,\n      width,\n      ...props\n    } = this.props\n\n    // This fixes an issue where the header of the DataGrid doesn't render on page load in Safari\n    const safariStyle = {\n      transform: `translateZ(0)`\n    }\n\n    const mappedChildren = this.mapChildren().filter(child => child !== null)\n    // TODO: Better type signature when accessing children props?\n    const spanData = mappedChildren.map((child: React.ReactElement) => ({\n      colspan: child.props.colspan ? child.props.colspan : () => 1,\n      width: child.props.width,\n      flexGrow: child.props.flexGrow\n    }))\n    this.columnTransformer = (\n      columns: React.ReactElement<any>[],\n      rowData,\n      rowIndex\n    ) => {\n      const spans = spanData.map(item => item.colspan(rowData))\n      const spannedColumns: React.ReactElement[] = []\n      let index = 0\n      while (index < columns.length) {\n        const span = spans[index]\n        const column = columns[index]\n        if (span > 1) {\n          const followingColumns = spanData.slice(index, span)\n          // find new width by adding current column width plus columns that are spanned over\n\n          const width = followingColumns.reduce(\n            (accumulator, { width }: { width: number }) => accumulator + width,\n            0\n          )\n\n          const flexGrow = followingColumns.reduce(\n            (accumulator, { flexGrow }: { flexGrow: number }) =>\n              accumulator + flexGrow,\n            0\n          )\n          spannedColumns.push(\n            React.cloneElement(column, {\n              'data-instrumentation': `${this.props['data-instrumentation']}-row${rowIndex}-col${index}`,\n              style: {\n                ...Elements.cellResetStyle,\n                flex: `0 1 ${width}px`,\n                flexGrow\n              }\n            })\n          )\n          index += span // skip columns that have been spanned over\n        } else {\n          spannedColumns.push(\n            React.cloneElement(column, {\n              'data-instrumentation': `${this.props['data-instrumentation']}-row${rowIndex}-col${index}`\n            })\n          )\n          index++\n        }\n      }\n      return spannedColumns\n    }\n\n    const TableComponent =\n      this.props.scrollWidth !== undefined ? ScrollableTable : Table\n\n    return (\n      <Elements.TableContainer\n        css={{ width, height }}\n        onKeyDown={this.handleKeyDown}\n        onKeyUp={this.handleKeyUp}\n        data-instrumentation={`${this.props['data-instrumentation']}-grid`}\n      >\n        <InfiniteLoader\n          isRowLoaded={this.isRowLoaded}\n          loadMoreRows={loadMoreRows || (() => Promise.resolve(false))}\n          rowCount={hasMoreRows ? rowCount + 1 : rowCount}\n          threshold={5}\n          ref={this.infiniteTable}\n        >\n          {({ onRowsRendered, registerChild }) => (\n            <TableComponent\n              style={safariStyle}\n              gridStyle={{ outline: 'none' }}\n              headerHeight={36}\n              rowHeight={30}\n              sortBy={sortBy}\n              sortDirection={sortDirection}\n              ref={registerChild}\n              noRowsRendered={() => <div>No results</div>}\n              onRowsRendered={onRowsRendered}\n              onScroll={\n                this.state.activeContextMenu\n                  ? () => this.closeContextMenu()\n                  : undefined\n              }\n              rowCount={hasMoreRows ? rowCount + 1 : rowCount}\n              headerRowRenderer={this.headerRowRenderer}\n              rowRenderer={this.rowRenderer}\n              // account for border width of container\n              width={width - 2}\n              height={height - 2}\n              {...props}\n            >\n              {mappedChildren}\n            </TableComponent>\n          )}\n        </InfiniteLoader>\n      </Elements.TableContainer>\n    )\n  }\n}\n\nexport default DataGrid\n"]} */"),
614
616
  onKeyDown: this.handleKeyDown,
615
617
  onKeyUp: this.handleKeyUp,
616
618
  "data-instrumentation": "".concat(this.props['data-instrumentation'], "-grid"),
@@ -12,6 +12,7 @@ export declare type BasicColumnProps = {
12
12
  sortAscIcon?: React.ReactNode;
13
13
  sortDescIcon?: React.ReactNode;
14
14
  sortRenderer?: ((arg0: sortRendererArguments) => React.ReactNode) | false | null;
15
+ hideColumn?: boolean;
15
16
  } & ColumnProps;
16
17
  export declare type BasicColumnMappedProps = {
17
18
  /** Locale */
@@ -38,5 +39,5 @@ export declare type BasicColumnMappedProps = {
38
39
  *
39
40
  * Any Column prop can be overridden. See https://github.com/bvaughn/react-virtualized/blob/master/docs/Column.md
40
41
  */
41
- export default function BasicColumn(columnProps: BasicColumnProps, mappedProps: BasicColumnMappedProps): import("@emotion/react/jsx-runtime").JSX.Element;
42
+ export default function BasicColumn(columnProps: BasicColumnProps, mappedProps: BasicColumnMappedProps): import("@emotion/react/jsx-runtime").JSX.Element | null;
42
43
  //# sourceMappingURL=BasicColumn.d.ts.map
@@ -61,6 +61,10 @@ function BasicColumn(columnProps, mappedProps) {
61
61
  localeStrings = _mappedProps$localeSt === void 0 ? {} : _mappedProps$localeSt,
62
62
  onHeaderPopoverOpen = mappedProps.onHeaderPopoverOpen;
63
63
 
64
+ if (columnProps.hideColumn) {
65
+ return null;
66
+ }
67
+
64
68
  var columnCellRenderer = function columnCellRenderer(cellRendererProps) {
65
69
  return (0, _jsxRuntime.jsx)(_elements.Cell, {
66
70
  "data-gs": gs("src", "datagrid", "components", "basic-column", "cell"),
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "td-stylekit",
3
- "version": "26.9.1",
3
+ "version": "26.10.2",
4
4
  "main": "dist/es/index.js",
5
5
  "module": "dist/es/index.js",
6
6
  "types": "dist/es/index.d.ts",
@@ -48,7 +48,7 @@
48
48
  "@emotion/styled": "^11.6.0",
49
49
  "@reach/dialog": "^0.11.2",
50
50
  "@reach/tabs": "^0.10.5",
51
- "@tippyjs/react": "^4.2.0",
51
+ "@tippyjs/react": "^4.2.6",
52
52
  "classnames": "^2.2.6",
53
53
  "color": "^3.1.2",
54
54
  "date-fns": "2.23.0",