@amboss/design-system 2.3.0 → 2.3.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/MaskedInput/MaskedInput.d.ts +1 -1
- package/build/cjs/components/Patterns/Modal/Modal.d.ts +4 -1
- package/build/cjs/components/Patterns/Modal/Modal.js +1 -1
- package/build/cjs/shared/useScrollDetection.d.ts +9 -0
- package/build/cjs/shared/useScrollDetection.js +1 -0
- package/build/cjs/web-tokens/_sizes.json +21 -1
- package/build/cjs/web-tokens/assets/icons.json +3 -0
- package/build/cjs/web-tokens/assets/icons16.json +3 -0
- package/build/cjs/web-tokens/visualConfig.d.ts +20 -1
- package/build/cjs/web-tokens/visualConfig.js +1 -1
- package/build/esm/components/Form/MaskedInput/MaskedInput.d.ts +1 -1
- package/build/esm/components/Patterns/Modal/Modal.d.ts +4 -1
- package/build/esm/components/Patterns/Modal/Modal.js +1 -1
- package/build/esm/shared/useScrollDetection.d.ts +9 -0
- package/build/esm/shared/useScrollDetection.js +1 -0
- package/build/esm/web-tokens/_sizes.json +21 -1
- package/build/esm/web-tokens/assets/icons.json +3 -0
- package/build/esm/web-tokens/assets/icons16.json +3 -0
- package/build/esm/web-tokens/visualConfig.d.ts +20 -1
- package/build/esm/web-tokens/visualConfig.js +1 -1
- package/build/scss/_variables.scss +5 -0
- package/package.json +1 -1
|
@@ -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:"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
|
+
import React,{useEffect,useRef,useState}from"react";import styled from"@emotion/styled";import{keyframes}from"@emotion/react";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{PictogramButton}from"../../PictogramButton/PictogramButton";import{Text}from"../../Typography/Text/Text";import{mqValue}from"../../../shared/mediaQueries";import{useOnEscapePress}from"../../../shared/useOnEscapePress";import{useOutsideClick}from"../../../shared/useOutsideClick";import{useScrollDetection}from"../../../shared/useScrollDetection";let fadeInBackdrop=keyframes("from{opacity:0;}to{opacity:1;}","fadeInBackdrop","/*# 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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AAkEuB"} */"),fadeInModal=keyframes("from{opacity:0;transform:translateY(20px);}to{opacity:1;transform:translateY(0);}","fadeInModal","/*# 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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AA2EoB"} */"),StyledBackdrop=styled("div",{target:"e1dtl8lk0",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",animation:`${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`}),"/*# 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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AAsFuB"} */"),StyledModalWrapper=styled("div",{target:"e1dtl8lk1",label:"StyledModalWrapper"})(({theme,isFullScreen})=>({animation:`${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,...mqValue({padding:[theme.variables.size.spacing.m,theme.variables.size.spacing.l],width:["100%",isFullScreen?"100%":theme.variables.size.dimension.modal.width.s]})}),"/*# 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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AAqG2B"} */"),StyledInner=styled("div",{target:"e1dtl8lk2",label:"StyledInner"})(({theme,isFullScreen})=>{let maxHeight=`calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;return{display:"flex",...mqValue({height:[`calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,isFullScreen?maxHeight:"auto"]}),...!isFullScreen&&{maxHeight},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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AAkHoB"} */"),StyledBody=styled("div",{target:"e1dtl8lk3",label:"StyledBody"})(({theme,borderBottom,borderTop})=>({overflowY:"auto",maxHeight:"100%",...borderBottom&&{borderBottom:`1px solid ${theme.values.color.divider.primary}`},...borderTop&&{borderTop:`1px solid ${theme.values.color.divider.primary}`}}),"/*# 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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AA8ImB"} */"),StyledImg=styled("img",{target:"e1dtl8lk4",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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AA2JkB"} */"),StyledMaxWidth=styled("div",{target:"e1dtl8lk5",label:"StyledMaxWidth"})(({theme,size})=>({maxWidth:theme.variables.size.dimension.modal.width[size],margin:"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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AAgKuB"} */"),StyledXContainer=styled("div",{target:"e1dtl8lk6",label:"StyledXContainer"})(({theme})=>({position:"absolute",top:theme.variables.size.spacing.l,right:theme.variables.size.spacing.l}),"/*# 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":["/* eslint-disable react/jsx-props-no-spreading */\nimport React, { useEffect, useRef, useState } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { keyframes } from \"@emotion/react\";\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 { PictogramButton } from \"../../PictogramButton/PictogramButton\";\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\";\nimport { useScrollDetection } from \"../../../shared/useScrollDetection\";\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  isFullScreen?: boolean;\n  isMaxWidthLimit?: boolean;\n  size?: \"s\" | \"m\" | \"l\";\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 fadeInBackdrop = keyframes`\n  from {\n    opacity: 0;\n  }\n  to {\n    opacity: 1;\n  }\n`;\n\nconst fadeInModal = keyframes`\n  from {\n    opacity: 0;\n    transform: translateY(20px);\n  }\n  to {\n    opacity: 1;\n    transform: translateY(0);\n  }\n`;\n\nconst StyledBackdrop = styled.div<Partial<BaseProps>>(({ 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  animation: `${fadeInBackdrop} ${theme.variables.animation.modalBackdrop.open.duration} ${theme.variables.animation.modalBackdrop.open.timingFunction} both`,\n}));\n\nconst StyledModalWrapper = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => ({\n    animation: `${fadeInModal} ${theme.variables.animation.modal.open.duration} ${theme.variables.animation.modal.open.delay} ${theme.variables.animation.modal.open.timingFunction} both`,\n    ...mqValue({\n      padding: [theme.variables.size.spacing.m, theme.variables.size.spacing.l],\n      width: [\n        \"100%\",\n        isFullScreen ? `100%` : theme.variables.size.dimension.modal.width.s,\n      ],\n    }),\n  })\n);\n\nconst StyledInner = styled.div<Partial<ModalProps>>(\n  ({ theme, isFullScreen }) => {\n    const maxHeight = `calc(100dvh - ${theme.variables.size.spacing.xl} * 2 - ${theme.variables.size.spacing.l} * 2)`;\n    return {\n      display: \"flex\",\n\n      ...mqValue({\n        height: [\n          `calc(100dvh - ${theme.variables.size.spacing.m} * 4)`,\n          isFullScreen ? maxHeight : \"auto\",\n        ],\n      }),\n\n      ...(!isFullScreen && {\n        maxHeight,\n      }),\n\n      flexDirection: \"column\",\n      overflow: \"hidden\",\n    };\n  }\n);\n\ntype StyledBodyProps = {\n  borderBottom: boolean;\n  borderTop: boolean;\n};\n\nconst StyledBody = styled.div<StyledBodyProps>(\n  ({ theme, borderBottom, borderTop }) => ({\n    overflowY: \"auto\",\n    maxHeight: \"100%\",\n    ...(borderBottom && {\n      borderBottom: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n    ...(borderTop && {\n      borderTop: `1px solid ${theme.values.color.divider.primary}`,\n    }),\n  })\n);\n\nconst StyledImg = styled.img(() => ({\n  width: \"100%\",\n  aspectRatio: \"16/9\",\n}));\n\nconst StyledMaxWidth = styled.div<Partial<ModalProps>>(({ theme, size }) => ({\n  maxWidth: theme.variables.size.dimension.modal.width[size],\n  margin: \"0 auto\",\n}));\n\nconst StyledXContainer = styled.div(({ theme }) => ({\n  position: \"absolute\",\n  top: theme.variables.size.spacing.l,\n  right: theme.variables.size.spacing.l,\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} & Partial<ModalProps>): React.ReactElement =>\n  skipPortal ? (\n    <>{children}</>\n  ) : (\n    <Portal>\n      <StyledBackdrop>{children}</StyledBackdrop>\n    </Portal>\n  );\n\nconst ButtonClose = ({ onClick }: { onClick: React.ReactEventHandler }) => (\n  <StyledXContainer>\n    <PictogramButton variant=\"tertiary\" icon=\"x\" size=\"s\" onClick={onClick} />\n  </StyledXContainer>\n);\n\nconst HorizontalOffset = ({\n  children,\n  isFullScreen,\n  isMaxWidthLimit,\n}: Partial<ModalProps>) => (\n  <Box space={[\"m\", \"xl\"]} vSpace=\"zero\">\n    {isFullScreen && isMaxWidthLimit ? (\n      <StyledMaxWidth>{children}</StyledMaxWidth>\n    ) : (\n      children\n    )}\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  isFullScreen,\n  isMaxWidthLimit = true,\n  actionButton,\n  role,\n  onAction,\n  size = \"s\",\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  const { scrollableRef, isScrollBottom, isScrollTop, hasScroll } =\n    useScrollDetection();\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  const horizontalOffsetProps = { isFullScreen, isMaxWidthLimit };\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        isFullScreen={isFullScreen}\n        size={size}\n      >\n        <Container elevation={4}>\n          <Box space=\"zero\" vSpace={[\"m\", \"xl\"]}>\n            <StyledInner isFullScreen={isFullScreen}>\n              <HorizontalOffset {...horizontalOffsetProps}>\n                <Box space=\"zero\" bSpace=\"s\">\n                  <Stack space={hasImage ? [\"s\", \"l\"] : \"zero\"}>\n                    {hasImage && (\n                      <>\n                        {imageUrl ? (\n                          <Box space=\"zero\" tSpace=\"l\">\n                            <StyledImg src={imageUrl} />\n                          </Box>\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                      {isDismissible && <ButtonClose onClick={handleClose} />}\n                    </Inline>\n                  </Stack>\n                </Box>\n              </HorizontalOffset>\n\n              <StyledBody\n                ref={scrollableRef}\n                borderBottom={hasScroll && !isScrollBottom}\n                borderTop={hasScroll && !isScrollTop}\n              >\n                <Box space=\"zero\" vSpace=\"xxxs\">\n                  <HorizontalOffset {...horizontalOffsetProps}>\n                    {children}\n                  </HorizontalOffset>\n                </Box>\n              </StyledBody>\n\n              {(actionButton || cancelButtonLabel || secondaryButton) && (\n                <HorizontalOffset {...horizontalOffsetProps}>\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":"AAqKyB"} */"),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(StyledXContainer,null,React.createElement(PictogramButton,{variant:"tertiary",icon:"x",size:"s",onClick:onClick})),HorizontalOffset=({children,isFullScreen,isMaxWidthLimit})=>React.createElement(Box,{space:["m","xl"],vSpace:"zero"},isFullScreen&&isMaxWidthLimit?React.createElement(StyledMaxWidth,null,children):children);export function Modal({header,subHeader,labelHeader,children,imageUrl,ImageComponent,secondaryButton,cancelButtonLabel,isDismissible=!0,isFullScreen,isMaxWidthLimit=!0,actionButton,role,onAction,size="s",privateProps:{skipPortal},"data-e2e-test-id":dataE2eTestId}){let modalRef=useRef(null),[isOpening,setOpening]=useState(!0),handleClose=()=>{isDismissible&&onAction("cancel")},{scrollableRef,isScrollBottom,isScrollTop,hasScroll}=useScrollDetection();useEffect(()=>{setOpening(!1)},[]),useOutsideClick(modalRef,handleClose,!skipPortal&&!isOpening),useOnEscapePress(handleClose,[handleClose]);let hasImage=imageUrl||ImageComponent,horizontalOffsetProps={isFullScreen,isMaxWidthLimit};return React.createElement(Wrapper,{skipPortal:skipPortal},React.createElement(StyledModalWrapper,{ref:modalRef,role:role,"data-e2e-test-id":dataE2eTestId,"data-ds-id":"Modal",isFullScreen:isFullScreen,size:size},React.createElement(Container,{elevation:4},React.createElement(Box,{space:"zero",vSpace:["m","xl"]},React.createElement(StyledInner,{isFullScreen:isFullScreen},React.createElement(HorizontalOffset,horizontalOffsetProps,React.createElement(Box,{space:"zero",bSpace:"s"},React.createElement(Stack,{space:hasImage?["s","l"]:"zero"},hasImage&&React.createElement(React.Fragment,null,imageUrl?React.createElement(Box,{space:"zero",tSpace:"l"},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))),isDismissible&&React.createElement(ButtonClose,{onClick:handleClose}))))),React.createElement(StyledBody,{ref:scrollableRef,borderBottom:hasScroll&&!isScrollBottom,borderTop:hasScroll&&!isScrollTop},React.createElement(Box,{space:"zero",vSpace:"xxxs"},React.createElement(HorizontalOffset,horizontalOffsetProps,children))),(actionButton||cancelButtonLabel||secondaryButton)&&React.createElement(HorizontalOffset,horizontalOffsetProps,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);
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/// <reference types="react" />
|
|
2
|
+
type useScrollDetectionReturnType = {
|
|
3
|
+
scrollableRef: React.RefObject<HTMLDivElement>;
|
|
4
|
+
isScrollBottom: boolean;
|
|
5
|
+
isScrollTop: boolean;
|
|
6
|
+
hasScroll: boolean;
|
|
7
|
+
};
|
|
8
|
+
export declare const useScrollDetection: () => useScrollDetectionReturnType;
|
|
9
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
import{useState,useEffect,useRef}from"react";export const useScrollDetection=()=>{let[isScrollTop,setIsScrollTop]=useState(!1),[isScrollBottom,setIsScrollBottom]=useState(!1),[hasScroll,setHasScroll]=useState(!0),scrollableRef=useRef(null);return useEffect(()=>{let resizeObserver,handleScroll;let element=scrollableRef.current;return element&&(handleScroll=()=>{if(!(element.scrollHeight>element.clientHeight)){setHasScroll(!1);return}let bottom=1>Math.abs(element.scrollHeight-element.scrollTop-element.clientHeight);bottom!==isScrollBottom&&setIsScrollBottom(bottom);let top=0===element.scrollTop;top!==isScrollTop&&setIsScrollTop(top),hasScroll||setHasScroll(!0)},(resizeObserver=new ResizeObserver(()=>{handleScroll()})).observe(element),element.addEventListener("scroll",handleScroll,{passive:!0})),()=>{element&&element.removeEventListener("scroll",handleScroll),resizeObserver&&resizeObserver.disconnect()}},[isScrollBottom,isScrollTop,hasScroll]),{scrollableRef,isScrollBottom,isScrollTop,hasScroll}};
|
|
@@ -356,9 +356,29 @@
|
|
|
356
356
|
"subitemName": "modal width",
|
|
357
357
|
"value": "560px",
|
|
358
358
|
"original": 560,
|
|
359
|
-
"variableName": "$size-dimension-modal-width",
|
|
359
|
+
"variableName": "$size-dimension-modal-width-s",
|
|
360
360
|
"isUsed": true
|
|
361
361
|
},
|
|
362
|
+
{
|
|
363
|
+
"category": "size",
|
|
364
|
+
"type": "dimension",
|
|
365
|
+
"name": "dimension modal",
|
|
366
|
+
"subitemName": "modal width",
|
|
367
|
+
"value": "780px",
|
|
368
|
+
"original": 780,
|
|
369
|
+
"variableName": "$size-dimension-modal-width-m",
|
|
370
|
+
"isUsed": false
|
|
371
|
+
},
|
|
372
|
+
{
|
|
373
|
+
"category": "size",
|
|
374
|
+
"type": "dimension",
|
|
375
|
+
"name": "dimension modal",
|
|
376
|
+
"subitemName": "modal width",
|
|
377
|
+
"value": "1280px",
|
|
378
|
+
"original": 1280,
|
|
379
|
+
"variableName": "$size-dimension-modal-width-l",
|
|
380
|
+
"isUsed": false
|
|
381
|
+
},
|
|
362
382
|
{
|
|
363
383
|
"category": "size",
|
|
364
384
|
"type": "dimension",
|
|
@@ -3,6 +3,7 @@
|
|
|
3
3
|
"alert-triangle": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-alert-triangle\"><path d=\"M10.29 3.86L1.82 18a2 2 0 0 0 1.71 3h16.94a2 2 0 0 0 1.71-3L13.71 3.86a2 2 0 0 0-3.42 0z\"></path><line x1=\"12\" y1=\"9\" x2=\"12\" y2=\"13\"></line><line x1=\"12\" y1=\"17\" x2=\"12.01\" y2=\"17\"></line></svg>",
|
|
4
4
|
"alert-octagon": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-alert-octagon\"><polygon points=\"7.86 2 16.14 2 22 7.86 22 16.14 16.14 22 7.86 22 2 16.14 2 7.86 7.86 2\"></polygon><line x1=\"12\" y1=\"8\" x2=\"12\" y2=\"12\"></line><line x1=\"12\" y1=\"16\" x2=\"12.01\" y2=\"16\"></line></svg>",
|
|
5
5
|
"align-left": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-align-left\"><line x1=\"17\" y1=\"10\" x2=\"3\" y2=\"10\"></line><line x1=\"21\" y1=\"6\" x2=\"3\" y2=\"6\"></line><line x1=\"21\" y1=\"14\" x2=\"3\" y2=\"14\"></line><line x1=\"17\" y1=\"18\" x2=\"3\" y2=\"18\"></line></svg>",
|
|
6
|
+
"amboss-logo": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\">\n <path fill=\"#047A88\" d=\"M17.2 19.77H6.56l.67-1.22h9.3l.67 1.22ZM13.29 7.5l5.39 9.83h-1.42l-3.98-7.25-.7 1.28 3.27 5.97H7.9l5.39-9.83Zm7.46 11.06L13.28 4.9l-.7 1.3L6.5 17.32H5.08l7.5-13.69-.7-1.28L3 18.55h2.82L4.48 21h14.8l-1.35-2.45h2.82Z\" clip-rule=\"evenodd\"/>\n</svg>",
|
|
6
7
|
"anamnesis": "<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n<path d=\"M3 11.8437C3 11.2915 3.44772 10.8437 4 10.8437H11C11.5523 10.8437 12 10.396 12 9.84374V9C12 8.44772 12.4477 8 13 8H20C20.5523 8 21 8.44772 21 9V21C21 21.5523 20.5523 22 20 22H4C3.44772 22 3 21.5523 3 21V11.8437Z\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linejoin=\"round\"/>\n<path d=\"M19 9V2H5V11\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linejoin=\"round\"/>\n<path d=\"M8 5L16 5\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\"/>\n<path d=\"M8 8H9\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\"/>\n</svg>\n",
|
|
7
8
|
"apple": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" viewBox=\"0 0 24 24\">\n <path fill=\"currentColor\" fill-rule=\"evenodd\"\n d=\"M16.749 0c.214 1.458-.379 2.886-1.162 3.896-.837 1.084-2.28 1.922-3.679 1.879-.255-1.396.398-2.834 1.194-3.8C13.974.906 15.469.088 16.749 0zm4.201 20.518c.721-1.105.99-1.662 1.55-2.91-4.07-1.547-4.723-7.335-.694-9.556-1.23-1.54-2.956-2.434-4.586-2.434-1.174 0-1.979.306-2.71.585-.61.232-1.169.445-1.849.445-.734 0-1.385-.233-2.066-.477-.748-.269-1.533-.55-2.508-.55-1.83 0-3.776 1.117-5.01 3.028C1.34 11.34 1.636 16.4 4.45 20.71c1.006 1.542 2.35 3.275 4.107 3.29.73.008 1.215-.21 1.74-.445.601-.27 1.255-.563 2.386-.569 1.138-.007 1.78.29 2.374.564.511.236.986.454 1.71.447 1.758-.014 3.176-1.935 4.182-3.477z\"\n clip-rule=\"evenodd\" />\n</svg>",
|
|
8
9
|
"arrow-down": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-arrow-down\"><line x1=\"12\" y1=\"5\" x2=\"12\" y2=\"19\"></line><polyline points=\"19 12 12 19 5 12\"></polyline></svg>",
|
|
@@ -26,6 +27,7 @@
|
|
|
26
27
|
"bubble-text": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" viewBox=\"0 0 24 24\">\n <path fill=\"currentColor\"\n d=\"M2.803 12l1.752 1.168c.278.186.445.499.445.832v5c0 .552.449 1 1 1h14a1 1 0 001-1V5c0-.551-.448-1-1-1H6c-.551 0-1 .449-1 1v5c0 .334-.167.646-.445.832L2.803 12zM20 22H6c-1.654 0-3-1.345-3-3v-4.464L.446 12.832a1 1 0 010-1.664L3 9.465V5c0-1.654 1.346-3 3-3h14c1.655 0 3 1.346 3 3v14c0 1.655-1.345 3-3 3zm-4.2-7H8a1 1 0 110-2h7.8a1 1 0 110 2zm2.2-4H8a1 1 0 010-2h10a1 1 0 110 2z\" />\n</svg>",
|
|
27
28
|
"bulb": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" stroke=\"currentColor\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" stroke-width=\"2\" viewBox=\"0 0 24 24\">\n <path\n d=\"M14 10h-4m2 0v7.9m4 0v-4.1c1.6-1.2 2.6-3.1 2.6-5.2 0-3.7-3-6.6-6.6-6.6S5.4 4.9 5.4 8.6c0 2.1 1 4 2.6 5.2v4.1h8zm-8 0l2 4.1h4l2-4.1\" />\n</svg>",
|
|
28
29
|
"calculator": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" stroke=\"currentColor\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" stroke-width=\"2\" viewBox=\"0 0 24 24\">\n <path\n d=\"M4 7h16M7.9 10.5h0m4 0h0m4 0h0m-8 4h0m4 0h0m4 0h0m-7.9 4h3.9m4 0h0M18 22H6c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h12c1.1 0 2 .9 2 2v16c0 1.1-.9 2-2 2z\" />\n</svg>",
|
|
30
|
+
"chart": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"currentColor\" stroke=\"currentColor\">\n <path d=\"M12.06 3A8.9 8.9 0 0 0 3 11.97a9.04 9.04 0 0 0 2.58 6.38 8.87 8.87 0 0 0 6.3 2.65A9.04 9.04 0 0 0 21 12.03 9.03 9.03 0 0 0 12.06 3Zm6.53 12.18c-1.2 0-1.22-1.14-1.41-1.99-.24-1.01-.39-2.06-.67-3.08-.52 1.42-.49 2.93-.8 4.38-.08.48-.2.96-.33 1.42-.17.5-.43 1.03-1.07 1-.57-.03-.82-.5-.98-.98-.4-1.22-.5-2.5-.68-3.76-.18-1.26-.36-2.58-.64-3.88-.51 1.56-.58 3.17-.83 4.76-.16 1-.26 2-.61 2.97-.17.45-.4.86-.91.89-.57.03-.87-.4-1.05-.88-.46-1.24-.5-2.57-.7-3.86-.12-.7-.22-1.38-.32-2.07l-.23.01c-.23 1.22-.44 2.44-.72 3.64-.16.69-.35 1.48-1.29 1.47-.93 0-1.1-.75-1.3-1.46 0-.03 0-.07-.02-.11-.1-.56-.58-1.36.19-1.58.86-.24.74.7.9 1.24.06.18.14.36.24.53.45-1.2.45-2.36.74-3.45.23-.84.2-2.01 1.4-2.02 1.19-.01 1.14 1.15 1.3 1.94.29 1.5.45 3.02.75 4.53.54-1.52.58-3.11.83-4.67.17-1.04.26-2.08.63-3.08.17-.46.4-.86.91-.88.59-.03.87.41 1.03.9.58 1.8.64 3.69.94 5.54l.4 2.64c.54-1.8.46-3.37.84-4.86.2-.83.1-2.03 1.3-2.06 1.2-.03 1.19 1.18 1.4 2 .3 1.07.3 2.2.72 3.36.33-.46.35-.9.48-1.31a.54.54 0 0 1 .6-.41c.15.02.3.1.39.22.09.12.13.27.11.43-.02.18-.06.37-.12.55-.22.83-.22 1.97-1.42 1.97Z\"/>\n</svg>",
|
|
29
31
|
"charts": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" viewBox=\"0 0 24 24\">\n <path stroke=\"currentColor\" stroke-linecap=\"square\" stroke-linejoin=\"round\" stroke-width=\"2\"\n d=\"M3 10.5s1.366.011 1.5 0a.75.75 0 00.621-.443l1.65-3.875a.75.75 0 011.4.078l3.077 6.717a.75.75 0 001.449-.072l3.095-9.307a.75.75 0 011.457-.052L18.75 9.75c.045.157 0 .75 1.5.75H21\" />\n <path fill=\"currentColor\"\n d=\"M4 19.5a1 1 0 10-2 0h2zm-2 0v3h2v-3H2zm11 0a1 1 0 10-2 0h2zm-2 0v3h2v-3h-2zm11 0a1 1 0 10-2 0h2zm-2 0v3h2v-3h-2zm-11.5-3a1 1 0 10-2 0h2zm-2 0v6h2v-6h-2zm11 0a1 1 0 10-2 0h2zm-2 0v6h2v-6h-2z\" />\n</svg>",
|
|
30
32
|
"check": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-check\"><polyline points=\"20 6 9 17 4 12\"></polyline></svg>",
|
|
31
33
|
"check-circle": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-check-circle\"><path d=\"M22 11.08V12a10 10 0 1 1-5.93-9.14\"></path><polyline points=\"22 4 12 14.01 9 11.01\"></polyline></svg>",
|
|
@@ -53,6 +55,7 @@
|
|
|
53
55
|
"edit-3-filled": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"currentColor\" stroke=\"currentColor\"\n viewBox=\"0 0 24 24\">\n <path fill=\"none\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M12 20h9\" />\n <path stroke=\"none\"\n d=\"M18 2.83c-.695 0-1.362.276-1.854.768l-12.5 12.5a.5.5 0 00-.131.232l-1 4a.5.5 0 00.606.606l4-1a.5.5 0 00.233-.131l12.5-12.5A2.62 2.62 0 0018 2.83z\" />\n</svg>",
|
|
54
56
|
"edit-square": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" viewBox=\"0 0 24 24\">\n <path stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"\n d=\"M7.333 2.667H2.666A1.333 1.333 0 001.333 4v9.334a1.333 1.333 0 001.333 1.333H12a1.334 1.334 0 001.333-1.333V8.667\" />\n <path stroke=\"currentColor\" stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\"\n d=\"M12.333 1.667a1.414 1.414 0 112 2L8 10.001l-2.667.666L6 8l6.333-6.333z\" />\n</svg>",
|
|
55
57
|
"education": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" stroke=\"currentColor\" viewBox=\"0 0 24 24\">\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M12 2L1 8l11 6 11-6-11-6z\"\n clip-rule=\"evenodd\" />\n <path stroke-linecap=\"round\" stroke-linejoin=\"round\" stroke-width=\"2\" d=\"M5 11v7l7 4.5 7-4.5v-7\" />\n <path stroke-linecap=\"round\" stroke-width=\"2\" d=\"M23 8v8\" />\n</svg>",
|
|
58
|
+
"effigos": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"currentColor\" viewBox=\"0 0 24 24\">\n <path fill-rule=\"evenodd\" d=\"M3.06 10.74c-.3 3.83.35 7.24 3.96 9.38.11.09.24.13.37.17.11.04.22.08.33.15 2.72.95 5.32.68 6.28-.73-1.1.07-2.2.08-3.31.02-3.06-.16-5.2-1.79-6.02-4.68a10.9 10.9 0 0 1-.02-5.65 6.06 6.06 0 0 1 3.18-4c2.32-1.22 4.76-1.22 7.2-.54 2.8.77 4.42 2.84 4.65 5.76.17 2.03-.9 3.4-2.9 3.68-1.6.21-3.21.25-4.81.12-1.69-.12-2.03-.53-2.03-2.25v-.85c.06-.7.48-1.2 1.18-1.26a9.84 9.84 0 0 1 1.88 0c.68.06 1.12.48 1.18 1.18.03.59.03 1.17 0 1.76v.04c0 .28-.02.57.39.54h.25c.31.02.65.03.69-.44v-.01c.08-.81.15-1.64-.05-2.44-.28-1.08-.95-1.77-2.07-1.94a9.12 9.12 0 0 0-2.44-.04C9.6 8.87 8.7 9.83 8.6 11.25c-.04.61-.04 1.22.02 1.83.15 1.52.97 2.4 2.49 2.56 2.11.21 4.23.24 6.34-.14 1.77-.32 3.02-1.37 3.38-3.21a7.01 7.01 0 0 0-2.11-6.87C16.1 3 12.94 2.62 9.6 3.3a8.03 8.03 0 0 0-6.55 7.43Zm6.73 7.4c2.04.46 4.15.42 6.16-.13l.26-.14c.16-.08.3-.16.44-.27a.73.73 0 0 1 .18-.17c.13-.1.26-.21.28-.42h-.59c-1.83.2-3.67.2-5.5.02-2.2-.23-3.56-1.45-3.84-3.58a9.95 9.95 0 0 1 0-2.67 3.92 3.92 0 0 1 3.22-3.53 8.45 8.45 0 0 1 3.6.06 4.03 4.03 0 0 1 3.05 3.92v1.33l-.02.18c-.03.18-.06.36.18.43.26.07.43-.16.56-.35a2 2 0 0 0 .26-.57 4.84 4.84 0 0 0-2.41-5.55 7.97 7.97 0 0 0-6.45-.31 4.71 4.71 0 0 0-3.14 3.67 10.7 10.7 0 0 0-.05 4 4.67 4.67 0 0 0 3.81 4.08Z\" clip-rule=\"evenodd\"/>\n</svg>",
|
|
56
59
|
"emergency": "<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\">\n<path d=\"M4.73204 15.7496L4.73204 14.7496L4.73204 14.7496L4.73204 15.7496ZM2.73204 13.7496L1.73204 13.7496L2.73204 13.7496ZM8.03524 15.7496L9.03524 15.7496L9.03524 14.7496L8.03524 14.7496L8.03524 15.7496ZM8.03524 19.2068L7.03524 19.2068L7.03524 19.2068L8.03524 19.2068ZM10.0352 21.2068L10.0352 22.2068L10.0352 22.2068L10.0352 21.2068ZM13.6885 21.2068L13.6885 22.2068L13.6885 22.2068L13.6885 21.2068ZM15.6885 19.2068L16.6885 19.2068L16.6885 19.2068L15.6885 19.2068ZM15.6885 15.7496L15.6885 14.7496L14.6885 14.7496L14.6885 15.7496L15.6885 15.7496ZM19.0691 15.7496L19.0691 14.7496L19.0691 14.7496L19.0691 15.7496ZM21.0691 13.7496L22.0691 13.7496V13.7496L21.0691 13.7496ZM21.0691 10.0964L22.0691 10.0964L21.0691 10.0964ZM19.0691 8.09638L19.0691 9.09638L19.0691 9.09638L19.0691 8.09638ZM15.6885 8.09638L14.6885 8.09638L14.6885 9.09638H15.6885V8.09638ZM15.6885 4.86979L14.6885 4.86978L14.6885 4.86979L15.6885 4.86979ZM13.6885 2.86978L13.6885 1.86978L13.6885 1.86978L13.6885 2.86978ZM10.0352 2.86979L10.0352 3.86979L10.0352 3.86979L10.0352 2.86979ZM8.03524 4.86978L7.03524 4.86978L7.03524 4.86978L8.03524 4.86978ZM8.03524 8.09638L8.03524 9.09638L9.03524 9.09638L9.03524 8.09638L8.03524 8.09638ZM4.73204 8.09638L4.73204 7.09638L4.73204 7.09638L4.73204 8.09638ZM2.73204 10.0964L3.73204 10.0964V10.0964L2.73204 10.0964ZM4.73204 14.7496C4.17976 14.7496 3.73204 14.3019 3.73204 13.7496L1.73204 13.7496C1.73204 15.4065 3.07519 16.7496 4.73204 16.7496L4.73204 14.7496ZM8.03524 14.7496L4.73204 14.7496L4.73204 16.7496L8.03524 16.7496L8.03524 14.7496ZM9.03524 19.2068L9.03524 15.7496L7.03524 15.7496L7.03524 19.2068L9.03524 19.2068ZM10.0352 20.2068C9.48295 20.2068 9.03524 19.7591 9.03524 19.2068L7.03524 19.2068C7.03524 20.8637 8.37838 22.2068 10.0352 22.2068L10.0352 20.2068ZM13.6885 20.2068L10.0352 20.2068L10.0352 22.2068L13.6885 22.2068L13.6885 20.2068ZM14.6885 19.2068C14.6885 19.7591 14.2407 20.2068 13.6885 20.2068L13.6885 22.2068C15.3453 22.2068 16.6885 20.8637 16.6885 19.2068L14.6885 19.2068ZM14.6885 15.7496L14.6885 19.2068L16.6885 19.2068L16.6885 15.7496L14.6885 15.7496ZM19.0691 14.7496L15.6885 14.7496L15.6885 16.7496L19.0691 16.7496L19.0691 14.7496ZM20.0691 13.7496C20.0691 14.3019 19.6214 14.7496 19.0691 14.7496L19.0691 16.7496C20.7259 16.7496 22.0691 15.4065 22.0691 13.7496L20.0691 13.7496ZM20.0691 10.0964L20.0691 13.7496L22.0691 13.7496L22.0691 10.0964L20.0691 10.0964ZM19.0691 9.09638C19.6214 9.09638 20.0691 9.5441 20.0691 10.0964L22.0691 10.0964C22.0691 8.43953 20.7259 7.09638 19.0691 7.09638L19.0691 9.09638ZM15.6885 9.09638H19.0691V7.09638L15.6885 7.09638V9.09638ZM14.6885 4.86979L14.6885 8.09638L16.6885 8.09638L16.6885 4.86979L14.6885 4.86979ZM13.6885 3.86978C14.2407 3.86979 14.6885 4.3175 14.6885 4.86978L16.6885 4.86979C16.6885 3.21293 15.3453 1.86979 13.6885 1.86978L13.6885 3.86978ZM10.0352 3.86979L13.6885 3.86978L13.6885 1.86978L10.0352 1.86979L10.0352 3.86979ZM9.03524 4.86978C9.03524 4.3175 9.48295 3.86978 10.0352 3.86979L10.0352 1.86979C8.37839 1.86978 7.03524 3.21293 7.03524 4.86978L9.03524 4.86978ZM9.03524 8.09638L9.03524 4.86978L7.03524 4.86978L7.03524 8.09638L9.03524 8.09638ZM4.73204 9.09638L8.03524 9.09638L8.03524 7.09638L4.73204 7.09638L4.73204 9.09638ZM3.73204 10.0964C3.73204 9.5441 4.17976 9.09638 4.73204 9.09638L4.73204 7.09638C3.07519 7.09638 1.73204 8.43953 1.73204 10.0964L3.73204 10.0964ZM3.73204 13.7496L3.73204 10.0964L1.73204 10.0964L1.73204 13.7496L3.73204 13.7496Z\" fill=\"currentColor\"/>\n</svg>\n",
|
|
57
60
|
"expand": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" fill=\"none\" stroke=\"currentColor\" stroke-linecap=\"round\"\n stroke-linejoin=\"round\" stroke-width=\"2\" viewBox=\"0 0 24 24\">\n <path d=\"M9 19.7l3 2.3 3-2.3M12 18v4m3-17.7L12 2 9 4.3M12 6V2m-7 8h14M5 14h14\" />\n</svg>",
|
|
58
61
|
"external-link": "<svg xmlns=\"http://www.w3.org/2000/svg\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"2\" stroke-linecap=\"round\" stroke-linejoin=\"round\" class=\"feather feather-external-link\"><path d=\"M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6\"></path><polyline points=\"15 3 21 3 21 9\"></polyline><line x1=\"10\" y1=\"14\" x2=\"21\" y2=\"3\"></line></svg>",
|