@amboss/design-system 2.2.2 → 2.2.4
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/BulkActionsToolbar/BulkActionsToolbar.js +1 -1
- package/build/cjs/components/Callout/Callout.d.ts +0 -3
- package/build/cjs/components/Callout/Callout.js +1 -1
- package/build/cjs/components/DataTable/useDataTableQueryFilter.js +1 -1
- package/build/cjs/components/Form/Checkbox/Checkbox.js +1 -1
- package/build/cjs/components/Form/Input/Input.js +1 -1
- package/build/cjs/components/Form/MaskedInput/MaskedInput.js +1 -1
- package/build/cjs/components/Form/Select/BaseSelect.js +1 -1
- package/build/cjs/components/Form/Select/MultiSelect.js +1 -1
- package/build/cjs/components/Form/Select/Select.d.ts +7 -0
- package/build/cjs/components/Form/Select/StyledSelectComponents.d.ts +1 -0
- package/build/cjs/components/Form/Select/StyledSelectComponents.js +1 -1
- package/build/cjs/components/Form/Select/constants.d.ts +2 -0
- package/build/cjs/components/Form/Select/constants.js +1 -0
- package/build/cjs/components/Link/Link.js +1 -1
- package/build/cjs/components/SearchResult/SearchResult.js +1 -1
- package/build/cjs/components/Toggletip/BasePopover.js +1 -1
- package/build/cjs/components/Tooltip/BaseTooltip.js +1 -1
- package/build/cjs/components/VirtualScrollList/VirtualScrollList.d.ts +2 -1
- package/build/cjs/components/VirtualScrollList/VirtualScrollList.js +1 -1
- package/build/cjs/components/VirtualScrollList/VirtualScrollListReducer.js +1 -1
- package/build/cjs/web-tokens/visualConfig.d.ts +6 -1
- package/build/cjs/web-tokens/visualConfig.js +1 -1
- package/build/esm/components/BulkActionsToolbar/BulkActionsToolbar.js +1 -1
- package/build/esm/components/Callout/Callout.d.ts +0 -3
- package/build/esm/components/Callout/Callout.js +1 -1
- package/build/esm/components/DataTable/useDataTableQueryFilter.js +1 -1
- package/build/esm/components/Form/Checkbox/Checkbox.js +1 -1
- package/build/esm/components/Form/Input/Input.js +1 -1
- package/build/esm/components/Form/MaskedInput/MaskedInput.js +1 -1
- package/build/esm/components/Form/Select/BaseSelect.js +1 -1
- package/build/esm/components/Form/Select/MultiSelect.js +1 -1
- package/build/esm/components/Form/Select/Select.d.ts +7 -0
- package/build/esm/components/Form/Select/StyledSelectComponents.d.ts +1 -0
- package/build/esm/components/Form/Select/StyledSelectComponents.js +1 -1
- package/build/esm/components/Form/Select/constants.d.ts +2 -0
- package/build/esm/components/Form/Select/constants.js +1 -0
- package/build/esm/components/Link/Link.js +1 -1
- package/build/esm/components/SearchResult/SearchResult.js +1 -1
- package/build/esm/components/Toggletip/BasePopover.js +1 -1
- package/build/esm/components/Tooltip/BaseTooltip.js +1 -1
- package/build/esm/components/VirtualScrollList/VirtualScrollList.d.ts +2 -1
- package/build/esm/components/VirtualScrollList/VirtualScrollList.js +1 -1
- package/build/esm/components/VirtualScrollList/VirtualScrollListReducer.js +1 -1
- package/build/esm/web-tokens/visualConfig.d.ts +6 -1
- package/build/esm/web-tokens/visualConfig.js +1 -1
- package/build/scss/_variables.scss +1 -0
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useRef}from"react";import styled from"@emotion/styled";import{Columns,Column}from"../Column/Columns";import{Icon}from"../Icon/Icon";import{Inline}from"../Inline/Inline";import{Stack}from"../Stack/Stack";import{StyledText}from"../Typography/StyledText/StyledText";import{Badge}from"../Badge/Badge";import{TextClamped}from"../Typography/TextClamped/TextClamped";import{Text}from"../Typography/Text/Text";let StyledResultContainer=styled("div",{target:"e1q21ux90",label:"StyledResultContainer"})(()=>({position:"relative",textDecoration:"none"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AAuB8B"} */"),StyledLinkTitle=styled("div",{target:"e1q21ux91",label:"StyledLinkTitle"})(({theme})=>({color:theme.values.color.text.accent.default}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AA4BwB"} */"),StyledLinkContainerOnHover=styled("div",{target:"e1q21ux92",label:"StyledLinkContainerOnHover"})(({theme})=>({"> a":{textDecoration:"none",color:theme.values.color.text.accent.default},"&:hover":{[`${StyledLinkTitle}`]:{textDecoration:"underline"}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AAgCmC"} */"),StyledSecLinkTitle=styled("span",{target:"e1q21ux93",label:"StyledSecLinkTitle"})(({theme})=>({display:"list-item",marginLeft:theme.variables.size.spacing.m,paddingTop:theme.variables.size.spacing.xxxs,paddingBottom:theme.variables.size.spacing.xxxs,"&:hover":{"> span":{textDecoration:"underline",color:theme.values.color.text.primary.default}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AA4C2B"} */"),StyledSecLinkContainerOnHover=styled("div",{target:"e1q21ux94",label:"StyledSecLinkContainerOnHover"})(({theme})=>({"> a":{textDecoration:"none",color:theme.values.color.text.tertiary.default},"&:hover":{[`${StyledSecLinkTitle}`]:{color:theme.values.color.text.tertiary.default}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AA0DsC"} */"),SecondaryTargets=({data})=>React.createElement(React.Fragment,null,data.map(({title,link:SecLink,body,"data-e2e-test-id":dataE2eTestId},i)=>{let key=`${title}-${i}`;return React.createElement(StyledSecLinkContainerOnHover,{key:key,"data-e2e-test-id":dataE2eTestId},React.createElement(SecLink,null,React.createElement(StyledSecLinkTitle,null,React.createElement(Text,{size:"m",color:"primary",as:"span"},React.createElement(StyledText,null,title))),body&&React.createElement(TextClamped,{size:"s",color:"tertiary",lines:2},React.createElement(StyledText,null,body))))})),BadgeWithBottomPadding=styled("div",{target:"e1q21ux95",label:"BadgeWithBottomPadding"})(({theme})=>({paddingBottom:theme.variables.size.spacing.xxxs}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AAsI+B"} */");export function SearchResult({title,subtitle,icon,details,body,secondaryTargets,link,labels,"data-e2e-test-id":dataE2eTestId}){let containerRef=useRef(null),spacing=subtitle||(null==labels?void 0:labels.length)?"xxs":"zero";return React.createElement(StyledResultContainer,{ref:containerRef,"data-ds-id":"SearchResult"},React.createElement(Columns,{gap:"m"},icon&&React.createElement(Column,{size:"narrow"},React.createElement(Icon,{name:icon,color:"tertiary"})),React.createElement(Column,{size:"fill"},React.createElement(Stack,{space:"xxs"},React.createElement(StyledLinkContainerOnHover,{"data-e2e-test-id":dataE2eTestId},React.createElement(link,null,React.createElement(Stack,{space:spacing},React.createElement(Columns,{gap:"xxs"},React.createElement(Column,{size:[12,12,"fill"]},React.createElement(StyledLinkTitle,null,React.createElement(Stack,{space:"zero"},React.createElement(TextClamped,{color:"accent",weight:"bold",lines:3,hyphens:"auto"},React.createElement(StyledText,null,title)),subtitle&&React.createElement(TextClamped,{color:"accent",lines:3,size:"s"},React.createElement(StyledText,null,subtitle))))),labels&&!!labels.length&&React.createElement(Column,{size:"narrow"},React.createElement(Inline,{space:"xs"},labels.map((label,i)=>{let key=`${i}${label}`;return React.createElement(BadgeWithBottomPadding,{key:key},React.createElement(Badge,{text:label}))})))),React.createElement(Stack,{space:"xxs"},details&&!!details.length&&React.createElement(Stack,{space:"zero"},details.slice(0,3).map(detail=>React.createElement(TextClamped,{color:"primary",size:"s",key:detail,lines:3},React.createElement(StyledText,null,detail)))),body&&React.createElement(TextClamped,{size:"s",lines:2},React.createElement(StyledText,null,body)))))),secondaryTargets&&!!secondaryTargets.length&&React.createElement(SecondaryTargets,{data:secondaryTargets})))))}SearchResult.defaultProps={icon:null,details:[],body:null,secondaryTargets:[],labels:[],"data-e2e-test-id":void 0};
|
|
1
|
+
import React,{useRef}from"react";import styled from"@emotion/styled";import{Columns,Column}from"../Column/Columns";import{Icon}from"../Icon/Icon";import{Inline}from"../Inline/Inline";import{Stack}from"../Stack/Stack";import{StyledText}from"../Typography/StyledText/StyledText";import{Badge}from"../Badge/Badge";import{TextClamped}from"../Typography/TextClamped/TextClamped";import{Text}from"../Typography/Text/Text";let StyledResultContainer=styled("div",{target:"e1q21ux90",label:"StyledResultContainer"})(()=>({position:"relative",textDecoration:"none"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AAuB8B"} */"),StyledLinkTitle=styled("div",{target:"e1q21ux91",label:"StyledLinkTitle"})(({theme})=>({color:theme.values.color.text.accent.default}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AA4BwB"} */"),StyledLinkContainerOnHover=styled("div",{target:"e1q21ux92",label:"StyledLinkContainerOnHover"})(({theme})=>({"> a":{textDecoration:"none",color:theme.values.color.text.accent.default},"&:hover":{[`${StyledLinkTitle}`]:{textDecoration:"underline"}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AAgCmC"} */"),StyledSecLinkTitle=styled("span",{target:"e1q21ux93",label:"StyledSecLinkTitle"})(({theme})=>({display:"list-item",marginLeft:theme.variables.size.spacing.m,paddingTop:theme.variables.size.spacing.xxxs,paddingBottom:theme.variables.size.spacing.xxxs,"&:hover":{"> span":{textDecoration:"underline",color:theme.values.color.text.primary.default}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AA4C2B"} */"),StyledSecLinkContainerOnHover=styled("div",{target:"e1q21ux94",label:"StyledSecLinkContainerOnHover"})(({theme})=>({"> a":{textDecoration:"none",color:theme.values.color.text.tertiary.default},"&:hover":{[`${StyledSecLinkTitle}`]:{color:theme.values.color.text.tertiary.default}}}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AA0DsC"} */"),SecondaryTargets=({data})=>React.createElement(React.Fragment,null,data.map(({title,link:SecLink,body,"data-e2e-test-id":dataE2eTestId},i)=>{let key=`${title}-${i}`;return React.createElement(StyledSecLinkContainerOnHover,{key:key,"data-e2e-test-id":dataE2eTestId},React.createElement(SecLink,null,React.createElement(StyledSecLinkTitle,null,React.createElement(Text,{size:"m",color:"primary",as:"span"},React.createElement(StyledText,null,title))),body&&React.createElement(TextClamped,{size:"s",color:"tertiary",lines:2},React.createElement(StyledText,null,body))))})),BadgeWithBottomPadding=styled("div",{target:"e1q21ux95",label:"BadgeWithBottomPadding"})(({theme})=>({paddingBottom:theme.variables.size.spacing.xxxs}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/SearchResult/SearchResult.tsx","sources":["src/components/SearchResult/SearchResult.tsx"],"sourcesContent":["import React, { useRef } from \"react\";\nimport styled from \"@emotion/styled\";\nimport { Columns, Column } from \"../Column/Columns\";\nimport type { IconName } from \"../Icon/Icon\";\nimport { Icon } from \"../Icon/Icon\";\nimport { Inline } from \"../Inline/Inline\";\nimport { Stack } from \"../Stack/Stack\";\nimport { StyledText } from \"../Typography/StyledText/StyledText\";\nimport { Badge } from \"../Badge/Badge\";\nimport { TextClamped } from \"../Typography/TextClamped/TextClamped\";\nimport { Text } from \"../Typography/Text/Text\";\n\nexport type SecondaryTarget = {\n  title: string;\n  body?: string;\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  \"data-e2e-test-id\"?: string;\n};\n\nexport type SecondaryTargetsProps = {\n  data: SecondaryTarget[];\n};\n\nconst StyledResultContainer = styled.div(() => ({\n  position: \"relative\",\n  textDecoration: \"none\",\n}));\n\nconst StyledLinkTitle = styled.div(({ theme }) => ({\n  color: theme.values.color.text.accent.default,\n}));\n\nconst StyledLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.accent.default,\n  },\n  \"&:hover\": {\n    [`${StyledLinkTitle}`]: {\n      textDecoration: \"underline\",\n    },\n  },\n}));\n\nconst StyledSecLinkTitle = styled.span(({ theme }) => ({\n  display: \"list-item\",\n  marginLeft: theme.variables.size.spacing.m,\n  paddingTop: theme.variables.size.spacing.xxxs,\n  paddingBottom: theme.variables.size.spacing.xxxs,\n  \"&:hover\": {\n    [`> span`]: {\n      textDecoration: \"underline\",\n      // this value controls text and its underline color\n      color: theme.values.color.text.primary.default,\n    },\n  },\n}));\n\nconst StyledSecLinkContainerOnHover = styled.div(({ theme }) => ({\n  \"> a\": {\n    textDecoration: \"none\",\n    color: theme.values.color.text.tertiary.default,\n  },\n  \"&:hover\": {\n    [`${StyledSecLinkTitle}`]: {\n      // this value controls the bolt point color\n      color: theme.values.color.text.tertiary.default,\n    },\n  },\n}));\n\nconst SecondaryTargets = ({ data }: SecondaryTargetsProps) => (\n  <>\n    {data.map(\n      (\n        { title, link: SecLink, body, \"data-e2e-test-id\": dataE2eTestId },\n        i\n      ) => {\n        const key = `${title}-${i}`;\n        return (\n          <StyledSecLinkContainerOnHover\n            key={key}\n            data-e2e-test-id={dataE2eTestId}\n          >\n            <SecLink>\n              <StyledSecLinkTitle>\n                <Text size=\"m\" color=\"primary\" as=\"span\">\n                  <StyledText>{title}</StyledText>\n                </Text>\n              </StyledSecLinkTitle>\n              {body && (\n                <TextClamped size=\"s\" color=\"tertiary\" lines={2}>\n                  <StyledText>{body}</StyledText>\n                </TextClamped>\n              )}\n            </SecLink>\n          </StyledSecLinkContainerOnHover>\n        );\n      }\n    )}\n  </>\n);\n\nexport type SearchResultProps = {\n  /** The result title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  title: string;\n  /** The subtitle that accompanies the title. Supports some nested HTML tags like <b> and <i>. Text gets clamped after 3 lines. */\n  subtitle?: string;\n  /** The primary AMBOSS target. */\n  link: (props: { children: React.ReactNode }) => React.ReactElement;\n  /** The left icon */\n  icon?: IconName;\n  /** List of details. Supports some nested HTML tags like <b> and <i>. */\n  details?: string[];\n  /** Usually a text content of the search result. Supports some nested HTML tags like <b> and <i>. Maximum 3 details items allowed where each test gets clamped after 3 lines. */\n  body?: string;\n  /** List of secondary AMBOSS targets with titles. Text gets clamped after 3 lines. */\n  secondaryTargets?: SecondaryTarget[];\n  /** List of labels. For example: preclinic, clinic, physician. */\n  labels?: string[];\n  /** The test id for the result's primary target. */\n  \"data-e2e-test-id\"?: string;\n};\n\nconst defaultProps: Partial<SearchResultProps> = {\n  icon: null,\n  details: [],\n  body: null,\n  secondaryTargets: [],\n  labels: [],\n  \"data-e2e-test-id\": undefined,\n};\n// this was a request by the designer to\n// have the backwards compatibility with the old search result\nconst BadgeWithBottomPadding = styled.div(({ theme }) => ({\n  paddingBottom: theme.variables.size.spacing.xxxs,\n}));\n\nexport function SearchResult({\n  title,\n  subtitle,\n  icon,\n  details,\n  body,\n  secondaryTargets,\n  link,\n  labels,\n  \"data-e2e-test-id\": dataE2eTestId,\n}: SearchResultProps): React.ReactElement {\n  const containerRef = useRef(null);\n\n  const MainLink = link;\n\n  const spacing = subtitle || labels?.length ? \"xxs\" : \"zero\";\n\n  return (\n    <StyledResultContainer ref={containerRef} data-ds-id=\"SearchResult\">\n      <Columns gap=\"m\">\n        {icon && (\n          <Column size=\"narrow\">\n            <Icon name={icon} color=\"tertiary\" />\n          </Column>\n        )}\n        <Column size=\"fill\">\n          <Stack space=\"xxs\">\n            <StyledLinkContainerOnHover data-e2e-test-id={dataE2eTestId}>\n              <MainLink>\n                <Stack space={spacing}>\n                  <Columns gap=\"xxs\">\n                    <Column size={[12, 12, \"fill\"]}>\n                      <StyledLinkTitle>\n                        <Stack space=\"zero\">\n                          <TextClamped\n                            color=\"accent\"\n                            weight=\"bold\"\n                            lines={3}\n                            hyphens=\"auto\"\n                          >\n                            <StyledText>{title}</StyledText>\n                          </TextClamped>\n                          {subtitle && (\n                            <TextClamped color=\"accent\" lines={3} size=\"s\">\n                              <StyledText>{subtitle}</StyledText>\n                            </TextClamped>\n                          )}\n                        </Stack>\n                      </StyledLinkTitle>\n                    </Column>\n                    {labels && !!labels.length && (\n                      <Column size=\"narrow\">\n                        <Inline space=\"xs\">\n                          {labels.map((label, i) => {\n                            const key = `${i}${label}`;\n                            return (\n                              <BadgeWithBottomPadding key={key}>\n                                <Badge text={label} />\n                              </BadgeWithBottomPadding>\n                            );\n                          })}\n                        </Inline>\n                      </Column>\n                    )}\n                  </Columns>\n                  <Stack space=\"xxs\">\n                    {details && !!details.length && (\n                      <Stack space=\"zero\">\n                        {details.slice(0, 3).map((detail) => (\n                          <TextClamped\n                            color=\"primary\"\n                            size=\"s\"\n                            key={detail}\n                            lines={3}\n                          >\n                            <StyledText>{detail}</StyledText>\n                          </TextClamped>\n                        ))}\n                      </Stack>\n                    )}\n                    {body && (\n                      <TextClamped size=\"s\" lines={2}>\n                        <StyledText>{body}</StyledText>\n                      </TextClamped>\n                    )}\n                  </Stack>\n                </Stack>\n              </MainLink>\n            </StyledLinkContainerOnHover>\n            {secondaryTargets && !!secondaryTargets.length && (\n              <SecondaryTargets data={secondaryTargets} />\n            )}\n          </Stack>\n        </Column>\n      </Columns>\n    </StyledResultContainer>\n  );\n}\n\nSearchResult.defaultProps = defaultProps;\n"],"names":[],"mappings":"AAsI+B"} */");export function SearchResult({title,subtitle,icon,details,body,secondaryTargets,link,labels,"data-e2e-test-id":dataE2eTestId}){let containerRef=useRef(null),spacing=subtitle||labels?.length?"xxs":"zero";return React.createElement(StyledResultContainer,{ref:containerRef,"data-ds-id":"SearchResult"},React.createElement(Columns,{gap:"m"},icon&&React.createElement(Column,{size:"narrow"},React.createElement(Icon,{name:icon,color:"tertiary"})),React.createElement(Column,{size:"fill"},React.createElement(Stack,{space:"xxs"},React.createElement(StyledLinkContainerOnHover,{"data-e2e-test-id":dataE2eTestId},React.createElement(link,null,React.createElement(Stack,{space:spacing},React.createElement(Columns,{gap:"xxs"},React.createElement(Column,{size:[12,12,"fill"]},React.createElement(StyledLinkTitle,null,React.createElement(Stack,{space:"zero"},React.createElement(TextClamped,{color:"accent",weight:"bold",lines:3,hyphens:"auto"},React.createElement(StyledText,null,title)),subtitle&&React.createElement(TextClamped,{color:"accent",lines:3,size:"s"},React.createElement(StyledText,null,subtitle))))),labels&&!!labels.length&&React.createElement(Column,{size:"narrow"},React.createElement(Inline,{space:"xs"},labels.map((label,i)=>{let key=`${i}${label}`;return React.createElement(BadgeWithBottomPadding,{key:key},React.createElement(Badge,{text:label}))})))),React.createElement(Stack,{space:"xxs"},details&&!!details.length&&React.createElement(Stack,{space:"zero"},details.slice(0,3).map(detail=>React.createElement(TextClamped,{color:"primary",size:"s",key:detail,lines:3},React.createElement(StyledText,null,detail)))),body&&React.createElement(TextClamped,{size:"s",lines:2},React.createElement(StyledText,null,body)))))),secondaryTargets&&!!secondaryTargets.length&&React.createElement(SecondaryTargets,{data:secondaryTargets})))))}SearchResult.defaultProps={icon:null,details:[],body:null,secondaryTargets:[],labels:[],"data-e2e-test-id":void 0};
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useState,useRef,useEffect,useCallback,useMemo}from"react";import styled from"@emotion/styled";import{TooltipContent}from"../Tooltip/TooltipContent";import{Sheet}from"../Sheet/Sheet";import{FocusTrapWrapper}from"../../shared/FocusTrapWrapper";import breakpoints from"../../web-tokens/_breakpoints.json";let StyledContainer=styled("div",{target:"e1afbh70",label:"StyledContainer"})(()=>({borderRadius:"inherit"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Toggletip/BasePopover.tsx","sources":["src/components/Toggletip/BasePopover.tsx"],"sourcesContent":["import type { ReactElement, PropsWithChildren } from \"react\";\nimport React, {\n  useState,\n  useRef,\n  useEffect,\n  useCallback,\n  useMemo,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { TooltipContentProps } from \"../Tooltip/TooltipContent\";\nimport { TooltipContent } from \"../Tooltip/TooltipContent\";\nimport type { TooltipConditionalProps } from \"../Tooltip/types\";\nimport { Sheet } from \"../Sheet/Sheet\";\nimport { FocusTrapWrapper } from \"../../shared/FocusTrapWrapper\";\nimport breakpoints from \"../../web-tokens/_breakpoints.json\";\n\ntype BaseProps = Pick<\n  TooltipContentProps,\n  | \"placement\"\n  | \"portalContainer\"\n  | \"maxWidth\"\n  | \"contentPadding\"\n  | \"hideArrow\"\n  | \"subTheme\"\n  | \"defaultVerticalPlacement\"\n> & {\n  name?: string;\n  // Popover content\n  content: ReactElement;\n  \"data-e2e-test-id\"?: string;\n  // Programmatically toggle Popover visibility with this prop\n  isVisible?: boolean;\n  /* Called when tooltip appears and disappears */\n  onVisibilityChange?: (isVisible: boolean, reason: string) => void;\n  /* Controls whether BasePopover  closes on outside click */\n  dismissOnOutsideClick?: boolean;\n  /* Option for focus-trap, controls whether the first focuable item recieves focus */\n  disableInitialFocus?: boolean;\n  /* Option for focus-trap, controls whether the trigger should receive back the focus on popover close */\n  disableReturnFocusToTrigger?: boolean;\n  /* Render as sheet on mobile web */\n  renderAsSheetOnMobile?: boolean;\n};\n\nexport type BasePopoverProps = BaseProps & TooltipConditionalProps;\n\nconst StyledContainer = styled.div(() => ({\n  borderRadius: \"inherit\",\n}));\n\nconst FocusTrapContent = React.forwardRef<\n  HTMLDivElement,\n  PropsWithChildren<unknown>\n>(({ children }, ref) => (\n  <StyledContainer ref={ref}>{children}</StyledContainer>\n));\n\nconst VisibilityChangeReason = {\n  triggerClick: \"triggerClick\",\n  outsideClick: \"outsideClick\",\n};\n\nexport function BasePopover({\n  placement = \"auto\",\n  content,\n  children,\n  contentPadding = \"m\",\n  maxWidth,\n  externalTriggerRef,\n  portalContainer,\n  name = \"Popover\",\n  isVisible: isPopoverVisible,\n  dismissOnOutsideClick = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  subTheme,\n  defaultVerticalPlacement,\n  onVisibilityChange,\n  disableInitialFocus = false,\n  disableReturnFocusToTrigger = false,\n  renderAsSheetOnMobile = false,\n  ...restContentProps\n}: BasePopoverProps): React.ReactElement {\n  const tooltipId = useMemo(\n    () => `DS${name}_${Math.floor(Date.now() * Math.random())}`,\n    [name]\n  );\n  const [isVisible, setVisible] = useState(isPopoverVisible);\n  const [isMobileBreakPoint, setIsMobileBreakpoint] = useState(false);\n  const internalTriggerRef = useRef(null);\n  const triggerRef = externalTriggerRef || internalTriggerRef;\n  const isOutsideClickOnTrigger = useRef(false);\n  const isPrevMobileBreakPointRef = useRef(false);\n\n  const toggleVisibility = useCallback(\n    (status: boolean, reason: string) => {\n      setVisible(status);\n\n      if (onVisibilityChange) {\n        onVisibilityChange(status, reason);\n      }\n    },\n    [onVisibilityChange]\n  );\n\n  // Outside click is also fired when the Popover is open and trigger is clicked. `isOutsideClickOnTrigger` saves this condition and we check for it so as to not toggle the Popover twice.\n  const handleTriggerClick = useCallback(() => {\n    if (!isOutsideClickOnTrigger.current) {\n      toggleVisibility(!isVisible, VisibilityChangeReason.triggerClick);\n    } else {\n      // reset this value so that Popover can open in next click\n      isOutsideClickOnTrigger.current = false;\n    }\n  }, [toggleVisibility, isVisible]);\n\n  const handleClickOutsideDeactivates = useCallback(\n    (evt) => {\n      if (triggerRef.current.contains(evt.target)) {\n        isOutsideClickOnTrigger.current = true;\n      }\n\n      return true;\n    },\n    [triggerRef, isOutsideClickOnTrigger]\n  );\n\n  const handlePostDeactivate = useCallback(() => {\n    const reason = isOutsideClickOnTrigger.current\n      ? VisibilityChangeReason.triggerClick\n      : VisibilityChangeReason.outsideClick;\n\n    toggleVisibility(false, reason);\n  }, [toggleVisibility]);\n\n  useEffect(() => {\n    setVisible(isPopoverVisible);\n  }, [isPopoverVisible]);\n\n  useEffect(() => {\n    // Check if this is a mobile breakpoint\n    if (renderAsSheetOnMobile) {\n      setIsMobileBreakpoint(window.innerWidth <= breakpoints.medium.value);\n    }\n  }, [isVisible, renderAsSheetOnMobile]);\n\n  useEffect(() => {\n    isPrevMobileBreakPointRef.current = isMobileBreakPoint;\n  }, [isMobileBreakPoint]);\n\n  useEffect(() => {\n    let trigger: HTMLElement;\n\n    if (externalTriggerRef?.current && !children) {\n      trigger = externalTriggerRef.current;\n\n      trigger.setAttribute(\"tabindex\", \"0\");\n      trigger.addEventListener(\"click\", handleTriggerClick);\n    }\n\n    return () => {\n      if (trigger) {\n        trigger.removeEventListener(\"click\", handleTriggerClick);\n      }\n    };\n  }, [externalTriggerRef, children, handleTriggerClick]);\n\n  useEffect(() => {\n    if (externalTriggerRef?.current && !children) {\n      const trigger = externalTriggerRef.current;\n\n      if (isVisible) {\n        trigger.setAttribute(\"aria-expanded\", true);\n        trigger.setAttribute(\"aria-controls\", tooltipId);\n      } else {\n        trigger.removeAttribute(\"aria-expanded\");\n        trigger.removeAttribute(\"aria-controls\");\n      }\n    }\n  }, [externalTriggerRef, children, tooltipId, isVisible]);\n\n  const triggerElm = children\n    ? React.cloneElement(children, {\n        ref: triggerRef,\n        ...(isVisible && {\n          \"aria-expanded\": true,\n          \"aria-controls\": tooltipId,\n        }),\n        tabIndex: 0,\n        onClick: (evt: React.MouseEvent) => {\n          handleTriggerClick();\n          if (children.props.onClick) {\n            children.props.onClick(evt);\n          }\n        },\n      })\n    : null;\n\n  if (isMobileBreakPoint) {\n    // render as Sheet\n    const sheetElm = (\n      <Sheet\n        id={tooltipId}\n        isVisible={isVisible}\n        portalContainer={portalContainer}\n        dismissOnOutsideClick={dismissOnOutsideClick}\n        disableInitialFocus={disableInitialFocus}\n        disableReturnFocusToTrigger={disableReturnFocusToTrigger}\n        onClose={handlePostDeactivate}\n        onClickOutsideDeactivates={handleClickOutsideDeactivates}\n      >\n        {content}\n      </Sheet>\n    );\n\n    return (\n      <>\n        {triggerElm}\n        {sheetElm}\n      </>\n    );\n  }\n\n  // render as Popover\n  const contentElm = (\n    <FocusTrapWrapper\n      focusTrapOptions={{\n        clickOutsideDeactivates:\n          dismissOnOutsideClick && handleClickOutsideDeactivates, // de-active focus trap on outside click\n        allowOutsideClick: true,\n        escapeDeactivates: true, // de-activate focus trap on escape key\n        fallbackFocus: `#${tooltipId}`, // set focus to tooltip content container if it has no focusable element\n        onPostDeactivate: handlePostDeactivate,\n        preventScroll: true,\n        initialFocus: () => !disableInitialFocus,\n        returnFocusOnDeactivate: !disableReturnFocusToTrigger,\n      }}\n    >\n      <FocusTrapContent>{content}</FocusTrapContent>\n    </FocusTrapWrapper>\n  );\n\n  const tooltipElm = (\n    <TooltipContent\n      {...restContentProps} // eslint-disable-line react/jsx-props-no-spreading\n      defaultVerticalPlacement={defaultVerticalPlacement}\n      dataDSId={name}\n      content={contentElm}\n      contentPadding={contentPadding}\n      maxWidth={maxWidth}\n      placement={placement}\n      portalContainer={portalContainer}\n      dataE2eTestId={dataE2eTestId}\n      subTheme={subTheme}\n      isVisible={isVisible}\n      tooltipId={tooltipId}\n      triggerRef={triggerRef}\n    />\n  );\n\n  return (\n    <>\n      {triggerElm}\n      {tooltipElm}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA8CwB"} */"),FocusTrapContent=React.forwardRef(({children},ref)=>React.createElement(StyledContainer,{ref:ref},children)),VisibilityChangeReason={triggerClick:"triggerClick",outsideClick:"outsideClick"};export function BasePopover({placement="auto",content,children,contentPadding="m",maxWidth,externalTriggerRef,portalContainer,name="Popover",isVisible:isPopoverVisible,dismissOnOutsideClick=!0,"data-e2e-test-id":dataE2eTestId,subTheme,defaultVerticalPlacement,onVisibilityChange,disableInitialFocus=!1,disableReturnFocusToTrigger=!1,renderAsSheetOnMobile=!1,...restContentProps}){let tooltipId=useMemo(()=>`DS${name}_${Math.floor(Date.now()*Math.random())}`,[name]),[isVisible,setVisible]=useState(isPopoverVisible),[isMobileBreakPoint,setIsMobileBreakpoint]=useState(!1),internalTriggerRef=useRef(null),triggerRef=externalTriggerRef||internalTriggerRef,isOutsideClickOnTrigger=useRef(!1),isPrevMobileBreakPointRef=useRef(!1),toggleVisibility=useCallback((status,reason)=>{setVisible(status),onVisibilityChange&&onVisibilityChange(status,reason)},[onVisibilityChange]),handleTriggerClick=useCallback(()=>{isOutsideClickOnTrigger.current?isOutsideClickOnTrigger.current=!1:toggleVisibility(!isVisible,VisibilityChangeReason.triggerClick)},[toggleVisibility,isVisible]),handleClickOutsideDeactivates=useCallback(evt=>(triggerRef.current.contains(evt.target)&&(isOutsideClickOnTrigger.current=!0),!0),[triggerRef,isOutsideClickOnTrigger]),handlePostDeactivate=useCallback(()=>{toggleVisibility(!1,isOutsideClickOnTrigger.current?VisibilityChangeReason.triggerClick:VisibilityChangeReason.outsideClick)},[toggleVisibility]);useEffect(()=>{setVisible(isPopoverVisible)},[isPopoverVisible]),useEffect(()=>{renderAsSheetOnMobile&&setIsMobileBreakpoint(window.innerWidth<=breakpoints.medium.value)},[isVisible,renderAsSheetOnMobile]),useEffect(()=>{isPrevMobileBreakPointRef.current=isMobileBreakPoint},[isMobileBreakPoint]),useEffect(()=>{let trigger;return(null==externalTriggerRef?void 0:externalTriggerRef.current)&&!children&&((trigger=externalTriggerRef.current).setAttribute("tabindex","0"),trigger.addEventListener("click",handleTriggerClick)),()=>{trigger&&trigger.removeEventListener("click",handleTriggerClick)}},[externalTriggerRef,children,handleTriggerClick]),useEffect(()=>{if((null==externalTriggerRef?void 0:externalTriggerRef.current)&&!children){let trigger=externalTriggerRef.current;isVisible?(trigger.setAttribute("aria-expanded",!0),trigger.setAttribute("aria-controls",tooltipId)):(trigger.removeAttribute("aria-expanded"),trigger.removeAttribute("aria-controls"))}},[externalTriggerRef,children,tooltipId,isVisible]);let triggerElm=children?React.cloneElement(children,{ref:triggerRef,...isVisible&&{"aria-expanded":!0,"aria-controls":tooltipId},tabIndex:0,onClick:evt=>{handleTriggerClick(),children.props.onClick&&children.props.onClick(evt)}}):null;if(isMobileBreakPoint){let sheetElm=React.createElement(Sheet,{id:tooltipId,isVisible:isVisible,portalContainer:portalContainer,dismissOnOutsideClick:dismissOnOutsideClick,disableInitialFocus:disableInitialFocus,disableReturnFocusToTrigger:disableReturnFocusToTrigger,onClose:handlePostDeactivate,onClickOutsideDeactivates:handleClickOutsideDeactivates},content);return React.createElement(React.Fragment,null,triggerElm,sheetElm)}let contentElm=React.createElement(FocusTrapWrapper,{focusTrapOptions:{clickOutsideDeactivates:dismissOnOutsideClick&&handleClickOutsideDeactivates,allowOutsideClick:!0,escapeDeactivates:!0,fallbackFocus:`#${tooltipId}`,onPostDeactivate:handlePostDeactivate,preventScroll:!0,initialFocus:()=>!disableInitialFocus,returnFocusOnDeactivate:!disableReturnFocusToTrigger}},React.createElement(FocusTrapContent,null,content)),tooltipElm=React.createElement(TooltipContent,{...restContentProps,defaultVerticalPlacement:defaultVerticalPlacement,dataDSId:name,content:contentElm,contentPadding:contentPadding,maxWidth:maxWidth,placement:placement,portalContainer:portalContainer,dataE2eTestId:dataE2eTestId,subTheme:subTheme,isVisible:isVisible,tooltipId:tooltipId,triggerRef:triggerRef});return React.createElement(React.Fragment,null,triggerElm,tooltipElm)}
|
|
1
|
+
import React,{useState,useRef,useEffect,useCallback,useMemo}from"react";import styled from"@emotion/styled";import{TooltipContent}from"../Tooltip/TooltipContent";import{Sheet}from"../Sheet/Sheet";import{FocusTrapWrapper}from"../../shared/FocusTrapWrapper";import breakpoints from"../../web-tokens/_breakpoints.json";let StyledContainer=styled("div",{target:"e1afbh70",label:"StyledContainer"})(()=>({borderRadius:"inherit"}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"src/components/Toggletip/BasePopover.tsx","sources":["src/components/Toggletip/BasePopover.tsx"],"sourcesContent":["import type { ReactElement, PropsWithChildren } from \"react\";\nimport React, {\n  useState,\n  useRef,\n  useEffect,\n  useCallback,\n  useMemo,\n} from \"react\";\nimport styled from \"@emotion/styled\";\nimport type { TooltipContentProps } from \"../Tooltip/TooltipContent\";\nimport { TooltipContent } from \"../Tooltip/TooltipContent\";\nimport type { TooltipConditionalProps } from \"../Tooltip/types\";\nimport { Sheet } from \"../Sheet/Sheet\";\nimport { FocusTrapWrapper } from \"../../shared/FocusTrapWrapper\";\nimport breakpoints from \"../../web-tokens/_breakpoints.json\";\n\ntype BaseProps = Pick<\n  TooltipContentProps,\n  | \"placement\"\n  | \"portalContainer\"\n  | \"maxWidth\"\n  | \"contentPadding\"\n  | \"hideArrow\"\n  | \"subTheme\"\n  | \"defaultVerticalPlacement\"\n> & {\n  name?: string;\n  // Popover content\n  content: ReactElement;\n  \"data-e2e-test-id\"?: string;\n  // Programmatically toggle Popover visibility with this prop\n  isVisible?: boolean;\n  /* Called when tooltip appears and disappears */\n  onVisibilityChange?: (isVisible: boolean, reason: string) => void;\n  /* Controls whether BasePopover  closes on outside click */\n  dismissOnOutsideClick?: boolean;\n  /* Option for focus-trap, controls whether the first focuable item recieves focus */\n  disableInitialFocus?: boolean;\n  /* Option for focus-trap, controls whether the trigger should receive back the focus on popover close */\n  disableReturnFocusToTrigger?: boolean;\n  /* Render as sheet on mobile web */\n  renderAsSheetOnMobile?: boolean;\n};\n\nexport type BasePopoverProps = BaseProps & TooltipConditionalProps;\n\nconst StyledContainer = styled.div(() => ({\n  borderRadius: \"inherit\",\n}));\n\nconst FocusTrapContent = React.forwardRef<\n  HTMLDivElement,\n  PropsWithChildren<unknown>\n>(({ children }, ref) => (\n  <StyledContainer ref={ref}>{children}</StyledContainer>\n));\n\nconst VisibilityChangeReason = {\n  triggerClick: \"triggerClick\",\n  outsideClick: \"outsideClick\",\n};\n\nexport function BasePopover({\n  placement = \"auto\",\n  content,\n  children,\n  contentPadding = \"m\",\n  maxWidth,\n  externalTriggerRef,\n  portalContainer,\n  name = \"Popover\",\n  isVisible: isPopoverVisible,\n  dismissOnOutsideClick = true,\n  \"data-e2e-test-id\": dataE2eTestId,\n  subTheme,\n  defaultVerticalPlacement,\n  onVisibilityChange,\n  disableInitialFocus = false,\n  disableReturnFocusToTrigger = false,\n  renderAsSheetOnMobile = false,\n  ...restContentProps\n}: BasePopoverProps): React.ReactElement {\n  const tooltipId = useMemo(\n    () => `DS${name}_${Math.floor(Date.now() * Math.random())}`,\n    [name]\n  );\n  const [isVisible, setVisible] = useState(isPopoverVisible);\n  const [isMobileBreakPoint, setIsMobileBreakpoint] = useState(false);\n  const internalTriggerRef = useRef(null);\n  const triggerRef = externalTriggerRef || internalTriggerRef;\n  const isOutsideClickOnTrigger = useRef(false);\n  const isPrevMobileBreakPointRef = useRef(false);\n\n  const toggleVisibility = useCallback(\n    (status: boolean, reason: string) => {\n      setVisible(status);\n\n      if (onVisibilityChange) {\n        onVisibilityChange(status, reason);\n      }\n    },\n    [onVisibilityChange]\n  );\n\n  // Outside click is also fired when the Popover is open and trigger is clicked. `isOutsideClickOnTrigger` saves this condition and we check for it so as to not toggle the Popover twice.\n  const handleTriggerClick = useCallback(() => {\n    if (!isOutsideClickOnTrigger.current) {\n      toggleVisibility(!isVisible, VisibilityChangeReason.triggerClick);\n    } else {\n      // reset this value so that Popover can open in next click\n      isOutsideClickOnTrigger.current = false;\n    }\n  }, [toggleVisibility, isVisible]);\n\n  const handleClickOutsideDeactivates = useCallback(\n    (evt) => {\n      if (triggerRef.current.contains(evt.target)) {\n        isOutsideClickOnTrigger.current = true;\n      }\n\n      return true;\n    },\n    [triggerRef, isOutsideClickOnTrigger]\n  );\n\n  const handlePostDeactivate = useCallback(() => {\n    const reason = isOutsideClickOnTrigger.current\n      ? VisibilityChangeReason.triggerClick\n      : VisibilityChangeReason.outsideClick;\n\n    toggleVisibility(false, reason);\n  }, [toggleVisibility]);\n\n  useEffect(() => {\n    setVisible(isPopoverVisible);\n  }, [isPopoverVisible]);\n\n  useEffect(() => {\n    // Check if this is a mobile breakpoint\n    if (renderAsSheetOnMobile) {\n      setIsMobileBreakpoint(window.innerWidth <= breakpoints.medium.value);\n    }\n  }, [isVisible, renderAsSheetOnMobile]);\n\n  useEffect(() => {\n    isPrevMobileBreakPointRef.current = isMobileBreakPoint;\n  }, [isMobileBreakPoint]);\n\n  useEffect(() => {\n    let trigger: HTMLElement;\n\n    if (externalTriggerRef?.current && !children) {\n      trigger = externalTriggerRef.current;\n\n      trigger.setAttribute(\"tabindex\", \"0\");\n      trigger.addEventListener(\"click\", handleTriggerClick);\n    }\n\n    return () => {\n      if (trigger) {\n        trigger.removeEventListener(\"click\", handleTriggerClick);\n      }\n    };\n  }, [externalTriggerRef, children, handleTriggerClick]);\n\n  useEffect(() => {\n    if (externalTriggerRef?.current && !children) {\n      const trigger = externalTriggerRef.current;\n\n      if (isVisible) {\n        trigger.setAttribute(\"aria-expanded\", true);\n        trigger.setAttribute(\"aria-controls\", tooltipId);\n      } else {\n        trigger.removeAttribute(\"aria-expanded\");\n        trigger.removeAttribute(\"aria-controls\");\n      }\n    }\n  }, [externalTriggerRef, children, tooltipId, isVisible]);\n\n  const triggerElm = children\n    ? React.cloneElement(children, {\n        ref: triggerRef,\n        ...(isVisible && {\n          \"aria-expanded\": true,\n          \"aria-controls\": tooltipId,\n        }),\n        tabIndex: 0,\n        onClick: (evt: React.MouseEvent) => {\n          handleTriggerClick();\n          if (children.props.onClick) {\n            children.props.onClick(evt);\n          }\n        },\n      })\n    : null;\n\n  if (isMobileBreakPoint) {\n    // render as Sheet\n    const sheetElm = (\n      <Sheet\n        id={tooltipId}\n        isVisible={isVisible}\n        portalContainer={portalContainer}\n        dismissOnOutsideClick={dismissOnOutsideClick}\n        disableInitialFocus={disableInitialFocus}\n        disableReturnFocusToTrigger={disableReturnFocusToTrigger}\n        onClose={handlePostDeactivate}\n        onClickOutsideDeactivates={handleClickOutsideDeactivates}\n      >\n        {content}\n      </Sheet>\n    );\n\n    return (\n      <>\n        {triggerElm}\n        {sheetElm}\n      </>\n    );\n  }\n\n  // render as Popover\n  const contentElm = (\n    <FocusTrapWrapper\n      focusTrapOptions={{\n        clickOutsideDeactivates:\n          dismissOnOutsideClick && handleClickOutsideDeactivates, // de-active focus trap on outside click\n        allowOutsideClick: true,\n        escapeDeactivates: true, // de-activate focus trap on escape key\n        fallbackFocus: `#${tooltipId}`, // set focus to tooltip content container if it has no focusable element\n        onPostDeactivate: handlePostDeactivate,\n        preventScroll: true,\n        initialFocus: () => !disableInitialFocus,\n        returnFocusOnDeactivate: !disableReturnFocusToTrigger,\n      }}\n    >\n      <FocusTrapContent>{content}</FocusTrapContent>\n    </FocusTrapWrapper>\n  );\n\n  const tooltipElm = (\n    <TooltipContent\n      {...restContentProps} // eslint-disable-line react/jsx-props-no-spreading\n      defaultVerticalPlacement={defaultVerticalPlacement}\n      dataDSId={name}\n      content={contentElm}\n      contentPadding={contentPadding}\n      maxWidth={maxWidth}\n      placement={placement}\n      portalContainer={portalContainer}\n      dataE2eTestId={dataE2eTestId}\n      subTheme={subTheme}\n      isVisible={isVisible}\n      tooltipId={tooltipId}\n      triggerRef={triggerRef}\n    />\n  );\n\n  return (\n    <>\n      {triggerElm}\n      {tooltipElm}\n    </>\n  );\n}\n"],"names":[],"mappings":"AA8CwB"} */"),FocusTrapContent=React.forwardRef(({children},ref)=>React.createElement(StyledContainer,{ref:ref},children)),VisibilityChangeReason={triggerClick:"triggerClick",outsideClick:"outsideClick"};export function BasePopover({placement="auto",content,children,contentPadding="m",maxWidth,externalTriggerRef,portalContainer,name="Popover",isVisible:isPopoverVisible,dismissOnOutsideClick=!0,"data-e2e-test-id":dataE2eTestId,subTheme,defaultVerticalPlacement,onVisibilityChange,disableInitialFocus=!1,disableReturnFocusToTrigger=!1,renderAsSheetOnMobile=!1,...restContentProps}){let tooltipId=useMemo(()=>`DS${name}_${Math.floor(Date.now()*Math.random())}`,[name]),[isVisible,setVisible]=useState(isPopoverVisible),[isMobileBreakPoint,setIsMobileBreakpoint]=useState(!1),internalTriggerRef=useRef(null),triggerRef=externalTriggerRef||internalTriggerRef,isOutsideClickOnTrigger=useRef(!1),isPrevMobileBreakPointRef=useRef(!1),toggleVisibility=useCallback((status,reason)=>{setVisible(status),onVisibilityChange&&onVisibilityChange(status,reason)},[onVisibilityChange]),handleTriggerClick=useCallback(()=>{isOutsideClickOnTrigger.current?isOutsideClickOnTrigger.current=!1:toggleVisibility(!isVisible,VisibilityChangeReason.triggerClick)},[toggleVisibility,isVisible]),handleClickOutsideDeactivates=useCallback(evt=>(triggerRef.current.contains(evt.target)&&(isOutsideClickOnTrigger.current=!0),!0),[triggerRef,isOutsideClickOnTrigger]),handlePostDeactivate=useCallback(()=>{toggleVisibility(!1,isOutsideClickOnTrigger.current?VisibilityChangeReason.triggerClick:VisibilityChangeReason.outsideClick)},[toggleVisibility]);useEffect(()=>{setVisible(isPopoverVisible)},[isPopoverVisible]),useEffect(()=>{renderAsSheetOnMobile&&setIsMobileBreakpoint(window.innerWidth<=breakpoints.medium.value)},[isVisible,renderAsSheetOnMobile]),useEffect(()=>{isPrevMobileBreakPointRef.current=isMobileBreakPoint},[isMobileBreakPoint]),useEffect(()=>{let trigger;return externalTriggerRef?.current&&!children&&((trigger=externalTriggerRef.current).setAttribute("tabindex","0"),trigger.addEventListener("click",handleTriggerClick)),()=>{trigger&&trigger.removeEventListener("click",handleTriggerClick)}},[externalTriggerRef,children,handleTriggerClick]),useEffect(()=>{if(externalTriggerRef?.current&&!children){let trigger=externalTriggerRef.current;isVisible?(trigger.setAttribute("aria-expanded",!0),trigger.setAttribute("aria-controls",tooltipId)):(trigger.removeAttribute("aria-expanded"),trigger.removeAttribute("aria-controls"))}},[externalTriggerRef,children,tooltipId,isVisible]);let triggerElm=children?React.cloneElement(children,{ref:triggerRef,...isVisible&&{"aria-expanded":!0,"aria-controls":tooltipId},tabIndex:0,onClick:evt=>{handleTriggerClick(),children.props.onClick&&children.props.onClick(evt)}}):null;if(isMobileBreakPoint){let sheetElm=React.createElement(Sheet,{id:tooltipId,isVisible:isVisible,portalContainer:portalContainer,dismissOnOutsideClick:dismissOnOutsideClick,disableInitialFocus:disableInitialFocus,disableReturnFocusToTrigger:disableReturnFocusToTrigger,onClose:handlePostDeactivate,onClickOutsideDeactivates:handleClickOutsideDeactivates},content);return React.createElement(React.Fragment,null,triggerElm,sheetElm)}let contentElm=React.createElement(FocusTrapWrapper,{focusTrapOptions:{clickOutsideDeactivates:dismissOnOutsideClick&&handleClickOutsideDeactivates,allowOutsideClick:!0,escapeDeactivates:!0,fallbackFocus:`#${tooltipId}`,onPostDeactivate:handlePostDeactivate,preventScroll:!0,initialFocus:()=>!disableInitialFocus,returnFocusOnDeactivate:!disableReturnFocusToTrigger}},React.createElement(FocusTrapContent,null,content)),tooltipElm=React.createElement(TooltipContent,{...restContentProps,defaultVerticalPlacement:defaultVerticalPlacement,dataDSId:name,content:contentElm,contentPadding:contentPadding,maxWidth:maxWidth,placement:placement,portalContainer:portalContainer,dataE2eTestId:dataE2eTestId,subTheme:subTheme,isVisible:isVisible,tooltipId:tooltipId,triggerRef:triggerRef});return React.createElement(React.Fragment,null,triggerElm,tooltipElm)}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import React,{useState,useRef,useEffect,useCallback,useMemo}from"react";import{TooltipContent}from"./TooltipContent";export function BaseTooltip({placement="auto",content,children,externalTriggerRef,portalContainer,contentPadding,maxWidth,subTheme,"data-e2e-test-id":dataE2eTestId,onVisibilityChange}){let tooltipId=useMemo(()=>`DSTooltip_${Math.floor(Date.now()*Math.random())}`,[]),[isVisible,setVisible]=useState(!1),triggeredByEvent=useRef(null),isTooltipHovered=useRef(!1),isTriggerHovered=useRef(!1),hideTooltipTimeoutId=useRef(null),showTooltipTimeoutId=useRef(null),internalTriggerRef=useRef(null),triggerRef=externalTriggerRef||internalTriggerRef,toggleVisibility=useCallback(status=>{setVisible(status),onVisibilityChange&&onVisibilityChange(status)},[onVisibilityChange]),handleTriggerPointerEnter=useCallback(()=>{isTriggerHovered.current=!0,isTooltipHovered.current||(clearTimeout(showTooltipTimeoutId.current),showTooltipTimeoutId.current=setTimeout(()=>{isTriggerHovered.current&&(triggeredByEvent.current="hover",toggleVisibility(!0))},200))},[toggleVisibility]),handleTriggerPointerLeave=useCallback(()=>{isTriggerHovered.current=!1,clearTimeout(hideTooltipTimeoutId.current),hideTooltipTimeoutId.current=setTimeout(()=>{isTooltipHovered.current||"hover"!==triggeredByEvent.current||isTriggerHovered.current||toggleVisibility(!1)},200)},[toggleVisibility]),handleTriggerFocus=useCallback(()=>{triggeredByEvent.current="focus",toggleVisibility(!0)},[toggleVisibility]),handleTriggerBlur=useCallback(()=>{"focus"===triggeredByEvent.current&&toggleVisibility(!1)},[toggleVisibility]),handleTriggerKeyDown=useCallback(evt=>{"Escape"===evt.key&&toggleVisibility(!1)},[toggleVisibility]);useEffect(()=>()=>{clearTimeout(showTooltipTimeoutId.current),clearTimeout(hideTooltipTimeoutId.current)},[]),useEffect(()=>{let trigger;return
|
|
1
|
+
import React,{useState,useRef,useEffect,useCallback,useMemo}from"react";import{TooltipContent}from"./TooltipContent";export function BaseTooltip({placement="auto",content,children,externalTriggerRef,portalContainer,contentPadding,maxWidth,subTheme,"data-e2e-test-id":dataE2eTestId,onVisibilityChange}){let tooltipId=useMemo(()=>`DSTooltip_${Math.floor(Date.now()*Math.random())}`,[]),[isVisible,setVisible]=useState(!1),triggeredByEvent=useRef(null),isTooltipHovered=useRef(!1),isTriggerHovered=useRef(!1),hideTooltipTimeoutId=useRef(null),showTooltipTimeoutId=useRef(null),internalTriggerRef=useRef(null),triggerRef=externalTriggerRef||internalTriggerRef,toggleVisibility=useCallback(status=>{setVisible(status),onVisibilityChange&&onVisibilityChange(status)},[onVisibilityChange]),handleTriggerPointerEnter=useCallback(()=>{isTriggerHovered.current=!0,isTooltipHovered.current||(clearTimeout(showTooltipTimeoutId.current),showTooltipTimeoutId.current=setTimeout(()=>{isTriggerHovered.current&&(triggeredByEvent.current="hover",toggleVisibility(!0))},200))},[toggleVisibility]),handleTriggerPointerLeave=useCallback(()=>{isTriggerHovered.current=!1,clearTimeout(hideTooltipTimeoutId.current),hideTooltipTimeoutId.current=setTimeout(()=>{isTooltipHovered.current||"hover"!==triggeredByEvent.current||isTriggerHovered.current||toggleVisibility(!1)},200)},[toggleVisibility]),handleTriggerFocus=useCallback(()=>{triggeredByEvent.current="focus",toggleVisibility(!0)},[toggleVisibility]),handleTriggerBlur=useCallback(()=>{"focus"===triggeredByEvent.current&&toggleVisibility(!1)},[toggleVisibility]),handleTriggerKeyDown=useCallback(evt=>{"Escape"===evt.key&&toggleVisibility(!1)},[toggleVisibility]);useEffect(()=>()=>{clearTimeout(showTooltipTimeoutId.current),clearTimeout(hideTooltipTimeoutId.current)},[]),useEffect(()=>{let trigger;return externalTriggerRef?.current&&!children&&((trigger=externalTriggerRef.current).setAttribute("tabindex","0"),trigger.addEventListener("pointerenter",handleTriggerPointerEnter),trigger.addEventListener("pointerleave",handleTriggerPointerLeave),trigger.addEventListener("focus",handleTriggerFocus),trigger.addEventListener("blur",handleTriggerBlur),trigger.addEventListener("keydown",handleTriggerKeyDown)),()=>{trigger&&(trigger.removeEventListener("pointerenter",handleTriggerPointerEnter),trigger.removeEventListener("pointerleave",handleTriggerPointerLeave),trigger.removeEventListener("focus",handleTriggerFocus),trigger.removeEventListener("blur",handleTriggerBlur),trigger.removeEventListener("keydown",handleTriggerKeyDown))}},[externalTriggerRef,children,handleTriggerPointerEnter,handleTriggerPointerLeave,handleTriggerFocus,handleTriggerBlur,handleTriggerKeyDown]),useEffect(()=>{if(externalTriggerRef?.current&&!children){let trigger=externalTriggerRef.current;isVisible?trigger.setAttribute("aria-describedby",tooltipId):trigger.removeAttribute("aria-describedby")}},[externalTriggerRef,children,tooltipId,isVisible]);let triggerElm=children?React.cloneElement(children,{ref:triggerRef,...isVisible&&{"aria-describedby":tooltipId},tabIndex:0,onPointerEnter:handleTriggerPointerEnter,onPointerLeave:handleTriggerPointerLeave,onFocus:evt=>{handleTriggerFocus(),children.props.onFocus&&children.props.onFocus(evt)},onBlur:evt=>{handleTriggerBlur(),children.props.onBlur&&children.props.onBlur&&children.props.onBlur(evt)},onKeyDown:handleTriggerKeyDown}):null,tooltipElm=React.createElement(TooltipContent,{dataDSId:"Tooltip",content:content,placement:placement,portalContainer:portalContainer,dataE2eTestId:dataE2eTestId,isVisible:isVisible,tooltipId:tooltipId,triggerRef:triggerRef,"aria-hidden":!0,role:"tooltip",contentPadding:contentPadding,maxWidth:maxWidth,subTheme:subTheme,onTooltipPointerEnter:()=>{isTooltipHovered.current=!0},onTooltipPointerLeave:()=>{isTooltipHovered.current=!1,"hover"===triggeredByEvent.current&&toggleVisibility(!1)}});return React.createElement(React.Fragment,null,triggerElm,tooltipElm)}
|
|
@@ -7,5 +7,6 @@ export type VirtualScrollListProps = {
|
|
|
7
7
|
emptyState: () => React.ReactNode;
|
|
8
8
|
itemInView: number;
|
|
9
9
|
itemTemplate: (index: number) => React.ReactNode;
|
|
10
|
+
containerStyle?: React.CSSProperties;
|
|
10
11
|
};
|
|
11
|
-
export declare function VirtualScrollList({ maxHeight, itemHeight, itemAmount, emptyState, itemInView, itemTemplate, "data-e2e-test-id": dataE2eTestId, }: VirtualScrollListProps): React.ReactElement;
|
|
12
|
+
export declare function VirtualScrollList({ maxHeight, itemHeight, itemAmount, emptyState, itemInView, itemTemplate, containerStyle, "data-e2e-test-id": dataE2eTestId, }: VirtualScrollListProps): React.ReactElement;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
import styled from"@emotion/styled";import React,{useRef,useReducer,useLayoutEffect,useEffect}from"react";import{VirtualScrollReducer}from"./VirtualScrollListReducer";let StyledContainer=styled("div",{target:"elzy6t10",label:"StyledContainer"})(({maxHeight})=>({overflow:"auto",width:"100%",height:"100%",maxHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xufTtcblxuY29uc3Qgb3ZlcnNjYW4gPSAxMDtcblxuZXhwb3J0IGZ1bmN0aW9uIFZpcnR1YWxTY3JvbGxMaXN0KHtcbiAgbWF4SGVpZ2h0LFxuICBpdGVtSGVpZ2h0LFxuICBpdGVtQW1vdW50LFxuICBlbXB0eVN0YXRlID0gKCkgPT4gbnVsbCxcbiAgaXRlbUluVmlldyxcbiAgaXRlbVRlbXBsYXRlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcblxuICBjb25zdCBbXG4gICAge1xuICAgICAgc2Nyb2xsZWRJdGVtQ291bnQsXG4gICAgICBhbW91bnRPZkl0ZW1zSW5WaWV3LFxuICAgICAgc2Nyb2xsZWRJblB4LFxuICAgICAgbWF4Q29udGVudEhlaWdodCA9IG1heEhlaWdodCxcbiAgICAgIHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24sXG4gICAgfSxcbiAgICBkaXNwYXRjaCxcbiAgXSA9IHVzZVJlZHVjZXIoVmlydHVhbFNjcm9sbFJlZHVjZXIsIHt9KTtcblxuICB1c2VMYXlvdXRFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHsgdHlwZTogXCJyZXNldFwiLCBpdGVtSGVpZ2h0LCBtYXhIZWlnaHQsIG92ZXJzY2FuIH0pO1xuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSAwO1xuICB9LCBbaXRlbUhlaWdodCwgaXRlbUFtb3VudCwgbWF4SGVpZ2h0XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInJlY29tbWVuZFNjcm9sbFBvc2l0aW9uXCIsXG4gICAgICBzY3JvbGxUb3A6IHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AsXG4gICAgICBpdGVtVG9CZUluVmlldzogaXRlbUluVmlldyxcbiAgICB9KTtcbiAgfSwgW2l0ZW1JblZpZXddKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uID09PSBudWxsKSByZXR1cm47XG5cbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbjtcbiAgfSwgW3JlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb25dKTtcbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3sgbWF4SGVpZ2h0IH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBS3dCIn0= */"),StyledScrollableContent=styled("div",{target:"elzy6t11",label:"StyledScrollableContent"})(({maxContentHeight})=>({overflow:"hidden",boxSizing:"border-box",height:maxContentHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xufTtcblxuY29uc3Qgb3ZlcnNjYW4gPSAxMDtcblxuZXhwb3J0IGZ1bmN0aW9uIFZpcnR1YWxTY3JvbGxMaXN0KHtcbiAgbWF4SGVpZ2h0LFxuICBpdGVtSGVpZ2h0LFxuICBpdGVtQW1vdW50LFxuICBlbXB0eVN0YXRlID0gKCkgPT4gbnVsbCxcbiAgaXRlbUluVmlldyxcbiAgaXRlbVRlbXBsYXRlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcblxuICBjb25zdCBbXG4gICAge1xuICAgICAgc2Nyb2xsZWRJdGVtQ291bnQsXG4gICAgICBhbW91bnRPZkl0ZW1zSW5WaWV3LFxuICAgICAgc2Nyb2xsZWRJblB4LFxuICAgICAgbWF4Q29udGVudEhlaWdodCA9IG1heEhlaWdodCxcbiAgICAgIHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24sXG4gICAgfSxcbiAgICBkaXNwYXRjaCxcbiAgXSA9IHVzZVJlZHVjZXIoVmlydHVhbFNjcm9sbFJlZHVjZXIsIHt9KTtcblxuICB1c2VMYXlvdXRFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHsgdHlwZTogXCJyZXNldFwiLCBpdGVtSGVpZ2h0LCBtYXhIZWlnaHQsIG92ZXJzY2FuIH0pO1xuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSAwO1xuICB9LCBbaXRlbUhlaWdodCwgaXRlbUFtb3VudCwgbWF4SGVpZ2h0XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInJlY29tbWVuZFNjcm9sbFBvc2l0aW9uXCIsXG4gICAgICBzY3JvbGxUb3A6IHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AsXG4gICAgICBpdGVtVG9CZUluVmlldzogaXRlbUluVmlldyxcbiAgICB9KTtcbiAgfSwgW2l0ZW1JblZpZXddKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uID09PSBudWxsKSByZXR1cm47XG5cbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbjtcbiAgfSwgW3JlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb25dKTtcbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3sgbWF4SGVpZ2h0IH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBY2dDIn0= */");export function VirtualScrollList({maxHeight,itemHeight,itemAmount,emptyState=()=>null,itemInView,itemTemplate,"data-e2e-test-id":dataE2eTestId}){let scrollableContainerRef=useRef(null),viewportRef=useRef(null),[{scrolledItemCount,amountOfItemsInView,scrolledInPx,maxContentHeight=maxHeight,recommendedScrollPosition},dispatch]=useReducer(VirtualScrollReducer,{});useLayoutEffect(()=>{dispatch({type:"reset",itemHeight,maxHeight,overscan:10}),scrollableContainerRef.current.scrollTop=0},[itemHeight,itemAmount,maxHeight]),useEffect(()=>{dispatch({type:"recommendScrollPosition",scrollTop:scrollableContainerRef.current.scrollTop,itemToBeInView:itemInView})},[itemInView]),useEffect(()=>{null!==recommendedScrollPosition&&(scrollableContainerRef.current.scrollTop=recommendedScrollPosition)},[recommendedScrollPosition]),useLayoutEffect(()=>{viewportRef.current&&dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})},[itemAmount,scrolledItemCount]);let itemCountToBeRendered=Math.min(Math.max(0,itemAmount-scrolledItemCount),amountOfItemsInView);return React.createElement(StyledContainer,{ref:scrollableContainerRef,maxHeight:maxHeight,style:{maxHeight},"data-e2e-test-id":dataE2eTestId,"data-ds-id":"VirtualScrollList",onScroll:e=>{dispatch({type:"scroll",scrollTop:e.currentTarget.scrollTop})}},0===itemAmount?emptyState():React.createElement(StyledScrollableContent,{maxContentHeight:maxContentHeight},React.createElement("div",{ref:viewportRef,style:{transform:`translateY(${scrolledInPx}px`}},!!itemCountToBeRendered&&Array(itemCountToBeRendered).fill(0).map((_,index)=>itemTemplate(scrolledItemCount+index)))))}
|
|
1
|
+
import styled from"@emotion/styled";import React,{useRef,useReducer,useLayoutEffect,useEffect}from"react";import{VirtualScrollReducer}from"./VirtualScrollListReducer";let StyledContainer=styled("div",{target:"ehwxt1v0",label:"StyledContainer"})(({maxHeight})=>({overflow:"auto",width:"100%",height:"100%",maxHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBtYXhIZWlnaHQsXG4gIGl0ZW1IZWlnaHQsXG4gIGl0ZW1BbW91bnQsXG4gIGVtcHR5U3RhdGUgPSAoKSA9PiBudWxsLFxuICBpdGVtSW5WaWV3LFxuICBpdGVtVGVtcGxhdGUsXG4gIGNvbnRhaW5lclN0eWxlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcblxuICBjb25zdCBbXG4gICAge1xuICAgICAgc2Nyb2xsZWRJdGVtQ291bnQsXG4gICAgICBhbW91bnRPZkl0ZW1zSW5WaWV3LFxuICAgICAgc2Nyb2xsZWRJblB4LFxuICAgICAgbWF4Q29udGVudEhlaWdodCA9IG1heEhlaWdodCxcbiAgICAgIHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24sXG4gICAgfSxcbiAgICBkaXNwYXRjaCxcbiAgXSA9IHVzZVJlZHVjZXIoVmlydHVhbFNjcm9sbFJlZHVjZXIsIHt9KTtcblxuICB1c2VMYXlvdXRFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHsgdHlwZTogXCJyZXNldFwiLCBpdGVtSGVpZ2h0LCBtYXhIZWlnaHQsIG92ZXJzY2FuIH0pO1xuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSAwO1xuICB9LCBbaXRlbUhlaWdodCwgaXRlbUFtb3VudCwgbWF4SGVpZ2h0XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInJlY29tbWVuZFNjcm9sbFBvc2l0aW9uXCIsXG4gICAgICBzY3JvbGxUb3A6IHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AsXG4gICAgICBpdGVtVG9CZUluVmlldzogaXRlbUluVmlldyxcbiAgICB9KTtcbiAgfSwgW2l0ZW1JblZpZXddKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uID09PSBudWxsKSByZXR1cm47XG5cbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbjtcbiAgfSwgW3JlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb25dKTtcbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgbWF4SGVpZ2h0LFxuICAgICAgICAuLi5jb250YWluZXJTdHlsZSxcbiAgICAgIH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBS3dCIn0= */"),StyledScrollableContent=styled("div",{target:"ehwxt1v1",label:"StyledScrollableContent"})(({maxContentHeight})=>({overflow:"hidden",boxSizing:"border-box",height:maxContentHeight}),"/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3JjL2NvbXBvbmVudHMvVmlydHVhbFNjcm9sbExpc3QvVmlydHVhbFNjcm9sbExpc3QudHN4Iiwic291cmNlcyI6WyJzcmMvY29tcG9uZW50cy9WaXJ0dWFsU2Nyb2xsTGlzdC9WaXJ0dWFsU2Nyb2xsTGlzdC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHN0eWxlZCBmcm9tIFwiQGVtb3Rpb24vc3R5bGVkXCI7XG5pbXBvcnQgUmVhY3QsIHsgdXNlUmVmLCB1c2VSZWR1Y2VyLCB1c2VMYXlvdXRFZmZlY3QsIHVzZUVmZmVjdCB9IGZyb20gXCJyZWFjdFwiO1xuXG5pbXBvcnQgeyBWaXJ0dWFsU2Nyb2xsUmVkdWNlciB9IGZyb20gXCIuL1ZpcnR1YWxTY3JvbGxMaXN0UmVkdWNlclwiO1xuXG5jb25zdCBTdHlsZWRDb250YWluZXIgPSBzdHlsZWQuZGl2PFBhcnRpYWw8VmlydHVhbFNjcm9sbExpc3RQcm9wcz4+KFxuICAoeyBtYXhIZWlnaHQgfSkgPT4gKHtcbiAgICBvdmVyZmxvdzogXCJhdXRvXCIsXG4gICAgd2lkdGg6IFwiMTAwJVwiLFxuICAgIGhlaWdodDogXCIxMDAlXCIsXG4gICAgbWF4SGVpZ2h0LFxuICB9KVxuKTtcblxuY29uc3QgU3R5bGVkU2Nyb2xsYWJsZUNvbnRlbnQgPSBzdHlsZWQuZGl2PHsgbWF4Q29udGVudEhlaWdodDogbnVtYmVyIH0+KFxuICAoeyBtYXhDb250ZW50SGVpZ2h0IH0pID0+ICh7XG4gICAgb3ZlcmZsb3c6IFwiaGlkZGVuXCIsXG4gICAgYm94U2l6aW5nOiBcImJvcmRlci1ib3hcIixcbiAgICBoZWlnaHQ6IG1heENvbnRlbnRIZWlnaHQsXG4gIH0pXG4pO1xuXG5leHBvcnQgdHlwZSBWaXJ0dWFsU2Nyb2xsTGlzdFByb3BzID0ge1xuICBcImRhdGEtZTJlLXRlc3QtaWRcIj86IHN0cmluZztcbiAgbWF4SGVpZ2h0OiBudW1iZXI7XG4gIGl0ZW1IZWlnaHQ6IG51bWJlcjtcbiAgaXRlbUFtb3VudDogbnVtYmVyO1xuICBlbXB0eVN0YXRlOiAoKSA9PiBSZWFjdC5SZWFjdE5vZGU7XG4gIGl0ZW1JblZpZXc6IG51bWJlcjtcbiAgaXRlbVRlbXBsYXRlOiAoaW5kZXg6IG51bWJlcikgPT4gUmVhY3QuUmVhY3ROb2RlO1xuICBjb250YWluZXJTdHlsZT86IFJlYWN0LkNTU1Byb3BlcnRpZXM7XG59O1xuXG5jb25zdCBvdmVyc2NhbiA9IDEwO1xuXG5leHBvcnQgZnVuY3Rpb24gVmlydHVhbFNjcm9sbExpc3Qoe1xuICBtYXhIZWlnaHQsXG4gIGl0ZW1IZWlnaHQsXG4gIGl0ZW1BbW91bnQsXG4gIGVtcHR5U3RhdGUgPSAoKSA9PiBudWxsLFxuICBpdGVtSW5WaWV3LFxuICBpdGVtVGVtcGxhdGUsXG4gIGNvbnRhaW5lclN0eWxlLFxuICBcImRhdGEtZTJlLXRlc3QtaWRcIjogZGF0YUUyZVRlc3RJZCxcbn06IFZpcnR1YWxTY3JvbGxMaXN0UHJvcHMpOiBSZWFjdC5SZWFjdEVsZW1lbnQge1xuICBjb25zdCBzY3JvbGxhYmxlQ29udGFpbmVyUmVmID0gdXNlUmVmKG51bGwpO1xuICBjb25zdCB2aWV3cG9ydFJlZiA9IHVzZVJlZihudWxsKTtcblxuICBjb25zdCBbXG4gICAge1xuICAgICAgc2Nyb2xsZWRJdGVtQ291bnQsXG4gICAgICBhbW91bnRPZkl0ZW1zSW5WaWV3LFxuICAgICAgc2Nyb2xsZWRJblB4LFxuICAgICAgbWF4Q29udGVudEhlaWdodCA9IG1heEhlaWdodCxcbiAgICAgIHJlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb24sXG4gICAgfSxcbiAgICBkaXNwYXRjaCxcbiAgXSA9IHVzZVJlZHVjZXIoVmlydHVhbFNjcm9sbFJlZHVjZXIsIHt9KTtcblxuICB1c2VMYXlvdXRFZmZlY3QoKCkgPT4ge1xuICAgIGRpc3BhdGNoKHsgdHlwZTogXCJyZXNldFwiLCBpdGVtSGVpZ2h0LCBtYXhIZWlnaHQsIG92ZXJzY2FuIH0pO1xuICAgIHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AgPSAwO1xuICB9LCBbaXRlbUhlaWdodCwgaXRlbUFtb3VudCwgbWF4SGVpZ2h0XSk7XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInJlY29tbWVuZFNjcm9sbFBvc2l0aW9uXCIsXG4gICAgICBzY3JvbGxUb3A6IHNjcm9sbGFibGVDb250YWluZXJSZWYuY3VycmVudC5zY3JvbGxUb3AsXG4gICAgICBpdGVtVG9CZUluVmlldzogaXRlbUluVmlldyxcbiAgICB9KTtcbiAgfSwgW2l0ZW1JblZpZXddKTtcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChyZWNvbW1lbmRlZFNjcm9sbFBvc2l0aW9uID09PSBudWxsKSByZXR1cm47XG5cbiAgICBzY3JvbGxhYmxlQ29udGFpbmVyUmVmLmN1cnJlbnQuc2Nyb2xsVG9wID0gcmVjb21tZW5kZWRTY3JvbGxQb3NpdGlvbjtcbiAgfSwgW3JlY29tbWVuZGVkU2Nyb2xsUG9zaXRpb25dKTtcbiAgdXNlTGF5b3V0RWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdwb3J0UmVmLmN1cnJlbnQpIHJldHVybjtcbiAgICBkaXNwYXRjaCh7XG4gICAgICB0eXBlOiBcInVwZGF0ZVZpZXdwb3J0XCIsXG4gICAgICB2aWV3cG9ydE5vZGU6IHZpZXdwb3J0UmVmLmN1cnJlbnQsXG4gICAgICBpdGVtQW1vdW50LFxuICAgIH0pO1xuICB9LCBbaXRlbUFtb3VudCwgc2Nyb2xsZWRJdGVtQ291bnRdKTtcblxuICBjb25zdCBpdGVtQ291bnRUb0JlUmVuZGVyZWQgPSBNYXRoLm1pbihcbiAgICBNYXRoLm1heCgwLCBpdGVtQW1vdW50IC0gc2Nyb2xsZWRJdGVtQ291bnQpLFxuICAgIGFtb3VudE9mSXRlbXNJblZpZXdcbiAgKTtcblxuICByZXR1cm4gKFxuICAgIDxTdHlsZWRDb250YWluZXJcbiAgICAgIHJlZj17c2Nyb2xsYWJsZUNvbnRhaW5lclJlZn1cbiAgICAgIG1heEhlaWdodD17bWF4SGVpZ2h0fVxuICAgICAgc3R5bGU9e3tcbiAgICAgICAgbWF4SGVpZ2h0LFxuICAgICAgICAuLi5jb250YWluZXJTdHlsZSxcbiAgICAgIH19XG4gICAgICBkYXRhLWUyZS10ZXN0LWlkPXtkYXRhRTJlVGVzdElkfVxuICAgICAgZGF0YS1kcy1pZD1cIlZpcnR1YWxTY3JvbGxMaXN0XCJcbiAgICAgIG9uU2Nyb2xsPXsoZTogUmVhY3QuVUlFdmVudDxIVE1MRWxlbWVudD4pID0+IHtcbiAgICAgICAgZGlzcGF0Y2goe1xuICAgICAgICAgIHR5cGU6IFwic2Nyb2xsXCIsXG4gICAgICAgICAgc2Nyb2xsVG9wOiBlLmN1cnJlbnRUYXJnZXQuc2Nyb2xsVG9wLFxuICAgICAgICB9KTtcbiAgICAgIH19XG4gICAgPlxuICAgICAge2l0ZW1BbW91bnQgPT09IDAgPyAoXG4gICAgICAgIGVtcHR5U3RhdGUoKVxuICAgICAgKSA6IChcbiAgICAgICAgPFN0eWxlZFNjcm9sbGFibGVDb250ZW50IG1heENvbnRlbnRIZWlnaHQ9e21heENvbnRlbnRIZWlnaHR9PlxuICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgIHJlZj17dmlld3BvcnRSZWZ9XG4gICAgICAgICAgICBzdHlsZT17e1xuICAgICAgICAgICAgICB0cmFuc2Zvcm06IGB0cmFuc2xhdGVZKCR7c2Nyb2xsZWRJblB4fXB4YCxcbiAgICAgICAgICAgIH19XG4gICAgICAgICAgPlxuICAgICAgICAgICAgeyEhaXRlbUNvdW50VG9CZVJlbmRlcmVkICYmXG4gICAgICAgICAgICAgIG5ldyBBcnJheShpdGVtQ291bnRUb0JlUmVuZGVyZWQpXG4gICAgICAgICAgICAgICAgLmZpbGwoMClcbiAgICAgICAgICAgICAgICAubWFwKChfLCBpbmRleCkgPT4gaXRlbVRlbXBsYXRlKHNjcm9sbGVkSXRlbUNvdW50ICsgaW5kZXgpKX1cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgPC9TdHlsZWRTY3JvbGxhYmxlQ29udGVudD5cbiAgICAgICl9XG4gICAgPC9TdHlsZWRDb250YWluZXI+XG4gICk7XG59XG4iXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBY2dDIn0= */");export function VirtualScrollList({maxHeight,itemHeight,itemAmount,emptyState=()=>null,itemInView,itemTemplate,containerStyle,"data-e2e-test-id":dataE2eTestId}){let scrollableContainerRef=useRef(null),viewportRef=useRef(null),[{scrolledItemCount,amountOfItemsInView,scrolledInPx,maxContentHeight=maxHeight,recommendedScrollPosition},dispatch]=useReducer(VirtualScrollReducer,{});useLayoutEffect(()=>{dispatch({type:"reset",itemHeight,maxHeight,overscan:10}),scrollableContainerRef.current.scrollTop=0},[itemHeight,itemAmount,maxHeight]),useEffect(()=>{dispatch({type:"recommendScrollPosition",scrollTop:scrollableContainerRef.current.scrollTop,itemToBeInView:itemInView})},[itemInView]),useEffect(()=>{null!==recommendedScrollPosition&&(scrollableContainerRef.current.scrollTop=recommendedScrollPosition)},[recommendedScrollPosition]),useLayoutEffect(()=>{viewportRef.current&&dispatch({type:"updateViewport",viewportNode:viewportRef.current,itemAmount})},[itemAmount,scrolledItemCount]);let itemCountToBeRendered=Math.min(Math.max(0,itemAmount-scrolledItemCount),amountOfItemsInView);return React.createElement(StyledContainer,{ref:scrollableContainerRef,maxHeight:maxHeight,style:{maxHeight,...containerStyle},"data-e2e-test-id":dataE2eTestId,"data-ds-id":"VirtualScrollList",onScroll:e=>{dispatch({type:"scroll",scrollTop:e.currentTarget.scrollTop})}},0===itemAmount?emptyState():React.createElement(StyledScrollableContent,{maxContentHeight:maxContentHeight},React.createElement("div",{ref:viewportRef,style:{transform:`translateY(${scrolledInPx}px`}},!!itemCountToBeRendered&&Array(itemCountToBeRendered).fill(0).map((_,index)=>itemTemplate(scrolledItemCount+index)))))}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
let findClosestOffsetIndex=(arr,val)=>{let start=0,end=arr.length-1,mid=Math.floor((start+end)/2);for(;start<=end&&arr[mid=Math.floor((start+end)/2)].amount!==val;)val<arr[mid].amount?end=mid-1:start=mid+1;return mid},getMaxHeight=(maxAmountOfItems,offsets,lastItemHeight=0)=>{
|
|
1
|
+
let findClosestOffsetIndex=(arr,val)=>{let start=0,end=arr.length-1,mid=Math.floor((start+end)/2);for(;start<=end&&arr[mid=Math.floor((start+end)/2)].amount!==val;)val<arr[mid].amount?end=mid-1:start=mid+1;return mid},getMaxHeight=(maxAmountOfItems,offsets,lastItemHeight=0)=>{let amountOfOffsets=Math.min(offsets.length,maxAmountOfItems);return(offsets[amountOfOffsets-1]?.amount||0)+lastItemHeight},reduceScroll=(prevState,scrollTop)=>{let{topOffsets,overscan:tolerance}=prevState,scrolledItemCount=Math.max(0,findClosestOffsetIndex(topOffsets,scrollTop)-tolerance/2),scrolledInPx=topOffsets[scrolledItemCount]?.amount||0;return{...prevState,scrolledItemCount,scrolledInPx}},reduceUpdateViewport=(prevState,viewportNode,itemAmount)=>{let{topOffsets=[],scrolledInPx,scrolledItemCount,avgItemHeight,lastItemHeight,maxHeight,overscan:tolerance}=prevState,newTopOffsets=[...topOffsets],updatedLastItemHeight=lastItemHeight;Array.from(viewportNode.children).forEach((child,i)=>{newTopOffsets[i+scrolledItemCount]={amount:scrolledInPx+child.offsetTop,isInterpolated:!1},i+scrolledItemCount!==itemAmount-1||lastItemHeight||(updatedLastItemHeight=child.getBoundingClientRect().height)});for(let i=0;i<itemAmount;i+=1)(!newTopOffsets[i]||newTopOffsets[i].isInterpolated)&&(newTopOffsets[i]={amount:newTopOffsets[i-1].amount+avgItemHeight,isInterpolated:!0});let newAvgItemHeight=Math.max(avgItemHeight,viewportNode.getBoundingClientRect().height/(viewportNode.childElementCount||1)),maxContentHeight=getMaxHeight(itemAmount,newTopOffsets,updatedLastItemHeight);return{...prevState,lastItemHeight:updatedLastItemHeight,avgItemHeight,topOffsets:newTopOffsets,amountOfItemsInView:Math.round(maxHeight/newAvgItemHeight)+tolerance,maxContentHeight}},reduceReset=(prevState,itemHeight,maxHeight,overscan)=>({...prevState,lastItemHeight:null,topOffsets:[{amount:0,isInterpolated:!1}],scrolledInPx:0,scrolledItemCount:0,maxHeight,avgItemHeight:itemHeight,overscan}),reduceRecommendScrollPosition=(prevState,scrollTop,itemToBeInView)=>{let{topOffsets,maxHeight}=prevState;if(!topOffsets)return prevState;let bottomOfTarget=topOffsets[itemToBeInView+1]?.amount,topOfTarget=topOffsets[itemToBeInView]?.amount,recommendedScrollPosition=null;return topOfTarget<scrollTop&&(recommendedScrollPosition=topOfTarget),bottomOfTarget>=scrollTop+maxHeight&&(recommendedScrollPosition=bottomOfTarget-maxHeight),itemToBeInView===topOffsets.length-1&&(recommendedScrollPosition=topOffsets[itemToBeInView].amount),{...prevState,recommendedScrollPosition}};export const VirtualScrollReducer=(prevState,action)=>{switch(action.type){case"reset":return reduceReset(prevState,action.itemHeight,action.maxHeight,action.overscan);case"scroll":return reduceScroll(prevState,action.scrollTop);case"updateViewport":return reduceUpdateViewport(prevState,action.viewportNode,action.itemAmount);case"recommendScrollPosition":return reduceRecommendScrollPosition(prevState,action.scrollTop,action.itemToBeInView);default:throw Error()}};
|