@amboss/design-system 2.1.6 → 2.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/cjs/components/Form/Select/BaseSelect.d.ts +1 -1
- package/build/cjs/components/Form/Select/BaseSelect.js +1 -1
- package/build/cjs/components/Form/Select/MultiSelect.js +1 -1
- package/build/cjs/components/Form/Select/Select.d.ts +3 -0
- package/build/cjs/components/Patterns/Modal/Modal.js +1 -1
- package/build/cjs/components/Popover/Popover.d.ts +1 -1
- package/build/cjs/components/Popover/Popover.js +1 -1
- package/build/cjs/components/Toggletip/BasePopover.d.ts +2 -2
- package/build/cjs/components/Toggletip/BasePopover.js +1 -1
- package/build/cjs/components/Toggletip/Toggletip.d.ts +1 -1
- package/build/cjs/components/Toggletip/Toggletip.js +1 -1
- package/build/cjs/components/Tooltip/BaseTooltip.d.ts +2 -2
- package/build/cjs/components/Tooltip/BaseTooltip.js +1 -1
- package/build/cjs/components/Tooltip/Tooltip.d.ts +1 -1
- package/build/cjs/components/Tooltip/Tooltip.js +1 -1
- package/build/cjs/components/Tooltip/TooltipContent.d.ts +3 -2
- package/build/cjs/components/Tooltip/TooltipContent.js +1 -1
- package/build/cjs/components/Tutorialtip/Tutorialtip.d.ts +5 -0
- package/build/cjs/components/Tutorialtip/Tutorialtip.js +1 -0
- package/build/cjs/components/UserHighlightTooltip/UserHighlightTooltip.js +1 -1
- package/build/cjs/index.d.ts +1 -0
- package/build/cjs/index.js +1 -1
- package/build/cjs/shared/useAutoPosition.d.ts +8 -1
- package/build/cjs/shared/useAutoPosition.js +1 -1
- package/build/esm/components/Form/Select/BaseSelect.d.ts +1 -1
- package/build/esm/components/Form/Select/BaseSelect.js +1 -1
- package/build/esm/components/Form/Select/MultiSelect.js +1 -1
- package/build/esm/components/Form/Select/Select.d.ts +3 -0
- package/build/esm/components/Patterns/Modal/Modal.js +1 -1
- package/build/esm/components/Popover/Popover.d.ts +1 -1
- package/build/esm/components/Popover/Popover.js +1 -1
- package/build/esm/components/Toggletip/BasePopover.d.ts +2 -2
- package/build/esm/components/Toggletip/BasePopover.js +1 -1
- package/build/esm/components/Toggletip/Toggletip.d.ts +1 -1
- package/build/esm/components/Toggletip/Toggletip.js +1 -1
- package/build/esm/components/Tooltip/BaseTooltip.d.ts +2 -2
- package/build/esm/components/Tooltip/BaseTooltip.js +1 -1
- package/build/esm/components/Tooltip/Tooltip.d.ts +1 -1
- package/build/esm/components/Tooltip/Tooltip.js +1 -1
- package/build/esm/components/Tooltip/TooltipContent.d.ts +3 -2
- package/build/esm/components/Tooltip/TooltipContent.js +1 -1
- package/build/esm/components/Tutorialtip/Tutorialtip.d.ts +5 -0
- package/build/esm/components/Tutorialtip/Tutorialtip.js +1 -0
- package/build/esm/components/UserHighlightTooltip/UserHighlightTooltip.js +1 -1
- package/build/esm/index.d.ts +1 -0
- package/build/esm/index.js +1 -1
- package/build/esm/shared/useAutoPosition.d.ts +8 -1
- package/build/esm/shared/useAutoPosition.js +1 -1
- package/package.json +5 -5
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useRef,useMemo,useEffect,useCallback,useReducer}from"react";import styled from"@emotion/styled";import{useKeyboard}from"../../../shared/useKeyboard";import{useAutoPosition}from"../../../shared/useAutoPosition";import{VirtualScrollList}from"../../VirtualScrollList/VirtualScrollList";import{InputRaw}from"../Input/Input";import{CheckboxRaw}from"../Checkbox/Checkbox";import{Text}from"../../Typography/Text/Text";import{Box}from"../../Box/Box";import{Tag}from"../../Tag/Tag";import{StyledOption,StyledContainer,StyledDropdown,StyledFakeInputWrap}from"./StyledSelectComponents";import{MultiSelectReducer}from"./MultiSelectReducer";let StyledInputWrap=styled("div",{target:"ex2053y0",label:"StyledInputWrap"})(({theme,hasValue})=>({zIndex:1,display:"flex",alignItems:"center",position:"relative","& input":{...hasValue&&{paddingLeft:theme.variables.size.spacing.xxs}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/MultiSelect.tsx","sources":["src/components/Form/Select/MultiSelect.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n  useReducer,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport type { CommonSelectProps, SelectOption } from \"./Select\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { InputRaw } from \"../Input/Input\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Tag } from \"../../Tag/Tag\";\nimport {\n  StyledOption,\n  StyledContainer,\n  StyledDropdown,\n  StyledFakeInputWrap,\n} from \"./StyledSelectComponents\";\nimport { MultiSelectReducer } from \"./MultiSelectReducer\";\n\nexport type ActionData = {\n  action: \"select-option\" | \"deselect-option\" | \"clear\";\n  name: string;\n  data: SelectOption[];\n};\n\nexport type UniqueMultiSelectProps = {\n  value: string[];\n  onChange?: (\n    value: string[],\n    actionData: ActionData,\n    event: React.ChangeEvent<HTMLSelectElement>\n  ) => void;\n  formatSelectedOptionsLabel?: (count?: number) => string;\n};\n\ntype MultiSelectProps = CommonSelectProps &\n  Pick<FormFieldProps, \"label\"> &\n  UniqueMultiSelectProps;\n\nconst StyledInputWrap = styled.div<{ hasValue: boolean }>(\n  ({ theme, hasValue }) => ({\n    zIndex: 1,\n    display: \"flex\",\n    alignItems: \"center\",\n    position: \"relative\",\n\n    \"& input\": {\n      ...(hasValue && {\n        paddingLeft: theme.variables.size.spacing.xxs,\n      }),\n    },\n  })\n);\n\nconst StyledTagWrap = styled.div(({ theme }) => ({\n  marginLeft: theme.variables.size.spacing.xs,\n  order: -1, // it's needed to be 2-nd in html because click on label affects 1-st element (input)\n}));\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst getSelectedOptions = (\n  options: SelectOption[],\n  value: string[]\n): SelectOption[] => {\n  const mapFromValue = new Set(value);\n  return options.filter((opt) => mapFromValue.has(opt.value));\n};\n\nexport function MultiSelect({\n  options,\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod,\n  onChange,\n  onBlur,\n  onFocus,\n  maxHeight,\n  formatSelectedOptionsLabel = (count) => `${count} selected`,\n  autoComplete,\n  disabled,\n  label,\n}: MultiSelectProps): React.ReactElement {\n  const selectRef = useRef<HTMLSelectElement>(null);\n  const initialSelectedOptions = useMemo(\n    () => getSelectedOptions(options, value),\n    [options, value]\n  );\n  const [\n    {\n      selectedOptions,\n      actionName,\n      changedOptions,\n      isOpen,\n      preselectedIndex,\n      innerValue,\n    },\n    dispatch,\n  ] = useReducer(MultiSelectReducer, {\n    selectedOptions: initialSelectedOptions,\n    isOpen: false,\n    preselectedIndex: -1,\n    innerValue: \"\",\n  });\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const handleOnChange = useCallback(\n    (event) => {\n      onChange(\n        selectedOptions.map((opt) => opt.value),\n        {\n          name,\n          action: actionName,\n          data: changedOptions,\n        },\n        event\n      );\n    },\n    [selectedOptions, actionName, changedOptions, name, onChange]\n  );\n\n  const forceChangeFakeSelect = useCallback(\n    (option: SelectOption) => {\n      dispatch({ type: \"onChange\", option });\n    },\n    [dispatch]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    innerInputRef,\n    dropDownRef,\n    isOpen\n  );\n\n  useEffect(() => {\n    dispatch({\n      type: \"updateSelectedOptions\",\n      selectedOptions: initialSelectedOptions,\n    });\n  }, [initialSelectedOptions]);\n\n  const closeDropdown = useCallback(() => {\n    dispatch({ type: \"close\" });\n    dispatch({ type: \"setInnerValue\", value: \"\" });\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n    onBlur();\n  }, [onBlur, dispatch]);\n\n  useEffect(() => {\n    if (selectRef.current) {\n      selectRef.current.dispatchEvent(new Event(\"change\", { bubbles: true }));\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [changedOptions, selectRef.current]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  useEffect(() => {\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n        }\n      },\n      ArrowUp: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.max(preselectedIndex - 1, 0),\n        });\n      },\n      ArrowDown: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.min(preselectedIndex + 1, filteredOptions.length - 1),\n        });\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        dispatch({ type: \"open\" });\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <StyledContainer\n      onBlur={() => {\n        closeDropdown();\n      }}\n    >\n      <StyledInputWrap hasValue={value.length > 0}>\n        <InputRaw\n          areaLabel={label}\n          name={`${name}-innerInput`}\n          value={innerValue}\n          privateProps={{ isTransparent: true, hideOutline: true }}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          disabled={disabled}\n          placeholder={value.length === 0 ? placeholder : undefined}\n          onFocus={() => {\n            dispatch({ type: \"open\" });\n            onFocus();\n          }}\n          onClick={() => {\n            dispatch({ type: \"open\" });\n          }}\n          onChange={(e) => {\n            if (e.currentTarget.value) {\n              dispatch({ type: \"open\" });\n            }\n            dispatch({\n              type: \"setInnerValue\",\n              value: e.currentTarget.value,\n            });\n          }}\n          ref={innerInputRef}\n          autoComplete={autoComplete}\n        />\n        {initialSelectedOptions.length > 0 && (\n          <StyledTagWrap>\n            <Tag\n              label={formatSelectedOptionsLabel(initialSelectedOptions.length)}\n              isRemovable\n              onClear={() => {\n                dispatch({ type: \"clear\" });\n              }}\n            />\n          </StyledTagWrap>\n        )}\n      </StyledInputWrap>\n      <StyledFakeInputWrap>\n        <InputRaw\n          name={`${name}-fakeInput`}\n          value={undefined}\n          type=\"text\"\n          onChange={() => null}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          tabIndex={-1}\n          autoComplete=\"off\"\n          hasError={hasError}\n        />\n      </StyledFakeInputWrap>\n\n      {isOpen && (\n        <StyledDropdown\n          dropdownPosition={verticalPosition}\n          ref={dropDownRef}\n          // this is to prevent known bug of Chrome when element\n          // loses focus on click on the scrollbar\n          onMouseDown={(e) => e.preventDefault()}\n        >\n          <VirtualScrollList\n            maxHeight={maxHeight}\n            itemHeight={36}\n            itemAmount={filteredOptions.length}\n            emptyState={() => (\n              <Box space=\"xs\">\n                <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n              </Box>\n            )}\n            itemInView={preselectedIndex}\n            itemTemplate={(index: number) => {\n              const option: SelectOption = filteredOptions[index];\n              const isActive = value.includes(option.value);\n\n              return (\n                <CustomOption\n                  key={option.value}\n                  active={isActive}\n                  preSelected={preselectedIndex === index}\n                  onMouseDown={() => {\n                    forceChangeFakeSelect(option);\n                  }}\n                >\n                  <CheckboxRaw\n                    name=\"\"\n                    checked={isActive}\n                    size=\"s\"\n                    label={option.label}\n                    onChange={() => null}\n                  />\n                </CustomOption>\n              );\n            }}\n          />\n        </StyledDropdown>\n      )}\n      <select\n        ref={selectRef}\n        name={name}\n        onChange={handleOnChange}\n        data-e2e-test-id=\"multiSelect\"\n        aria-hidden\n        hidden\n        value={value}\n        multiple\n      >\n        {selectedOptions.map((o) => (\n          <option key={o.value} value={o.value}>\n            {o.label}\n          </option>\n        ))}\n      </select>\n    </StyledContainer>\n  );\n}\n"],"names":[],"mappings":"AAgDwB"} */"),StyledTagWrap=styled("div",{target:"ex2053y1",label:"StyledTagWrap"})(({theme})=>({marginLeft:theme.variables.size.spacing.xs,order:-1}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/MultiSelect.tsx","sources":["src/components/Form/Select/MultiSelect.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n  useReducer,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport type { CommonSelectProps, SelectOption } from \"./Select\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { InputRaw } from \"../Input/Input\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Tag } from \"../../Tag/Tag\";\nimport {\n  StyledOption,\n  StyledContainer,\n  StyledDropdown,\n  StyledFakeInputWrap,\n} from \"./StyledSelectComponents\";\nimport { MultiSelectReducer } from \"./MultiSelectReducer\";\n\nexport type ActionData = {\n  action: \"select-option\" | \"deselect-option\" | \"clear\";\n  name: string;\n  data: SelectOption[];\n};\n\nexport type UniqueMultiSelectProps = {\n  value: string[];\n  onChange?: (\n    value: string[],\n    actionData: ActionData,\n    event: React.ChangeEvent<HTMLSelectElement>\n  ) => void;\n  formatSelectedOptionsLabel?: (count?: number) => string;\n};\n\ntype MultiSelectProps = CommonSelectProps &\n  Pick<FormFieldProps, \"label\"> &\n  UniqueMultiSelectProps;\n\nconst StyledInputWrap = styled.div<{ hasValue: boolean }>(\n  ({ theme, hasValue }) => ({\n    zIndex: 1,\n    display: \"flex\",\n    alignItems: \"center\",\n    position: \"relative\",\n\n    \"& input\": {\n      ...(hasValue && {\n        paddingLeft: theme.variables.size.spacing.xxs,\n      }),\n    },\n  })\n);\n\nconst StyledTagWrap = styled.div(({ theme }) => ({\n  marginLeft: theme.variables.size.spacing.xs,\n  order: -1, // it's needed to be 2-nd in html because click on label affects 1-st element (input)\n}));\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst getSelectedOptions = (\n  options: SelectOption[],\n  value: string[]\n): SelectOption[] => {\n  const mapFromValue = new Set(value);\n  return options.filter((opt) => mapFromValue.has(opt.value));\n};\n\nexport function MultiSelect({\n  options,\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod,\n  onChange,\n  onBlur,\n  onFocus,\n  maxHeight,\n  formatSelectedOptionsLabel = (count) => `${count} selected`,\n  autoComplete,\n  disabled,\n  label,\n}: MultiSelectProps): React.ReactElement {\n  const selectRef = useRef<HTMLSelectElement>(null);\n  const initialSelectedOptions = useMemo(\n    () => getSelectedOptions(options, value),\n    [options, value]\n  );\n  const [\n    {\n      selectedOptions,\n      actionName,\n      changedOptions,\n      isOpen,\n      preselectedIndex,\n      innerValue,\n    },\n    dispatch,\n  ] = useReducer(MultiSelectReducer, {\n    selectedOptions: initialSelectedOptions,\n    isOpen: false,\n    preselectedIndex: -1,\n    innerValue: \"\",\n  });\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const handleOnChange = useCallback(\n    (event) => {\n      onChange(\n        selectedOptions.map((opt) => opt.value),\n        {\n          name,\n          action: actionName,\n          data: changedOptions,\n        },\n        event\n      );\n    },\n    [selectedOptions, actionName, changedOptions, name, onChange]\n  );\n\n  const forceChangeFakeSelect = useCallback(\n    (option: SelectOption) => {\n      dispatch({ type: \"onChange\", option });\n    },\n    [dispatch]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    innerInputRef,\n    dropDownRef,\n    isOpen\n  );\n\n  useEffect(() => {\n    dispatch({\n      type: \"updateSelectedOptions\",\n      selectedOptions: initialSelectedOptions,\n    });\n  }, [initialSelectedOptions]);\n\n  const closeDropdown = useCallback(() => {\n    dispatch({ type: \"close\" });\n    dispatch({ type: \"setInnerValue\", value: \"\" });\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n    onBlur();\n  }, [onBlur, dispatch]);\n\n  useEffect(() => {\n    if (selectRef.current) {\n      selectRef.current.dispatchEvent(new Event(\"change\", { bubbles: true }));\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [changedOptions, selectRef.current]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  useEffect(() => {\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n        }\n      },\n      ArrowUp: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.max(preselectedIndex - 1, 0),\n        });\n      },\n      ArrowDown: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.min(preselectedIndex + 1, filteredOptions.length - 1),\n        });\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        dispatch({ type: \"open\" });\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <StyledContainer\n      onBlur={() => {\n        closeDropdown();\n      }}\n    >\n      <StyledInputWrap hasValue={value.length > 0}>\n        <InputRaw\n          areaLabel={label}\n          name={`${name}-innerInput`}\n          value={innerValue}\n          privateProps={{ isTransparent: true, hideOutline: true }}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          disabled={disabled}\n          placeholder={value.length === 0 ? placeholder : undefined}\n          onFocus={() => {\n            dispatch({ type: \"open\" });\n            onFocus();\n          }}\n          onClick={() => {\n            dispatch({ type: \"open\" });\n          }}\n          onChange={(e) => {\n            if (e.currentTarget.value) {\n              dispatch({ type: \"open\" });\n            }\n            dispatch({\n              type: \"setInnerValue\",\n              value: e.currentTarget.value,\n            });\n          }}\n          ref={innerInputRef}\n          autoComplete={autoComplete}\n        />\n        {initialSelectedOptions.length > 0 && (\n          <StyledTagWrap>\n            <Tag\n              label={formatSelectedOptionsLabel(initialSelectedOptions.length)}\n              isRemovable\n              onClear={() => {\n                dispatch({ type: \"clear\" });\n              }}\n            />\n          </StyledTagWrap>\n        )}\n      </StyledInputWrap>\n      <StyledFakeInputWrap>\n        <InputRaw\n          name={`${name}-fakeInput`}\n          value={undefined}\n          type=\"text\"\n          onChange={() => null}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          tabIndex={-1}\n          autoComplete=\"off\"\n          hasError={hasError}\n        />\n      </StyledFakeInputWrap>\n\n      {isOpen && (\n        <StyledDropdown\n          dropdownPosition={verticalPosition}\n          ref={dropDownRef}\n          // this is to prevent known bug of Chrome when element\n          // loses focus on click on the scrollbar\n          onMouseDown={(e) => e.preventDefault()}\n        >\n          <VirtualScrollList\n            maxHeight={maxHeight}\n            itemHeight={36}\n            itemAmount={filteredOptions.length}\n            emptyState={() => (\n              <Box space=\"xs\">\n                <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n              </Box>\n            )}\n            itemInView={preselectedIndex}\n            itemTemplate={(index: number) => {\n              const option: SelectOption = filteredOptions[index];\n              const isActive = value.includes(option.value);\n\n              return (\n                <CustomOption\n                  key={option.value}\n                  active={isActive}\n                  preSelected={preselectedIndex === index}\n                  onMouseDown={() => {\n                    forceChangeFakeSelect(option);\n                  }}\n                >\n                  <CheckboxRaw\n                    name=\"\"\n                    checked={isActive}\n                    size=\"s\"\n                    label={option.label}\n                    onChange={() => null}\n                  />\n                </CustomOption>\n              );\n            }}\n          />\n        </StyledDropdown>\n      )}\n      <select\n        ref={selectRef}\n        name={name}\n        onChange={handleOnChange}\n        data-e2e-test-id=\"multiSelect\"\n        aria-hidden\n        hidden\n        value={value}\n        multiple\n      >\n        {selectedOptions.map((o) => (\n          <option key={o.value} value={o.value}>\n            {o.label}\n          </option>\n        ))}\n      </select>\n    </StyledContainer>\n  );\n}\n"],"names":[],"mappings":"AA+DsB"} */"),CustomOption=styled(StyledOption,{target:"ex2053y2",label:"CustomOption"})(({theme})=>({paddingTop:0,paddingBottom:0,"& > div":{paddingTop:theme.variables.size.spacing.xs,paddingBottom:theme.variables.size.spacing.xs},"& > div *":{marginTop:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/MultiSelect.tsx","sources":["src/components/Form/Select/MultiSelect.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n  useReducer,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport type { CommonSelectProps, SelectOption } from \"./Select\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { InputRaw } from \"../Input/Input\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Tag } from \"../../Tag/Tag\";\nimport {\n  StyledOption,\n  StyledContainer,\n  StyledDropdown,\n  StyledFakeInputWrap,\n} from \"./StyledSelectComponents\";\nimport { MultiSelectReducer } from \"./MultiSelectReducer\";\n\nexport type ActionData = {\n  action: \"select-option\" | \"deselect-option\" | \"clear\";\n  name: string;\n  data: SelectOption[];\n};\n\nexport type UniqueMultiSelectProps = {\n  value: string[];\n  onChange?: (\n    value: string[],\n    actionData: ActionData,\n    event: React.ChangeEvent<HTMLSelectElement>\n  ) => void;\n  formatSelectedOptionsLabel?: (count?: number) => string;\n};\n\ntype MultiSelectProps = CommonSelectProps &\n  Pick<FormFieldProps, \"label\"> &\n  UniqueMultiSelectProps;\n\nconst StyledInputWrap = styled.div<{ hasValue: boolean }>(\n  ({ theme, hasValue }) => ({\n    zIndex: 1,\n    display: \"flex\",\n    alignItems: \"center\",\n    position: \"relative\",\n\n    \"& input\": {\n      ...(hasValue && {\n        paddingLeft: theme.variables.size.spacing.xxs,\n      }),\n    },\n  })\n);\n\nconst StyledTagWrap = styled.div(({ theme }) => ({\n  marginLeft: theme.variables.size.spacing.xs,\n  order: -1, // it's needed to be 2-nd in html because click on label affects 1-st element (input)\n}));\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst getSelectedOptions = (\n  options: SelectOption[],\n  value: string[]\n): SelectOption[] => {\n  const mapFromValue = new Set(value);\n  return options.filter((opt) => mapFromValue.has(opt.value));\n};\n\nexport function MultiSelect({\n  options,\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod,\n  onChange,\n  onBlur,\n  onFocus,\n  maxHeight,\n  formatSelectedOptionsLabel = (count) => `${count} selected`,\n  autoComplete,\n  disabled,\n  label,\n}: MultiSelectProps): React.ReactElement {\n  const selectRef = useRef<HTMLSelectElement>(null);\n  const initialSelectedOptions = useMemo(\n    () => getSelectedOptions(options, value),\n    [options, value]\n  );\n  const [\n    {\n      selectedOptions,\n      actionName,\n      changedOptions,\n      isOpen,\n      preselectedIndex,\n      innerValue,\n    },\n    dispatch,\n  ] = useReducer(MultiSelectReducer, {\n    selectedOptions: initialSelectedOptions,\n    isOpen: false,\n    preselectedIndex: -1,\n    innerValue: \"\",\n  });\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const handleOnChange = useCallback(\n    (event) => {\n      onChange(\n        selectedOptions.map((opt) => opt.value),\n        {\n          name,\n          action: actionName,\n          data: changedOptions,\n        },\n        event\n      );\n    },\n    [selectedOptions, actionName, changedOptions, name, onChange]\n  );\n\n  const forceChangeFakeSelect = useCallback(\n    (option: SelectOption) => {\n      dispatch({ type: \"onChange\", option });\n    },\n    [dispatch]\n  );\n\n  const [verticalPosition] = useAutoPosition(\n    innerInputRef,\n    dropDownRef,\n    isOpen\n  );\n\n  useEffect(() => {\n    dispatch({\n      type: \"updateSelectedOptions\",\n      selectedOptions: initialSelectedOptions,\n    });\n  }, [initialSelectedOptions]);\n\n  const closeDropdown = useCallback(() => {\n    dispatch({ type: \"close\" });\n    dispatch({ type: \"setInnerValue\", value: \"\" });\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n    onBlur();\n  }, [onBlur, dispatch]);\n\n  useEffect(() => {\n    if (selectRef.current) {\n      selectRef.current.dispatchEvent(new Event(\"change\", { bubbles: true }));\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [changedOptions, selectRef.current]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  useEffect(() => {\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n        }\n      },\n      ArrowUp: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.max(preselectedIndex - 1, 0),\n        });\n      },\n      ArrowDown: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.min(preselectedIndex + 1, filteredOptions.length - 1),\n        });\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        dispatch({ type: \"open\" });\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <StyledContainer\n      onBlur={() => {\n        closeDropdown();\n      }}\n    >\n      <StyledInputWrap hasValue={value.length > 0}>\n        <InputRaw\n          areaLabel={label}\n          name={`${name}-innerInput`}\n          value={innerValue}\n          privateProps={{ isTransparent: true, hideOutline: true }}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          disabled={disabled}\n          placeholder={value.length === 0 ? placeholder : undefined}\n          onFocus={() => {\n            dispatch({ type: \"open\" });\n            onFocus();\n          }}\n          onClick={() => {\n            dispatch({ type: \"open\" });\n          }}\n          onChange={(e) => {\n            if (e.currentTarget.value) {\n              dispatch({ type: \"open\" });\n            }\n            dispatch({\n              type: \"setInnerValue\",\n              value: e.currentTarget.value,\n            });\n          }}\n          ref={innerInputRef}\n          autoComplete={autoComplete}\n        />\n        {initialSelectedOptions.length > 0 && (\n          <StyledTagWrap>\n            <Tag\n              label={formatSelectedOptionsLabel(initialSelectedOptions.length)}\n              isRemovable\n              onClear={() => {\n                dispatch({ type: \"clear\" });\n              }}\n            />\n          </StyledTagWrap>\n        )}\n      </StyledInputWrap>\n      <StyledFakeInputWrap>\n        <InputRaw\n          name={`${name}-fakeInput`}\n          value={undefined}\n          type=\"text\"\n          onChange={() => null}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          tabIndex={-1}\n          autoComplete=\"off\"\n          hasError={hasError}\n        />\n      </StyledFakeInputWrap>\n\n      {isOpen && (\n        <StyledDropdown\n          dropdownPosition={verticalPosition}\n          ref={dropDownRef}\n          // this is to prevent known bug of Chrome when element\n          // loses focus on click on the scrollbar\n          onMouseDown={(e) => e.preventDefault()}\n        >\n          <VirtualScrollList\n            maxHeight={maxHeight}\n            itemHeight={36}\n            itemAmount={filteredOptions.length}\n            emptyState={() => (\n              <Box space=\"xs\">\n                <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n              </Box>\n            )}\n            itemInView={preselectedIndex}\n            itemTemplate={(index: number) => {\n              const option: SelectOption = filteredOptions[index];\n              const isActive = value.includes(option.value);\n\n              return (\n                <CustomOption\n                  key={option.value}\n                  active={isActive}\n                  preSelected={preselectedIndex === index}\n                  onMouseDown={() => {\n                    forceChangeFakeSelect(option);\n                  }}\n                >\n                  <CheckboxRaw\n                    name=\"\"\n                    checked={isActive}\n                    size=\"s\"\n                    label={option.label}\n                    onChange={() => null}\n                  />\n                </CustomOption>\n              );\n            }}\n          />\n        </StyledDropdown>\n      )}\n      <select\n        ref={selectRef}\n        name={name}\n        onChange={handleOnChange}\n        data-e2e-test-id=\"multiSelect\"\n        aria-hidden\n        hidden\n        value={value}\n        multiple\n      >\n        {selectedOptions.map((o) => (\n          <option key={o.value} value={o.value}>\n            {o.label}\n          </option>\n        ))}\n      </select>\n    </StyledContainer>\n  );\n}\n"],"names":[],"mappings":"AAoEqB"} */"),getSelectedOptions=(options,value)=>{let mapFromValue=new Set(value);return options.filter(opt=>mapFromValue.has(opt.value))};export function MultiSelect({options,name,value,placeholder,emptyStateMessage,hasError,filterMethod,onChange,onBlur,onFocus,maxHeight,formatSelectedOptionsLabel=count=>`${count} selected`,autoComplete,disabled,label}){let selectRef=useRef(null),initialSelectedOptions=useMemo(()=>getSelectedOptions(options,value),[options,value]),[{selectedOptions,actionName,changedOptions,isOpen,preselectedIndex,innerValue},dispatch]=useReducer(MultiSelectReducer,{selectedOptions:initialSelectedOptions,isOpen:!1,preselectedIndex:-1,innerValue:""}),dropDownRef=useRef(null),innerInputRef=useRef(null),handleOnChange=useCallback(event=>{onChange(selectedOptions.map(opt=>opt.value),{name,action:actionName,data:changedOptions},event)},[selectedOptions,actionName,changedOptions,name,onChange]),forceChangeFakeSelect=useCallback(option=>{dispatch({type:"onChange",option})},[dispatch]),[verticalPosition]=useAutoPosition(innerInputRef,dropDownRef,isOpen);useEffect(()=>{dispatch({type:"updateSelectedOptions",selectedOptions:initialSelectedOptions})},[initialSelectedOptions]);let closeDropdown=useCallback(()=>{dispatch({type:"close"}),dispatch({type:"setInnerValue",value:""}),dispatch({type:"setPreselectedIndex",index:-1}),onBlur()},[onBlur,dispatch]);useEffect(()=>{selectRef.current&&selectRef.current.dispatchEvent(new Event("change",{bubbles:!0}))},[changedOptions,selectRef.current]);let filteredOptions=useMemo(()=>innerValue?options.filter(option=>filterMethod(option,innerValue)):options,[options,filterMethod,innerValue]);return useEffect(()=>{dispatch({type:"setPreselectedIndex",index:-1})},[filteredOptions]),useKeyboard({Escape:()=>closeDropdown(),Enter:()=>{filteredOptions[preselectedIndex]&&forceChangeFakeSelect(filteredOptions[preselectedIndex])},ArrowUp:()=>{dispatch({type:"setPreselectedIndex",index:Math.max(preselectedIndex-1,0)})},ArrowDown:()=>{dispatch({type:"setPreselectedIndex",index:Math.min(preselectedIndex+1,filteredOptions.length-1)})}},innerInputRef,isOpen&&!disabled),useKeyboard({"ArrowUp ArrowDown":()=>{dispatch({type:"open"})}},innerInputRef,!isOpen&&!disabled),React.createElement(StyledContainer,{onBlur:()=>{closeDropdown()}},React.createElement(StyledInputWrap,{hasValue:value.length>0},React.createElement(InputRaw,{areaLabel:label,name:`${name}-innerInput`,value:innerValue,privateProps:{isTransparent:!0,hideOutline:!0},icon:isOpen?"chevron-up":"chevron-down",disabled:disabled,placeholder:0===value.length?placeholder:void 0,onFocus:()=>{dispatch({type:"open"}),onFocus()},onClick:()=>{dispatch({type:"open"})},onChange:e=>{e.currentTarget.value&&dispatch({type:"open"}),dispatch({type:"setInnerValue",value:e.currentTarget.value})},ref:innerInputRef,autoComplete:autoComplete}),initialSelectedOptions.length>0&&React.createElement(StyledTagWrap,null,React.createElement(Tag,{label:formatSelectedOptionsLabel(initialSelectedOptions.length),isRemovable:!0,onClear:()=>{dispatch({type:"clear"})}}))),React.createElement(StyledFakeInputWrap,null,React.createElement(InputRaw,{name:`${name}-fakeInput`,value:void 0,type:"text",onChange:()=>null,icon:isOpen?"chevron-up":"chevron-down",tabIndex:-1,autoComplete:"off",hasError:hasError})),isOpen&&React.createElement(StyledDropdown,{dropdownPosition:verticalPosition,ref:dropDownRef,onMouseDown:e=>e.preventDefault()},React.createElement(VirtualScrollList,{maxHeight:maxHeight,itemHeight:36,itemAmount:filteredOptions.length,emptyState:()=>React.createElement(Box,{space:"xs"},React.createElement(Text,null,emptyStateMessage||"\uD83E\uDD37\uD83C\uDFFB♀️")),itemInView:preselectedIndex,itemTemplate:index=>{let option=filteredOptions[index],isActive=value.includes(option.value);return React.createElement(CustomOption,{key:option.value,active:isActive,preSelected:preselectedIndex===index,onMouseDown:()=>{forceChangeFakeSelect(option)}},React.createElement(CheckboxRaw,{name:"",checked:isActive,size:"s",label:option.label,onChange:()=>null}))}})),React.createElement("select",{ref:selectRef,name:name,onChange:handleOnChange,"data-e2e-test-id":"multiSelect","aria-hidden":!0,hidden:!0,value:value,multiple:!0},selectedOptions.map(o=>React.createElement("option",{key:o.value,value:o.value},o.label))))}
|
|
1
|
+
import React,{useRef,useMemo,useEffect,useCallback,useReducer}from"react";import styled from"@emotion/styled";import{useKeyboard}from"../../../shared/useKeyboard";import{useAutoPosition}from"../../../shared/useAutoPosition";import{VirtualScrollList}from"../../VirtualScrollList/VirtualScrollList";import{InputRaw}from"../Input/Input";import{CheckboxRaw}from"../Checkbox/Checkbox";import{Text}from"../../Typography/Text/Text";import{Box}from"../../Box/Box";import{Tag}from"../../Tag/Tag";import{StyledOption,StyledContainer,StyledDropdown,StyledFakeInputWrap}from"./StyledSelectComponents";import{MultiSelectReducer}from"./MultiSelectReducer";let StyledInputWrap=styled("div",{target:"e1gnddyy0",label:"StyledInputWrap"})(({theme,hasValue})=>({zIndex:1,display:"flex",alignItems:"center",position:"relative","& input":{...hasValue&&{paddingLeft:theme.variables.size.spacing.xxs}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/MultiSelect.tsx","sources":["src/components/Form/Select/MultiSelect.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n  useReducer,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport type { CommonSelectProps, SelectOption } from \"./Select\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { InputRaw } from \"../Input/Input\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Tag } from \"../../Tag/Tag\";\nimport {\n  StyledOption,\n  StyledContainer,\n  StyledDropdown,\n  StyledFakeInputWrap,\n} from \"./StyledSelectComponents\";\nimport { MultiSelectReducer } from \"./MultiSelectReducer\";\n\nexport type ActionData = {\n  action: \"select-option\" | \"deselect-option\" | \"clear\";\n  name: string;\n  data: SelectOption[];\n};\n\nexport type UniqueMultiSelectProps = {\n  value: string[];\n  onChange?: (\n    value: string[],\n    actionData: ActionData,\n    event: React.ChangeEvent<HTMLSelectElement>\n  ) => void;\n  formatSelectedOptionsLabel?: (count?: number) => string;\n};\n\ntype MultiSelectProps = CommonSelectProps &\n  Pick<FormFieldProps, \"label\"> &\n  UniqueMultiSelectProps;\n\nconst StyledInputWrap = styled.div<{ hasValue: boolean }>(\n  ({ theme, hasValue }) => ({\n    zIndex: 1,\n    display: \"flex\",\n    alignItems: \"center\",\n    position: \"relative\",\n\n    \"& input\": {\n      ...(hasValue && {\n        paddingLeft: theme.variables.size.spacing.xxs,\n      }),\n    },\n  })\n);\n\nconst StyledTagWrap = styled.div(({ theme }) => ({\n  marginLeft: theme.variables.size.spacing.xs,\n  order: -1, // it's needed to be 2-nd in html because click on label affects 1-st element (input)\n}));\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst getSelectedOptions = (\n  options: SelectOption[],\n  value: string[]\n): SelectOption[] => {\n  const mapFromValue = new Set(value);\n  return options.filter((opt) => mapFromValue.has(opt.value));\n};\n\nexport function MultiSelect({\n  options,\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod,\n  onChange,\n  onBlur,\n  onFocus,\n  maxHeight,\n  formatSelectedOptionsLabel = (count) => `${count} selected`,\n  autoComplete,\n  disabled,\n  label,\n}: MultiSelectProps): React.ReactElement {\n  const selectRef = useRef<HTMLSelectElement>(null);\n  const initialSelectedOptions = useMemo(\n    () => getSelectedOptions(options, value),\n    [options, value]\n  );\n  const [\n    {\n      selectedOptions,\n      actionName,\n      changedOptions,\n      isOpen,\n      preselectedIndex,\n      innerValue,\n    },\n    dispatch,\n  ] = useReducer(MultiSelectReducer, {\n    selectedOptions: initialSelectedOptions,\n    isOpen: false,\n    preselectedIndex: -1,\n    innerValue: \"\",\n  });\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const handleOnChange = useCallback(\n    (event) => {\n      onChange(\n        selectedOptions.map((opt) => opt.value),\n        {\n          name,\n          action: actionName,\n          data: changedOptions,\n        },\n        event\n      );\n    },\n    [selectedOptions, actionName, changedOptions, name, onChange]\n  );\n\n  const forceChangeFakeSelect = useCallback(\n    (option: SelectOption) => {\n      dispatch({ type: \"onChange\", option });\n    },\n    [dispatch]\n  );\n\n  const [verticalPosition] = useAutoPosition({\n    anchorRef: innerInputRef,\n    contentRef: dropDownRef,\n    isActive: isOpen,\n  });\n\n  useEffect(() => {\n    dispatch({\n      type: \"updateSelectedOptions\",\n      selectedOptions: initialSelectedOptions,\n    });\n  }, [initialSelectedOptions]);\n\n  const closeDropdown = useCallback(() => {\n    dispatch({ type: \"close\" });\n    dispatch({ type: \"setInnerValue\", value: \"\" });\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n    onBlur();\n  }, [onBlur, dispatch]);\n\n  useEffect(() => {\n    if (selectRef.current) {\n      selectRef.current.dispatchEvent(new Event(\"change\", { bubbles: true }));\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [changedOptions, selectRef.current]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  useEffect(() => {\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n        }\n      },\n      ArrowUp: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.max(preselectedIndex - 1, 0),\n        });\n      },\n      ArrowDown: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.min(preselectedIndex + 1, filteredOptions.length - 1),\n        });\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        dispatch({ type: \"open\" });\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <StyledContainer\n      onBlur={() => {\n        closeDropdown();\n      }}\n    >\n      <StyledInputWrap hasValue={value.length > 0}>\n        <InputRaw\n          areaLabel={label}\n          name={`${name}-innerInput`}\n          value={innerValue}\n          privateProps={{ isTransparent: true, hideOutline: true }}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          disabled={disabled}\n          placeholder={value.length === 0 ? placeholder : undefined}\n          onFocus={() => {\n            dispatch({ type: \"open\" });\n            onFocus();\n          }}\n          onClick={() => {\n            dispatch({ type: \"open\" });\n          }}\n          onChange={(e) => {\n            if (e.currentTarget.value) {\n              dispatch({ type: \"open\" });\n            }\n            dispatch({\n              type: \"setInnerValue\",\n              value: e.currentTarget.value,\n            });\n          }}\n          ref={innerInputRef}\n          autoComplete={autoComplete}\n        />\n        {initialSelectedOptions.length > 0 && (\n          <StyledTagWrap>\n            <Tag\n              label={formatSelectedOptionsLabel(initialSelectedOptions.length)}\n              isRemovable\n              onClear={() => {\n                dispatch({ type: \"clear\" });\n              }}\n            />\n          </StyledTagWrap>\n        )}\n      </StyledInputWrap>\n      <StyledFakeInputWrap>\n        <InputRaw\n          name={`${name}-fakeInput`}\n          value={undefined}\n          type=\"text\"\n          onChange={() => null}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          tabIndex={-1}\n          autoComplete=\"off\"\n          hasError={hasError}\n        />\n      </StyledFakeInputWrap>\n\n      {isOpen && (\n        <StyledDropdown\n          dropdownPosition={verticalPosition}\n          ref={dropDownRef}\n          // this is to prevent known bug of Chrome when element\n          // loses focus on click on the scrollbar\n          onMouseDown={(e) => e.preventDefault()}\n        >\n          <VirtualScrollList\n            maxHeight={maxHeight}\n            itemHeight={36}\n            itemAmount={filteredOptions.length}\n            emptyState={() => (\n              <Box space=\"xs\">\n                <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n              </Box>\n            )}\n            itemInView={preselectedIndex}\n            itemTemplate={(index: number) => {\n              const option: SelectOption = filteredOptions[index];\n              const isActive = value.includes(option.value);\n\n              return (\n                <CustomOption\n                  key={option.value}\n                  active={isActive}\n                  preSelected={preselectedIndex === index}\n                  onMouseDown={() => {\n                    forceChangeFakeSelect(option);\n                  }}\n                >\n                  <CheckboxRaw\n                    name=\"\"\n                    checked={isActive}\n                    size=\"s\"\n                    label={option.label}\n                    onChange={() => null}\n                  />\n                </CustomOption>\n              );\n            }}\n          />\n        </StyledDropdown>\n      )}\n      <select\n        ref={selectRef}\n        name={name}\n        onChange={handleOnChange}\n        data-e2e-test-id=\"multiSelect\"\n        aria-hidden\n        hidden\n        value={value}\n        multiple\n      >\n        {selectedOptions.map((o) => (\n          <option key={o.value} value={o.value}>\n            {o.label}\n          </option>\n        ))}\n      </select>\n    </StyledContainer>\n  );\n}\n"],"names":[],"mappings":"AAgDwB"} */"),StyledTagWrap=styled("div",{target:"e1gnddyy1",label:"StyledTagWrap"})(({theme})=>({marginLeft:theme.variables.size.spacing.xs,order:-1}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/MultiSelect.tsx","sources":["src/components/Form/Select/MultiSelect.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n  useReducer,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport type { CommonSelectProps, SelectOption } from \"./Select\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { InputRaw } from \"../Input/Input\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Tag } from \"../../Tag/Tag\";\nimport {\n  StyledOption,\n  StyledContainer,\n  StyledDropdown,\n  StyledFakeInputWrap,\n} from \"./StyledSelectComponents\";\nimport { MultiSelectReducer } from \"./MultiSelectReducer\";\n\nexport type ActionData = {\n  action: \"select-option\" | \"deselect-option\" | \"clear\";\n  name: string;\n  data: SelectOption[];\n};\n\nexport type UniqueMultiSelectProps = {\n  value: string[];\n  onChange?: (\n    value: string[],\n    actionData: ActionData,\n    event: React.ChangeEvent<HTMLSelectElement>\n  ) => void;\n  formatSelectedOptionsLabel?: (count?: number) => string;\n};\n\ntype MultiSelectProps = CommonSelectProps &\n  Pick<FormFieldProps, \"label\"> &\n  UniqueMultiSelectProps;\n\nconst StyledInputWrap = styled.div<{ hasValue: boolean }>(\n  ({ theme, hasValue }) => ({\n    zIndex: 1,\n    display: \"flex\",\n    alignItems: \"center\",\n    position: \"relative\",\n\n    \"& input\": {\n      ...(hasValue && {\n        paddingLeft: theme.variables.size.spacing.xxs,\n      }),\n    },\n  })\n);\n\nconst StyledTagWrap = styled.div(({ theme }) => ({\n  marginLeft: theme.variables.size.spacing.xs,\n  order: -1, // it's needed to be 2-nd in html because click on label affects 1-st element (input)\n}));\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst getSelectedOptions = (\n  options: SelectOption[],\n  value: string[]\n): SelectOption[] => {\n  const mapFromValue = new Set(value);\n  return options.filter((opt) => mapFromValue.has(opt.value));\n};\n\nexport function MultiSelect({\n  options,\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod,\n  onChange,\n  onBlur,\n  onFocus,\n  maxHeight,\n  formatSelectedOptionsLabel = (count) => `${count} selected`,\n  autoComplete,\n  disabled,\n  label,\n}: MultiSelectProps): React.ReactElement {\n  const selectRef = useRef<HTMLSelectElement>(null);\n  const initialSelectedOptions = useMemo(\n    () => getSelectedOptions(options, value),\n    [options, value]\n  );\n  const [\n    {\n      selectedOptions,\n      actionName,\n      changedOptions,\n      isOpen,\n      preselectedIndex,\n      innerValue,\n    },\n    dispatch,\n  ] = useReducer(MultiSelectReducer, {\n    selectedOptions: initialSelectedOptions,\n    isOpen: false,\n    preselectedIndex: -1,\n    innerValue: \"\",\n  });\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const handleOnChange = useCallback(\n    (event) => {\n      onChange(\n        selectedOptions.map((opt) => opt.value),\n        {\n          name,\n          action: actionName,\n          data: changedOptions,\n        },\n        event\n      );\n    },\n    [selectedOptions, actionName, changedOptions, name, onChange]\n  );\n\n  const forceChangeFakeSelect = useCallback(\n    (option: SelectOption) => {\n      dispatch({ type: \"onChange\", option });\n    },\n    [dispatch]\n  );\n\n  const [verticalPosition] = useAutoPosition({\n    anchorRef: innerInputRef,\n    contentRef: dropDownRef,\n    isActive: isOpen,\n  });\n\n  useEffect(() => {\n    dispatch({\n      type: \"updateSelectedOptions\",\n      selectedOptions: initialSelectedOptions,\n    });\n  }, [initialSelectedOptions]);\n\n  const closeDropdown = useCallback(() => {\n    dispatch({ type: \"close\" });\n    dispatch({ type: \"setInnerValue\", value: \"\" });\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n    onBlur();\n  }, [onBlur, dispatch]);\n\n  useEffect(() => {\n    if (selectRef.current) {\n      selectRef.current.dispatchEvent(new Event(\"change\", { bubbles: true }));\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [changedOptions, selectRef.current]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  useEffect(() => {\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n        }\n      },\n      ArrowUp: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.max(preselectedIndex - 1, 0),\n        });\n      },\n      ArrowDown: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.min(preselectedIndex + 1, filteredOptions.length - 1),\n        });\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        dispatch({ type: \"open\" });\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <StyledContainer\n      onBlur={() => {\n        closeDropdown();\n      }}\n    >\n      <StyledInputWrap hasValue={value.length > 0}>\n        <InputRaw\n          areaLabel={label}\n          name={`${name}-innerInput`}\n          value={innerValue}\n          privateProps={{ isTransparent: true, hideOutline: true }}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          disabled={disabled}\n          placeholder={value.length === 0 ? placeholder : undefined}\n          onFocus={() => {\n            dispatch({ type: \"open\" });\n            onFocus();\n          }}\n          onClick={() => {\n            dispatch({ type: \"open\" });\n          }}\n          onChange={(e) => {\n            if (e.currentTarget.value) {\n              dispatch({ type: \"open\" });\n            }\n            dispatch({\n              type: \"setInnerValue\",\n              value: e.currentTarget.value,\n            });\n          }}\n          ref={innerInputRef}\n          autoComplete={autoComplete}\n        />\n        {initialSelectedOptions.length > 0 && (\n          <StyledTagWrap>\n            <Tag\n              label={formatSelectedOptionsLabel(initialSelectedOptions.length)}\n              isRemovable\n              onClear={() => {\n                dispatch({ type: \"clear\" });\n              }}\n            />\n          </StyledTagWrap>\n        )}\n      </StyledInputWrap>\n      <StyledFakeInputWrap>\n        <InputRaw\n          name={`${name}-fakeInput`}\n          value={undefined}\n          type=\"text\"\n          onChange={() => null}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          tabIndex={-1}\n          autoComplete=\"off\"\n          hasError={hasError}\n        />\n      </StyledFakeInputWrap>\n\n      {isOpen && (\n        <StyledDropdown\n          dropdownPosition={verticalPosition}\n          ref={dropDownRef}\n          // this is to prevent known bug of Chrome when element\n          // loses focus on click on the scrollbar\n          onMouseDown={(e) => e.preventDefault()}\n        >\n          <VirtualScrollList\n            maxHeight={maxHeight}\n            itemHeight={36}\n            itemAmount={filteredOptions.length}\n            emptyState={() => (\n              <Box space=\"xs\">\n                <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n              </Box>\n            )}\n            itemInView={preselectedIndex}\n            itemTemplate={(index: number) => {\n              const option: SelectOption = filteredOptions[index];\n              const isActive = value.includes(option.value);\n\n              return (\n                <CustomOption\n                  key={option.value}\n                  active={isActive}\n                  preSelected={preselectedIndex === index}\n                  onMouseDown={() => {\n                    forceChangeFakeSelect(option);\n                  }}\n                >\n                  <CheckboxRaw\n                    name=\"\"\n                    checked={isActive}\n                    size=\"s\"\n                    label={option.label}\n                    onChange={() => null}\n                  />\n                </CustomOption>\n              );\n            }}\n          />\n        </StyledDropdown>\n      )}\n      <select\n        ref={selectRef}\n        name={name}\n        onChange={handleOnChange}\n        data-e2e-test-id=\"multiSelect\"\n        aria-hidden\n        hidden\n        value={value}\n        multiple\n      >\n        {selectedOptions.map((o) => (\n          <option key={o.value} value={o.value}>\n            {o.label}\n          </option>\n        ))}\n      </select>\n    </StyledContainer>\n  );\n}\n"],"names":[],"mappings":"AA+DsB"} */"),CustomOption=styled(StyledOption,{target:"e1gnddyy2",label:"CustomOption"})(({theme})=>({paddingTop:0,paddingBottom:0,"& > div":{paddingTop:theme.variables.size.spacing.xs,paddingBottom:theme.variables.size.spacing.xs},"& > div *":{marginTop:0}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Form/Select/MultiSelect.tsx","sources":["src/components/Form/Select/MultiSelect.tsx"],"sourcesContent":["/* eslint-disable react/jsx-props-no-spreading */\n\nimport React, {\n  useRef,\n  useMemo,\n  useEffect,\n  useCallback,\n  useReducer,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { FormFieldProps } from \"../FormField/FormField\";\nimport type { CommonSelectProps, SelectOption } from \"./Select\";\nimport { useKeyboard } from \"../../../shared/useKeyboard\";\nimport { useAutoPosition } from \"../../../shared/useAutoPosition\";\nimport { VirtualScrollList } from \"../../VirtualScrollList/VirtualScrollList\";\nimport { InputRaw } from \"../Input/Input\";\nimport { CheckboxRaw } from \"../Checkbox/Checkbox\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { Box } from \"../../Box/Box\";\nimport { Tag } from \"../../Tag/Tag\";\nimport {\n  StyledOption,\n  StyledContainer,\n  StyledDropdown,\n  StyledFakeInputWrap,\n} from \"./StyledSelectComponents\";\nimport { MultiSelectReducer } from \"./MultiSelectReducer\";\n\nexport type ActionData = {\n  action: \"select-option\" | \"deselect-option\" | \"clear\";\n  name: string;\n  data: SelectOption[];\n};\n\nexport type UniqueMultiSelectProps = {\n  value: string[];\n  onChange?: (\n    value: string[],\n    actionData: ActionData,\n    event: React.ChangeEvent<HTMLSelectElement>\n  ) => void;\n  formatSelectedOptionsLabel?: (count?: number) => string;\n};\n\ntype MultiSelectProps = CommonSelectProps &\n  Pick<FormFieldProps, \"label\"> &\n  UniqueMultiSelectProps;\n\nconst StyledInputWrap = styled.div<{ hasValue: boolean }>(\n  ({ theme, hasValue }) => ({\n    zIndex: 1,\n    display: \"flex\",\n    alignItems: \"center\",\n    position: \"relative\",\n\n    \"& input\": {\n      ...(hasValue && {\n        paddingLeft: theme.variables.size.spacing.xxs,\n      }),\n    },\n  })\n);\n\nconst StyledTagWrap = styled.div(({ theme }) => ({\n  marginLeft: theme.variables.size.spacing.xs,\n  order: -1, // it's needed to be 2-nd in html because click on label affects 1-st element (input)\n}));\n\nconst CustomOption = styled(StyledOption)(({ theme }) => ({\n  // this is needed for using padding of checkboxes in order to use its hover state\n  paddingTop: 0,\n  paddingBottom: 0,\n  \"& > div\": {\n    paddingTop: theme.variables.size.spacing.xs,\n    paddingBottom: theme.variables.size.spacing.xs,\n  },\n  \"& > div *\": {\n    marginTop: 0,\n  },\n}));\n\nconst getSelectedOptions = (\n  options: SelectOption[],\n  value: string[]\n): SelectOption[] => {\n  const mapFromValue = new Set(value);\n  return options.filter((opt) => mapFromValue.has(opt.value));\n};\n\nexport function MultiSelect({\n  options,\n  name,\n  value,\n  placeholder,\n  emptyStateMessage,\n  hasError,\n  filterMethod,\n  onChange,\n  onBlur,\n  onFocus,\n  maxHeight,\n  formatSelectedOptionsLabel = (count) => `${count} selected`,\n  autoComplete,\n  disabled,\n  label,\n}: MultiSelectProps): React.ReactElement {\n  const selectRef = useRef<HTMLSelectElement>(null);\n  const initialSelectedOptions = useMemo(\n    () => getSelectedOptions(options, value),\n    [options, value]\n  );\n  const [\n    {\n      selectedOptions,\n      actionName,\n      changedOptions,\n      isOpen,\n      preselectedIndex,\n      innerValue,\n    },\n    dispatch,\n  ] = useReducer(MultiSelectReducer, {\n    selectedOptions: initialSelectedOptions,\n    isOpen: false,\n    preselectedIndex: -1,\n    innerValue: \"\",\n  });\n\n  const dropDownRef = useRef(null);\n  const innerInputRef = useRef(null);\n\n  const handleOnChange = useCallback(\n    (event) => {\n      onChange(\n        selectedOptions.map((opt) => opt.value),\n        {\n          name,\n          action: actionName,\n          data: changedOptions,\n        },\n        event\n      );\n    },\n    [selectedOptions, actionName, changedOptions, name, onChange]\n  );\n\n  const forceChangeFakeSelect = useCallback(\n    (option: SelectOption) => {\n      dispatch({ type: \"onChange\", option });\n    },\n    [dispatch]\n  );\n\n  const [verticalPosition] = useAutoPosition({\n    anchorRef: innerInputRef,\n    contentRef: dropDownRef,\n    isActive: isOpen,\n  });\n\n  useEffect(() => {\n    dispatch({\n      type: \"updateSelectedOptions\",\n      selectedOptions: initialSelectedOptions,\n    });\n  }, [initialSelectedOptions]);\n\n  const closeDropdown = useCallback(() => {\n    dispatch({ type: \"close\" });\n    dispatch({ type: \"setInnerValue\", value: \"\" });\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n    onBlur();\n  }, [onBlur, dispatch]);\n\n  useEffect(() => {\n    if (selectRef.current) {\n      selectRef.current.dispatchEvent(new Event(\"change\", { bubbles: true }));\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [changedOptions, selectRef.current]);\n\n  const filteredOptions = useMemo(() => {\n    if (!innerValue) {\n      return options;\n    }\n\n    return options.filter((option) => filterMethod(option, innerValue));\n  }, [options, filterMethod, innerValue]);\n\n  useEffect(() => {\n    dispatch({ type: \"setPreselectedIndex\", index: -1 });\n  }, [filteredOptions]);\n\n  useKeyboard(\n    {\n      Escape: () => closeDropdown(),\n      Enter: () => {\n        if (filteredOptions[preselectedIndex]) {\n          forceChangeFakeSelect(filteredOptions[preselectedIndex]);\n        }\n      },\n      ArrowUp: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.max(preselectedIndex - 1, 0),\n        });\n      },\n      ArrowDown: () => {\n        dispatch({\n          type: \"setPreselectedIndex\",\n          index: Math.min(preselectedIndex + 1, filteredOptions.length - 1),\n        });\n      },\n    },\n    innerInputRef,\n    isOpen && !disabled\n  );\n\n  useKeyboard(\n    {\n      \"ArrowUp ArrowDown\": () => {\n        dispatch({ type: \"open\" });\n      },\n    },\n    innerInputRef,\n    !isOpen && !disabled\n  );\n\n  return (\n    <StyledContainer\n      onBlur={() => {\n        closeDropdown();\n      }}\n    >\n      <StyledInputWrap hasValue={value.length > 0}>\n        <InputRaw\n          areaLabel={label}\n          name={`${name}-innerInput`}\n          value={innerValue}\n          privateProps={{ isTransparent: true, hideOutline: true }}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          disabled={disabled}\n          placeholder={value.length === 0 ? placeholder : undefined}\n          onFocus={() => {\n            dispatch({ type: \"open\" });\n            onFocus();\n          }}\n          onClick={() => {\n            dispatch({ type: \"open\" });\n          }}\n          onChange={(e) => {\n            if (e.currentTarget.value) {\n              dispatch({ type: \"open\" });\n            }\n            dispatch({\n              type: \"setInnerValue\",\n              value: e.currentTarget.value,\n            });\n          }}\n          ref={innerInputRef}\n          autoComplete={autoComplete}\n        />\n        {initialSelectedOptions.length > 0 && (\n          <StyledTagWrap>\n            <Tag\n              label={formatSelectedOptionsLabel(initialSelectedOptions.length)}\n              isRemovable\n              onClear={() => {\n                dispatch({ type: \"clear\" });\n              }}\n            />\n          </StyledTagWrap>\n        )}\n      </StyledInputWrap>\n      <StyledFakeInputWrap>\n        <InputRaw\n          name={`${name}-fakeInput`}\n          value={undefined}\n          type=\"text\"\n          onChange={() => null}\n          icon={isOpen ? \"chevron-up\" : \"chevron-down\"}\n          tabIndex={-1}\n          autoComplete=\"off\"\n          hasError={hasError}\n        />\n      </StyledFakeInputWrap>\n\n      {isOpen && (\n        <StyledDropdown\n          dropdownPosition={verticalPosition}\n          ref={dropDownRef}\n          // this is to prevent known bug of Chrome when element\n          // loses focus on click on the scrollbar\n          onMouseDown={(e) => e.preventDefault()}\n        >\n          <VirtualScrollList\n            maxHeight={maxHeight}\n            itemHeight={36}\n            itemAmount={filteredOptions.length}\n            emptyState={() => (\n              <Box space=\"xs\">\n                <Text>{emptyStateMessage || \"🤷🏻‍♀️\"}</Text>\n              </Box>\n            )}\n            itemInView={preselectedIndex}\n            itemTemplate={(index: number) => {\n              const option: SelectOption = filteredOptions[index];\n              const isActive = value.includes(option.value);\n\n              return (\n                <CustomOption\n                  key={option.value}\n                  active={isActive}\n                  preSelected={preselectedIndex === index}\n                  onMouseDown={() => {\n                    forceChangeFakeSelect(option);\n                  }}\n                >\n                  <CheckboxRaw\n                    name=\"\"\n                    checked={isActive}\n                    size=\"s\"\n                    label={option.label}\n                    onChange={() => null}\n                  />\n                </CustomOption>\n              );\n            }}\n          />\n        </StyledDropdown>\n      )}\n      <select\n        ref={selectRef}\n        name={name}\n        onChange={handleOnChange}\n        data-e2e-test-id=\"multiSelect\"\n        aria-hidden\n        hidden\n        value={value}\n        multiple\n      >\n        {selectedOptions.map((o) => (\n          <option key={o.value} value={o.value}>\n            {o.label}\n          </option>\n        ))}\n      </select>\n    </StyledContainer>\n  );\n}\n"],"names":[],"mappings":"AAoEqB"} */"),getSelectedOptions=(options,value)=>{let mapFromValue=new Set(value);return options.filter(opt=>mapFromValue.has(opt.value))};export function MultiSelect({options,name,value,placeholder,emptyStateMessage,hasError,filterMethod,onChange,onBlur,onFocus,maxHeight,formatSelectedOptionsLabel=count=>`${count} selected`,autoComplete,disabled,label}){let selectRef=useRef(null),initialSelectedOptions=useMemo(()=>getSelectedOptions(options,value),[options,value]),[{selectedOptions,actionName,changedOptions,isOpen,preselectedIndex,innerValue},dispatch]=useReducer(MultiSelectReducer,{selectedOptions:initialSelectedOptions,isOpen:!1,preselectedIndex:-1,innerValue:""}),dropDownRef=useRef(null),innerInputRef=useRef(null),handleOnChange=useCallback(event=>{onChange(selectedOptions.map(opt=>opt.value),{name,action:actionName,data:changedOptions},event)},[selectedOptions,actionName,changedOptions,name,onChange]),forceChangeFakeSelect=useCallback(option=>{dispatch({type:"onChange",option})},[dispatch]),[verticalPosition]=useAutoPosition({anchorRef:innerInputRef,contentRef:dropDownRef,isActive:isOpen});useEffect(()=>{dispatch({type:"updateSelectedOptions",selectedOptions:initialSelectedOptions})},[initialSelectedOptions]);let closeDropdown=useCallback(()=>{dispatch({type:"close"}),dispatch({type:"setInnerValue",value:""}),dispatch({type:"setPreselectedIndex",index:-1}),onBlur()},[onBlur,dispatch]);useEffect(()=>{selectRef.current&&selectRef.current.dispatchEvent(new Event("change",{bubbles:!0}))},[changedOptions,selectRef.current]);let filteredOptions=useMemo(()=>innerValue?options.filter(option=>filterMethod(option,innerValue)):options,[options,filterMethod,innerValue]);return useEffect(()=>{dispatch({type:"setPreselectedIndex",index:-1})},[filteredOptions]),useKeyboard({Escape:()=>closeDropdown(),Enter:()=>{filteredOptions[preselectedIndex]&&forceChangeFakeSelect(filteredOptions[preselectedIndex])},ArrowUp:()=>{dispatch({type:"setPreselectedIndex",index:Math.max(preselectedIndex-1,0)})},ArrowDown:()=>{dispatch({type:"setPreselectedIndex",index:Math.min(preselectedIndex+1,filteredOptions.length-1)})}},innerInputRef,isOpen&&!disabled),useKeyboard({"ArrowUp ArrowDown":()=>{dispatch({type:"open"})}},innerInputRef,!isOpen&&!disabled),React.createElement(StyledContainer,{onBlur:()=>{closeDropdown()}},React.createElement(StyledInputWrap,{hasValue:value.length>0},React.createElement(InputRaw,{areaLabel:label,name:`${name}-innerInput`,value:innerValue,privateProps:{isTransparent:!0,hideOutline:!0},icon:isOpen?"chevron-up":"chevron-down",disabled:disabled,placeholder:0===value.length?placeholder:void 0,onFocus:()=>{dispatch({type:"open"}),onFocus()},onClick:()=>{dispatch({type:"open"})},onChange:e=>{e.currentTarget.value&&dispatch({type:"open"}),dispatch({type:"setInnerValue",value:e.currentTarget.value})},ref:innerInputRef,autoComplete:autoComplete}),initialSelectedOptions.length>0&&React.createElement(StyledTagWrap,null,React.createElement(Tag,{label:formatSelectedOptionsLabel(initialSelectedOptions.length),isRemovable:!0,onClear:()=>{dispatch({type:"clear"})}}))),React.createElement(StyledFakeInputWrap,null,React.createElement(InputRaw,{name:`${name}-fakeInput`,value:void 0,type:"text",onChange:()=>null,icon:isOpen?"chevron-up":"chevron-down",tabIndex:-1,autoComplete:"off",hasError:hasError})),isOpen&&React.createElement(StyledDropdown,{dropdownPosition:verticalPosition,ref:dropDownRef,onMouseDown:e=>e.preventDefault()},React.createElement(VirtualScrollList,{maxHeight:maxHeight,itemHeight:36,itemAmount:filteredOptions.length,emptyState:()=>React.createElement(Box,{space:"xs"},React.createElement(Text,null,emptyStateMessage||"\uD83E\uDD37\uD83C\uDFFB♀️")),itemInView:preselectedIndex,itemTemplate:index=>{let option=filteredOptions[index],isActive=value.includes(option.value);return React.createElement(CustomOption,{key:option.value,active:isActive,preSelected:preselectedIndex===index,onMouseDown:()=>{forceChangeFakeSelect(option)}},React.createElement(CheckboxRaw,{name:"",checked:isActive,size:"s",label:option.label,onChange:()=>null}))}})),React.createElement("select",{ref:selectRef,name:name,onChange:handleOnChange,"data-e2e-test-id":"multiSelect","aria-hidden":!0,hidden:!0,value:value,multiple:!0},selectedOptions.map(o=>React.createElement("option",{key:o.value,value:o.value},o.label))))}
|
|
@@ -17,6 +17,9 @@ export type CommonSelectProps = {
|
|
|
17
17
|
onBlur?: () => void;
|
|
18
18
|
onFocus?: () => void;
|
|
19
19
|
maxHeight?: number;
|
|
20
|
+
privateProps?: {
|
|
21
|
+
dropdownPlacement?: "up" | "down";
|
|
22
|
+
};
|
|
20
23
|
multiple?: boolean;
|
|
21
24
|
autoComplete?: string;
|
|
22
25
|
onChange?: (selectedValues: SelectOption["value"][] & React.ChangeEvent<HTMLSelectElement>, actionData?: ActionData, e?: React.ChangeEvent<HTMLSelectElement>) => void;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useEffect,useRef,useState}from"react";import styled from"@emotion/styled";import{Portal}from"../../Portal/Portal";import{H2,H4,H6}from"../../Typography/Header/Header";import{Stack}from"../../Stack/Stack";import{Container}from"../../Container/Container";import{Box}from"../../Box/Box";import{Inline}from"../../Inline/Inline";import{ButtonGroup}from"../ButtonGroup/ButtonGroup";import{Button}from"../../Button/Button";import{Text}from"../../Typography/Text/Text";import{mqValue}from"../../../shared/mediaQueries";import{useOnEscapePress}from"../../../shared/useOnEscapePress";import{useOutsideClick}from"../../../shared/useOutsideClick";let StyledBackdrop=styled("div",{target:"e5b26840",label:"StyledBackdrop"})(({theme})=>({position:"fixed",top:0,left:0,width:"100%",height:"100dvh",background:theme.values.color.background.backdrop.default,display:"flex",flexDirection:"column",justifyContent:"center",alignItems:"center",flex:"1 0 auto"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <Box space=\"zero\" bSpace=\"s\">\n                <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                  {hasImage && (\n                    <>\n                      {isDismissible && (\n                        <Inline alignItems=\"right\">\n                          <ButtonClose onClick={handleClose} />\n                        </Inline>\n                      )}\n                      {imageUrl ? (\n                        <StyledImg src={imageUrl} />\n                      ) : (\n                        <ImageComponent />\n                      )}\n                    </>\n                  )}\n                  <Inline alignItems=\"spaceBetween\" noWrap>\n                    <Stack space=\"xxs\">\n                      {labelHeader && <H6>{labelHeader}</H6>}\n                      <Stack space=\"zero\">\n                        <H2>{header}</H2>\n                        {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                      </Stack>\n                    </Stack>\n                    {!hasImage && isDismissible && (\n                      <ButtonClose onClick={handleClose} />\n                    )}\n                  </Inline>\n                </Stack>\n              </Box>\n\n              <StyledBody>{children}</StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <Footer>\n                  <ButtonGroup\n                    actionButton={actionButton}\n                    secondaryButton={secondaryButton}\n                    cancelButtonText={cancelButtonLabel}\n                    onButtonClick={onAction}\n                  />\n                </Footer>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AA2DuB"} */"),StyledModalWrapper=styled("div",{target:"e5b26841",label:"StyledModalWrapper"})(({theme})=>({maxWidth:theme.variables.size.dimension.modal.width,...mqValue({width:[`calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,theme.variables.size.dimension.modal.width]})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <Box space=\"zero\" bSpace=\"s\">\n                <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                  {hasImage && (\n                    <>\n                      {isDismissible && (\n                        <Inline alignItems=\"right\">\n                          <ButtonClose onClick={handleClose} />\n                        </Inline>\n                      )}\n                      {imageUrl ? (\n                        <StyledImg src={imageUrl} />\n                      ) : (\n                        <ImageComponent />\n                      )}\n                    </>\n                  )}\n                  <Inline alignItems=\"spaceBetween\" noWrap>\n                    <Stack space=\"xxs\">\n                      {labelHeader && <H6>{labelHeader}</H6>}\n                      <Stack space=\"zero\">\n                        <H2>{header}</H2>\n                        {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                      </Stack>\n                    </Stack>\n                    {!hasImage && isDismissible && (\n                      <ButtonClose onClick={handleClose} />\n                    )}\n                  </Inline>\n                </Stack>\n              </Box>\n\n              <StyledBody>{children}</StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <Footer>\n                  <ButtonGroup\n                    actionButton={actionButton}\n                    secondaryButton={secondaryButton}\n                    cancelButtonText={cancelButtonLabel}\n                    onButtonClick={onAction}\n                  />\n                </Footer>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AAyE2B"} */"),StyledInner=styled("div",{target:"e5b26842",label:"StyledInner"})(()=>({display:"flex",maxHeight:"calc(100dvh - 100px)",flexDirection:"column",overflow:"hidden"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <Box space=\"zero\" bSpace=\"s\">\n                <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                  {hasImage && (\n                    <>\n                      {isDismissible && (\n                        <Inline alignItems=\"right\">\n                          <ButtonClose onClick={handleClose} />\n                        </Inline>\n                      )}\n                      {imageUrl ? (\n                        <StyledImg src={imageUrl} />\n                      ) : (\n                        <ImageComponent />\n                      )}\n                    </>\n                  )}\n                  <Inline alignItems=\"spaceBetween\" noWrap>\n                    <Stack space=\"xxs\">\n                      {labelHeader && <H6>{labelHeader}</H6>}\n                      <Stack space=\"zero\">\n                        <H2>{header}</H2>\n                        {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                      </Stack>\n                    </Stack>\n                    {!hasImage && isDismissible && (\n                      <ButtonClose onClick={handleClose} />\n                    )}\n                  </Inline>\n                </Stack>\n              </Box>\n\n              <StyledBody>{children}</StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <Footer>\n                  <ButtonGroup\n                    actionButton={actionButton}\n                    secondaryButton={secondaryButton}\n                    cancelButtonText={cancelButtonLabel}\n                    onButtonClick={onAction}\n                  />\n                </Footer>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AAmFoB"} */"),StyledBody=styled("div",{target:"e5b26843",label:"StyledBody"})(()=>({overflowY:"auto",maxHeight:"100%"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <Box space=\"zero\" bSpace=\"s\">\n                <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                  {hasImage && (\n                    <>\n                      {isDismissible && (\n                        <Inline alignItems=\"right\">\n                          <ButtonClose onClick={handleClose} />\n                        </Inline>\n                      )}\n                      {imageUrl ? (\n                        <StyledImg src={imageUrl} />\n                      ) : (\n                        <ImageComponent />\n                      )}\n                    </>\n                  )}\n                  <Inline alignItems=\"spaceBetween\" noWrap>\n                    <Stack space=\"xxs\">\n                      {labelHeader && <H6>{labelHeader}</H6>}\n                      <Stack space=\"zero\">\n                        <H2>{header}</H2>\n                        {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                      </Stack>\n                    </Stack>\n                    {!hasImage && isDismissible && (\n                      <ButtonClose onClick={handleClose} />\n                    )}\n                  </Inline>\n                </Stack>\n              </Box>\n\n              <StyledBody>{children}</StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <Footer>\n                  <ButtonGroup\n                    actionButton={actionButton}\n                    secondaryButton={secondaryButton}\n                    cancelButtonText={cancelButtonLabel}\n                    onButtonClick={onAction}\n                  />\n                </Footer>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AA0FmB"} */"),StyledImg=styled("img",{target:"e5b26844",label:"StyledImg"})(()=>({width:"100%",aspectRatio:"16/9"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <Box space=\"zero\" bSpace=\"s\">\n                <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                  {hasImage && (\n                    <>\n                      {isDismissible && (\n                        <Inline alignItems=\"right\">\n                          <ButtonClose onClick={handleClose} />\n                        </Inline>\n                      )}\n                      {imageUrl ? (\n                        <StyledImg src={imageUrl} />\n                      ) : (\n                        <ImageComponent />\n                      )}\n                    </>\n                  )}\n                  <Inline alignItems=\"spaceBetween\" noWrap>\n                    <Stack space=\"xxs\">\n                      {labelHeader && <H6>{labelHeader}</H6>}\n                      <Stack space=\"zero\">\n                        <H2>{header}</H2>\n                        {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                      </Stack>\n                    </Stack>\n                    {!hasImage && isDismissible && (\n                      <ButtonClose onClick={handleClose} />\n                    )}\n                  </Inline>\n                </Stack>\n              </Box>\n\n              <StyledBody>{children}</StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <Footer>\n                  <ButtonGroup\n                    actionButton={actionButton}\n                    secondaryButton={secondaryButton}\n                    cancelButtonText={cancelButtonLabel}\n                    onButtonClick={onAction}\n                  />\n                </Footer>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AA+FkB"} */"),Footer=({children,alignItems=["stretch","right"]})=>React.createElement(Box,{space:"zero",tSpace:"l"},React.createElement(Stack,{alignItems:alignItems},children)),Wrapper=({skipPortal,children})=>skipPortal?React.createElement(React.Fragment,null,children):React.createElement(Portal,null,React.createElement(StyledBackdrop,null,children)),ButtonClose=({onClick})=>React.createElement(Button,{variant:"tertiary",leftIcon:"x",size:"s",onClick:onClick});export function Modal({header,subHeader,labelHeader,children,imageUrl,ImageComponent,secondaryButton,cancelButtonLabel,isDismissible=!0,actionButton,role,onAction,privateProps:{skipPortal},"data-e2e-test-id":dataE2eTestId}){let modalRef=useRef(null),[isOpening,setOpening]=useState(!0),handleClose=()=>{isDismissible&&onAction("cancel")};useEffect(()=>{setOpening(!1)},[]),useOutsideClick(modalRef,handleClose,!skipPortal&&!isOpening),useOnEscapePress(handleClose,[handleClose]);let hasImage=imageUrl||ImageComponent;return React.createElement(Wrapper,{skipPortal:skipPortal},React.createElement(StyledModalWrapper,{ref:modalRef,role:role,"data-e2e-test-id":dataE2eTestId,"data-ds-id":"Modal"},React.createElement(Container,{elevation:4},React.createElement(Box,{space:["m","xl"]},React.createElement(StyledInner,null,React.createElement(Box,{space:"zero",bSpace:"s"},React.createElement(Stack,{space:hasImage?["s","l"]:"zero"},hasImage&&React.createElement(React.Fragment,null,isDismissible&&React.createElement(Inline,{alignItems:"right"},React.createElement(ButtonClose,{onClick:handleClose})),imageUrl?React.createElement(StyledImg,{src:imageUrl}):React.createElement(ImageComponent,null)),React.createElement(Inline,{alignItems:"spaceBetween",noWrap:!0},React.createElement(Stack,{space:"xxs"},labelHeader&&React.createElement(H6,null,labelHeader),React.createElement(Stack,{space:"zero"},React.createElement(H2,null,header),subHeader&&React.createElement(H4,{color:"tertiary"},subHeader))),!hasImage&&isDismissible&&React.createElement(ButtonClose,{onClick:handleClose})))),React.createElement(StyledBody,null,children),(actionButton||cancelButtonLabel||secondaryButton)&&React.createElement(Footer,null,React.createElement(ButtonGroup,{actionButton:actionButton,secondaryButton:secondaryButton,cancelButtonText:cancelButtonLabel,onButtonClick:onAction})))))))}Modal.defaultProps={role:"dialog","data-e2e-test-id":void 0,privateProps:{skipPortal:!1}},Modal.Stack=({children,...rest})=>React.createElement(Stack,{...rest,space:"m"},children),Modal.Text=({children,...rest})=>React.createElement(Text,{...rest,size:"m",color:"tertiary"},children);
|
|
1
|
+
import React,{useEffect,useRef,useState}from"react";import styled from"@emotion/styled";import{Portal}from"../../Portal/Portal";import{H2,H4,H6}from"../../Typography/Header/Header";import{Stack}from"../../Stack/Stack";import{Container}from"../../Container/Container";import{Box}from"../../Box/Box";import{Inline}from"../../Inline/Inline";import{ButtonGroup}from"../ButtonGroup/ButtonGroup";import{Button}from"../../Button/Button";import{Text}from"../../Typography/Text/Text";import{mqValue}from"../../../shared/mediaQueries";import{useOnEscapePress}from"../../../shared/useOnEscapePress";import{useOutsideClick}from"../../../shared/useOutsideClick";let StyledBackdrop=styled("div",{target:"e1248jbd0",label:"StyledBackdrop"})(({theme})=>({position:"fixed",top:0,left:0,width:"100%",height:"100dvh",background:theme.values.color.background.backdrop.default,display:"flex",flexDirection:"column",justifyContent:"center",alignItems:"center",flex:"1 0 auto"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\n\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nconst HorizontalOffset: React.FC = ({ children }) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {children}\n  </Box>\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <HorizontalOffset>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {isDismissible && (\n                          <Inline alignItems=\"right\">\n                            <ButtonClose onClick={handleClose} />\n                          </Inline>\n                        )}\n                        {imageUrl ? (\n                          <StyledImg src={imageUrl} />\n                        ) : (\n                          <ImageComponent />\n                        )}\n                      </>\n                    )}\n                    <Inline alignItems=\"spaceBetween\" noWrap>\n                      <Stack space=\"xxs\">\n                        {labelHeader && <H6>{labelHeader}</H6>}\n                        <Stack space=\"zero\">\n                          <H2>{header}</H2>\n                          {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                        </Stack>\n                      </Stack>\n                      {!hasImage && isDismissible && (\n                        <ButtonClose onClick={handleClose} />\n                      )}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody>\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset>{children}</HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset>\n                  <Footer>\n                    <ButtonGroup\n                      actionButton={actionButton}\n                      secondaryButton={secondaryButton}\n                      cancelButtonText={cancelButtonLabel}\n                      onButtonClick={onAction}\n                    />\n                  </Footer>\n                </HorizontalOffset>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AA4DuB"} */"),StyledModalWrapper=styled("div",{target:"e1248jbd1",label:"StyledModalWrapper"})(({theme})=>({maxWidth:theme.variables.size.dimension.modal.width,...mqValue({width:[`calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,theme.variables.size.dimension.modal.width]})}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\n\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nconst HorizontalOffset: React.FC = ({ children }) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {children}\n  </Box>\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <HorizontalOffset>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {isDismissible && (\n                          <Inline alignItems=\"right\">\n                            <ButtonClose onClick={handleClose} />\n                          </Inline>\n                        )}\n                        {imageUrl ? (\n                          <StyledImg src={imageUrl} />\n                        ) : (\n                          <ImageComponent />\n                        )}\n                      </>\n                    )}\n                    <Inline alignItems=\"spaceBetween\" noWrap>\n                      <Stack space=\"xxs\">\n                        {labelHeader && <H6>{labelHeader}</H6>}\n                        <Stack space=\"zero\">\n                          <H2>{header}</H2>\n                          {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                        </Stack>\n                      </Stack>\n                      {!hasImage && isDismissible && (\n                        <ButtonClose onClick={handleClose} />\n                      )}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody>\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset>{children}</HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset>\n                  <Footer>\n                    <ButtonGroup\n                      actionButton={actionButton}\n                      secondaryButton={secondaryButton}\n                      cancelButtonText={cancelButtonLabel}\n                      onButtonClick={onAction}\n                    />\n                  </Footer>\n                </HorizontalOffset>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AA0E2B"} */"),StyledInner=styled("div",{target:"e1248jbd2",label:"StyledInner"})(()=>({display:"flex",maxHeight:"calc(100dvh - 100px)",flexDirection:"column",overflow:"hidden"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\n\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nconst HorizontalOffset: React.FC = ({ children }) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {children}\n  </Box>\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <HorizontalOffset>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {isDismissible && (\n                          <Inline alignItems=\"right\">\n                            <ButtonClose onClick={handleClose} />\n                          </Inline>\n                        )}\n                        {imageUrl ? (\n                          <StyledImg src={imageUrl} />\n                        ) : (\n                          <ImageComponent />\n                        )}\n                      </>\n                    )}\n                    <Inline alignItems=\"spaceBetween\" noWrap>\n                      <Stack space=\"xxs\">\n                        {labelHeader && <H6>{labelHeader}</H6>}\n                        <Stack space=\"zero\">\n                          <H2>{header}</H2>\n                          {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                        </Stack>\n                      </Stack>\n                      {!hasImage && isDismissible && (\n                        <ButtonClose onClick={handleClose} />\n                      )}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody>\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset>{children}</HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset>\n                  <Footer>\n                    <ButtonGroup\n                      actionButton={actionButton}\n                      secondaryButton={secondaryButton}\n                      cancelButtonText={cancelButtonLabel}\n                      onButtonClick={onAction}\n                    />\n                  </Footer>\n                </HorizontalOffset>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AAoFoB"} */"),StyledBody=styled("div",{target:"e1248jbd3",label:"StyledBody"})(()=>({overflowY:"auto",maxHeight:"100%"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\n\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nconst HorizontalOffset: React.FC = ({ children }) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {children}\n  </Box>\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <HorizontalOffset>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {isDismissible && (\n                          <Inline alignItems=\"right\">\n                            <ButtonClose onClick={handleClose} />\n                          </Inline>\n                        )}\n                        {imageUrl ? (\n                          <StyledImg src={imageUrl} />\n                        ) : (\n                          <ImageComponent />\n                        )}\n                      </>\n                    )}\n                    <Inline alignItems=\"spaceBetween\" noWrap>\n                      <Stack space=\"xxs\">\n                        {labelHeader && <H6>{labelHeader}</H6>}\n                        <Stack space=\"zero\">\n                          <H2>{header}</H2>\n                          {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                        </Stack>\n                      </Stack>\n                      {!hasImage && isDismissible && (\n                        <ButtonClose onClick={handleClose} />\n                      )}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody>\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset>{children}</HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset>\n                  <Footer>\n                    <ButtonGroup\n                      actionButton={actionButton}\n                      secondaryButton={secondaryButton}\n                      cancelButtonText={cancelButtonLabel}\n                      onButtonClick={onAction}\n                    />\n                  </Footer>\n                </HorizontalOffset>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AA2FmB"} */"),StyledImg=styled("img",{target:"e1248jbd4",label:"StyledImg"})(()=>({width:"100%",aspectRatio:"16/9"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Patterns/Modal/Modal.tsx","sources":["src/components/Patterns/Modal/Modal.tsx"],"sourcesContent":["import React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Portal } from \"../../Portal/Portal\";\nimport { H2, H4, H6 } from \"../../Typography/Header/Header\";\nimport type { StackProps } from \"../../Stack/Stack\";\nimport { Stack } from \"../../Stack/Stack\";\nimport { Container } from \"../../Container/Container\";\nimport { Box } from \"../../Box/Box\";\n\nimport { Inline } from \"../../Inline/Inline\";\nimport type { ButtonGroupButtonProps } from \"../ButtonGroup/ButtonGroup\";\nimport { ButtonGroup } from \"../ButtonGroup/ButtonGroup\";\nimport { Button } from \"../../Button/Button\";\nimport type { TextProps } from \"../../Typography/Text/Text\";\nimport { Text } from \"../../Typography/Text/Text\";\nimport { mqValue } from \"../../../shared/mediaQueries\";\nimport { useOnEscapePress } from \"../../../shared/useOnEscapePress\";\nimport { useOutsideClick } from \"../../../shared/useOutsideClick\";\n\ntype BaseProps = {\n  header: string;\n  onAction: (action: \"cancel\" | \"action\") => void;\n  role?: \"dialog\" | \"alertdialog\";\n  labelHeader?: string;\n  /** The black color is going to be replaced with gray one */\n  subHeader?: string;\n  /** Text and other props for action Button */\n  actionButton?: ButtonGroupButtonProps;\n  /** Text and other props for action Button */\n  secondaryButton?: ButtonGroupButtonProps;\n  /** @deprecated Use secondaryButton to pass in text and other props for secondary button */\n  cancelButtonLabel?: string;\n  /** Aim to use <Modal.Text> and <Modal.Stack> */\n  children: React.ReactNode;\n  \"data-e2e-test-id\"?: string;\n  privateProps?: { skipPortal?: boolean };\n  /** It's a bad pattern to use non-dismissible Modal */\n  isDismissible: boolean;\n};\n\ntype ConditionalProps =\n  | {\n      /** Aspect ratio 16:9 */\n      imageUrl?: string;\n      ImageComponent?: never;\n    }\n  | {\n      imageUrl?: never;\n      /** Accept ratio 16:9 */\n      ImageComponent?: React.ElementType<any>;\n    };\n\nexport type ModalProps = BaseProps & ConditionalProps;\n\nconst defaultModalProps: Partial<ModalProps> = {\n  role: \"dialog\",\n  \"data-e2e-test-id\": undefined,\n  privateProps: { skipPortal: false },\n};\n\nconst StyledBackdrop = styled.div(({ theme }) => ({\n  position: \"fixed\",\n  top: 0,\n  left: 0,\n  width: \"100%\",\n  height: \"100dvh\",\n  background: theme.values.color.background.backdrop.default,\n  display: \"flex\",\n  flexDirection: \"column\",\n  justifyContent: \"center\",\n  alignItems: \"center\",\n  flex: \"1 0 auto\",\n}));\n\nconst StyledModalWrapper = styled.div(({ theme }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width,\n  ...mqValue({\n    width: [\n      `calc(100vw - calc(${theme.variables.size.spacing.m} * 2))`,\n      theme.variables.size.dimension.modal.width,\n    ],\n  }),\n}));\n\nconst StyledInner = styled.div(() => ({\n  display: \"flex\",\n  maxHeight: `calc(100dvh - 100px)`,\n  flexDirection: \"column\",\n  overflow: \"hidden\",\n}));\n\nconst StyledBody = styled.div(() => ({\n  overflowY: \"auto\",\n  maxHeight: \"100%\",\n}));\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst ModalStack = ({ children, ...rest }: Omit<StackProps, \"space\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Stack {...rest} space=\"m\">\n    {children}\n  </Stack>\n);\n\nconst ModalText = ({\n  children,\n  ...rest\n}: Omit<TextProps, \"size\" | \"variant\">) => (\n  // eslint-disable-next-line react/jsx-props-no-spreading\n  <Text {...rest} size=\"m\" color=\"tertiary\">\n    {children}\n  </Text>\n);\n\nconst Footer = ({\n  children,\n  alignItems = [\"stretch\", \"right\"],\n}: {\n  children: React.ReactNode[] | React.ReactElement;\n  alignItems?: StackProps[\"alignItems\"];\n}) => (\n  <Box space=\"zero\" tSpace=\"l\">\n    <Stack alignItems={alignItems}>{children}</Stack>\n  </Box>\n);\n\nconst Wrapper = ({\n  skipPortal,\n  children,\n}: {\n  skipPortal: boolean;\n  children: React.ReactElement;\n}): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <Button variant=\"tertiary\" leftIcon=\"x\" size=\"s\" onClick={onClick} />\n);\n\nconst HorizontalOffset: React.FC = ({ children }) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {children}\n  </Box>\n);\n\nexport function Modal({\n  header,\n  subHeader,\n  labelHeader,\n  children,\n  imageUrl,\n  ImageComponent,\n  secondaryButton,\n  cancelButtonLabel,\n  isDismissible = true,\n  actionButton,\n  role,\n  onAction,\n  privateProps: { skipPortal },\n  \"data-e2e-test-id\": dataE2eTestId,\n}: ModalProps): React.ReactElement {\n  const modalRef = useRef(null);\n  const [isOpening, setOpening] = useState(true);\n  const handleClose = () => {\n    if (isDismissible) {\n      onAction(\"cancel\");\n    }\n  };\n  const hasBackdropDismiss = !skipPortal;\n\n  useEffect(() => {\n    setOpening(false);\n  }, []);\n\n  useOutsideClick(modalRef, handleClose, hasBackdropDismiss && !isOpening);\n  useOnEscapePress(handleClose, [handleClose]);\n\n  const hasImage = imageUrl || ImageComponent;\n\n  return (\n    <Wrapper skipPortal={skipPortal}>\n      <StyledModalWrapper\n        ref={modalRef}\n        role={role}\n        data-e2e-test-id={dataE2eTestId}\n        data-ds-id=\"Modal\"\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner>\n              <HorizontalOffset>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {isDismissible && (\n                          <Inline alignItems=\"right\">\n                            <ButtonClose onClick={handleClose} />\n                          </Inline>\n                        )}\n                        {imageUrl ? (\n                          <StyledImg src={imageUrl} />\n                        ) : (\n                          <ImageComponent />\n                        )}\n                      </>\n                    )}\n                    <Inline alignItems=\"spaceBetween\" noWrap>\n                      <Stack space=\"xxs\">\n                        {labelHeader && <H6>{labelHeader}</H6>}\n                        <Stack space=\"zero\">\n                          <H2>{header}</H2>\n                          {subHeader && <H4 color=\"tertiary\">{subHeader}</H4>}\n                        </Stack>\n                      </Stack>\n                      {!hasImage && isDismissible && (\n                        <ButtonClose onClick={handleClose} />\n                      )}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody>\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset>{children}</HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset>\n                  <Footer>\n                    <ButtonGroup\n                      actionButton={actionButton}\n                      secondaryButton={secondaryButton}\n                      cancelButtonText={cancelButtonLabel}\n                      onButtonClick={onAction}\n                    />\n                  </Footer>\n                </HorizontalOffset>\n              )}\n            </StyledInner>\n          </Box>\n        </Container>\n      </StyledModalWrapper>\n    </Wrapper>\n  );\n}\n\nModal.defaultProps = defaultModalProps;\n\nModal.Stack = ModalStack;\nModal.Text = ModalText;\n"],"names":[],"mappings":"AAgGkB"} */"),Footer=({children,alignItems=["stretch","right"]})=>React.createElement(Box,{space:"zero",tSpace:"l"},React.createElement(Stack,{alignItems:alignItems},children)),Wrapper=({skipPortal,children})=>skipPortal?React.createElement(React.Fragment,null,children):React.createElement(Portal,null,React.createElement(StyledBackdrop,null,children)),ButtonClose=({onClick})=>React.createElement(Button,{variant:"tertiary",leftIcon:"x",size:"s",onClick:onClick}),HorizontalOffset=({children})=>React.createElement(Box,{space:["m","xl"],vSpace:"zero"},children);export function Modal({header,subHeader,labelHeader,children,imageUrl,ImageComponent,secondaryButton,cancelButtonLabel,isDismissible=!0,actionButton,role,onAction,privateProps:{skipPortal},"data-e2e-test-id":dataE2eTestId}){let modalRef=useRef(null),[isOpening,setOpening]=useState(!0),handleClose=()=>{isDismissible&&onAction("cancel")};useEffect(()=>{setOpening(!1)},[]),useOutsideClick(modalRef,handleClose,!skipPortal&&!isOpening),useOnEscapePress(handleClose,[handleClose]);let hasImage=imageUrl||ImageComponent;return React.createElement(Wrapper,{skipPortal:skipPortal},React.createElement(StyledModalWrapper,{ref:modalRef,role:role,"data-e2e-test-id":dataE2eTestId,"data-ds-id":"Modal"},React.createElement(Container,{elevation:4},React.createElement(Box,{space:"zero",vSpace:["m","xl"]},React.createElement(StyledInner,null,React.createElement(HorizontalOffset,null,React.createElement(Box,{space:"zero",bSpace:"s"},React.createElement(Stack,{space:hasImage?["s","l"]:"zero"},hasImage&&React.createElement(React.Fragment,null,isDismissible&&React.createElement(Inline,{alignItems:"right"},React.createElement(ButtonClose,{onClick:handleClose})),imageUrl?React.createElement(StyledImg,{src:imageUrl}):React.createElement(ImageComponent,null)),React.createElement(Inline,{alignItems:"spaceBetween",noWrap:!0},React.createElement(Stack,{space:"xxs"},labelHeader&&React.createElement(H6,null,labelHeader),React.createElement(Stack,{space:"zero"},React.createElement(H2,null,header),subHeader&&React.createElement(H4,{color:"tertiary"},subHeader))),!hasImage&&isDismissible&&React.createElement(ButtonClose,{onClick:handleClose}))))),React.createElement(StyledBody,null,React.createElement(Box,{space:"zero",vSpace:"xxxs"},React.createElement(HorizontalOffset,null,children))),(actionButton||cancelButtonLabel||secondaryButton)&&React.createElement(HorizontalOffset,null,React.createElement(Footer,null,React.createElement(ButtonGroup,{actionButton:actionButton,secondaryButton:secondaryButton,cancelButtonText:cancelButtonLabel,onButtonClick:onAction}))))))))}Modal.defaultProps={role:"dialog","data-e2e-test-id":void 0,privateProps:{skipPortal:!1}},Modal.Stack=({children,...rest})=>React.createElement(Stack,{...rest,space:"m"},children),Modal.Text=({children,...rest})=>React.createElement(Text,{...rest,size:"m",color:"tertiary"},children);
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import React from "react";
|
|
2
2
|
import type { BasePopoverProps } from "../Toggletip/BasePopover";
|
|
3
3
|
import type { TooltipConditionalProps } from "../Tooltip/types";
|
|
4
|
-
export type PopoverProps = Omit<BasePopoverProps, "contentPadding" | "name" | "
|
|
4
|
+
export type PopoverProps = Omit<BasePopoverProps, "contentPadding" | "name" | "subTheme" | "defaultVerticalPlacement"> & TooltipConditionalProps;
|
|
5
5
|
export declare function Popover({ renderAsSheetOnMobile, ...rest }: PopoverProps): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React from"react";import{BasePopover}from"../Toggletip/BasePopover";export function Popover({renderAsSheetOnMobile=!0,...rest}){return React.createElement(BasePopover,{name:"Popover",maxWidth:400,defaultVerticalPlacement:"bottom",
|
|
1
|
+
import React from"react";import{BasePopover}from"../Toggletip/BasePopover";export function Popover({renderAsSheetOnMobile=!0,...rest}){return React.createElement(BasePopover,{name:"Popover",maxWidth:400,defaultVerticalPlacement:"bottom",renderAsSheetOnMobile:renderAsSheetOnMobile,...rest})}
|
|
@@ -2,7 +2,7 @@ import type { ReactElement } from "react";
|
|
|
2
2
|
import React from "react";
|
|
3
3
|
import type { TooltipContentProps } from "../Tooltip/TooltipContent";
|
|
4
4
|
import type { TooltipConditionalProps } from "../Tooltip/types";
|
|
5
|
-
type BaseProps = Pick<TooltipContentProps, "placement" | "portalContainer" | "maxWidth" | "contentPadding" | "hideArrow" | "
|
|
5
|
+
type BaseProps = Pick<TooltipContentProps, "placement" | "portalContainer" | "maxWidth" | "contentPadding" | "hideArrow" | "subTheme" | "defaultVerticalPlacement"> & {
|
|
6
6
|
name?: string;
|
|
7
7
|
content: ReactElement;
|
|
8
8
|
"data-e2e-test-id"?: string;
|
|
@@ -14,5 +14,5 @@ type BaseProps = Pick<TooltipContentProps, "placement" | "portalContainer" | "ma
|
|
|
14
14
|
renderAsSheetOnMobile?: boolean;
|
|
15
15
|
};
|
|
16
16
|
export type BasePopoverProps = BaseProps & TooltipConditionalProps;
|
|
17
|
-
export declare function BasePopover({ placement, content, children, contentPadding, maxWidth, externalTriggerRef, portalContainer, name, isVisible: isPopoverVisible, dismissOnOutsideClick, "data-e2e-test-id": dataE2eTestId,
|
|
17
|
+
export declare function BasePopover({ placement, content, children, contentPadding, maxWidth, externalTriggerRef, portalContainer, name, isVisible: isPopoverVisible, dismissOnOutsideClick, "data-e2e-test-id": dataE2eTestId, subTheme, defaultVerticalPlacement, onVisibilityChange, disableInitialFocus, disableReturnFocusToTrigger, renderAsSheetOnMobile, ...restContentProps }: BasePopoverProps): React.ReactElement;
|
|
18
18
|
export {};
|