@codecademy/brand 3.19.0-alpha.c35a1cb4bd.0 → 3.19.0-alpha.c8b1055646.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (39) hide show
  1. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/MarketingBanner.d.ts +1 -3
  2. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/MarketingBanner.js +7 -4
  3. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/NavPanels.d.ts +0 -1
  4. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/NavPanels.js +3 -4
  5. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/consts.d.ts +2 -9
  6. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/consts.js +2 -48
  7. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/index.js +8 -20
  8. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/tabConfig.d.ts +2 -0
  9. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/tabConfig.js +48 -0
  10. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdown/index.js +9 -14
  11. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderDropdownLink.js +3 -2
  12. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderMenuItem.js +5 -0
  13. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/StyledAppHeaderLink.js +5 -2
  14. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/index.js +3 -2
  15. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderDropdownNavButton.js +5 -5
  16. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderMenuNavButton.js +9 -7
  17. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/NavPanels.js +12 -6
  18. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/consts.d.ts +1 -8
  19. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/consts.js +1 -39
  20. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.js +7 -22
  21. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/tabConfig.d.ts +2 -0
  22. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/tabConfig.js +39 -0
  23. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/AppHeaderSection.test.js +215 -30
  24. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/MobileBackButton.js +4 -5
  25. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/MobileNavMenu.d.ts +2 -5
  26. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/MobileNavMenu.js +7 -5
  27. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/consts.d.ts +2 -0
  28. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/consts.js +37 -0
  29. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/elements.d.ts +37 -0
  30. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/elements.js +86 -12
  31. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/index.d.ts +9 -6
  32. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/index.js +123 -41
  33. package/dist/AppHeader/shared/elements.d.ts +15 -3
  34. package/dist/AppHeader/shared/elements.js +138 -27
  35. package/dist/AppHeaderMobile/AppHeaderSubMenuMobile/index.js +9 -11
  36. package/dist/LearningOutcomeTile/index.js +1 -2
  37. package/package.json +2 -1
  38. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/NavSection.d.ts +0 -21
  39. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/NavSection.js +0 -162
@@ -13,16 +13,19 @@ import { AppHeaderNavButton } from '../AppHeaderNavButton';
13
13
  import { useAppHeaderContext } from '../AppHeaderProvider';
14
14
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
15
15
  const topSpacing = {
16
+ xl: '3.5rem',
16
17
  lg: '3.2rem',
17
18
  md: '2.75rem',
18
19
  sm: '2.25rem'
19
20
  };
20
21
  const getSpacing = ({
22
+ displayGlobalNavRedesign,
21
23
  isProfileDropdown,
22
24
  isGenericDropdown,
23
25
  isIconDropdown,
24
26
  isStandalone
25
27
  }) => {
28
+ if (displayGlobalNavRedesign) return topSpacing.xl;
26
29
  if (isProfileDropdown || isIconDropdown) return topSpacing.lg;
27
30
  if (isGenericDropdown && isStandalone) return topSpacing.md;
28
31
  return topSpacing.sm;
@@ -33,7 +36,7 @@ const StyledLinkSection = /*#__PURE__*/_styled(AppHeaderLinkSections, {
33
36
  })(css({
34
37
  padding: `0.75rem 0`,
35
38
  position: `absolute`
36
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx"],"names":[],"mappings":"AAiD0B","file":"../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx","sourcesContent":["import { Menu } from '@codecademy/gamut';\nimport { ColorMode, css } from '@codecademy/gamut-styles';\nimport { Scale } from '@codecademy/variance/dist/types/config';\nimport styled from '@emotion/styled';\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react';\nimport * as React from 'react';\nimport { useIsomorphicLayoutEffect } from 'react-use';\n\nimport { GlobalNavRedesignContext } from '../../../GlobalHeader';\nimport {\n  AnimatedSimpleDropdown,\n  AppHeaderAction,\n  AppHeaderDropdownItem,\n} from '../../shared';\nimport { AppHeaderDropdownProvider } from '../AppHeaderDropdownProvider';\nimport { AppHeaderLinkSections } from '../AppHeaderLinkSections';\nimport { AppHeaderMenuProvider } from '../AppHeaderMenuProvider';\nimport { AppHeaderNavButton } from '../AppHeaderNavButton';\nimport { useAppHeaderContext } from '../AppHeaderProvider';\n\nconst topSpacing = {\n  lg: '3.2rem',\n  md: '2.75rem',\n  sm: '2.25rem',\n};\n\nconst getSpacing = ({\n  isProfileDropdown,\n  isGenericDropdown,\n  isIconDropdown,\n  isStandalone,\n}: {\n  isProfileDropdown?: boolean;\n  isGenericDropdown?: boolean;\n  isIconDropdown?: boolean;\n  isStandalone?: boolean;\n}) => {\n  if (isProfileDropdown || isIconDropdown) return topSpacing.lg;\n  if (isGenericDropdown && isStandalone) return topSpacing.md;\n  return topSpacing.sm;\n};\n\nconst StyledLinkSection = styled(AppHeaderLinkSections)(\n  css({\n    padding: `0.75rem 0`,\n    position: `absolute`,\n  })\n);\n\nconst StyledDropdownMenu = styled(Menu)<{\n  displayGlobalNavRedesign: boolean;\n}>`\n  ${({ displayGlobalNavRedesign }) =>\n    displayGlobalNavRedesign &&\n    css({\n      border: 1,\n      borderColor: 'border-tertiary',\n      borderRadius: 'lg',\n    })}\n`;\n\nexport type AppHeaderDropdownProps = AppHeaderAction &\n  Pick<React.ComponentProps<typeof Menu>, 'spacing'> & {\n    item: AppHeaderDropdownItem;\n    onKeyDown?: (event: React.KeyboardEvent) => void;\n    /**\n     * If true, the dropdown is being rendered as a standalone component, rather than\n     * part of the global header.\n     */\n    standalone?: boolean;\n  };\n\nexport const AppHeaderDropdown: React.FC<AppHeaderDropdownProps> = ({\n  action,\n  item,\n  spacing = 'normal',\n  standalone,\n}) => {\n  const listRef = useRef<HTMLUListElement>(null);\n  const buttonRef = useRef<HTMLButtonElement>(null);\n\n  const [dimensions, setDimensions] = useState({ height: 0, width: 0 });\n  const [isOpen, setIsOpen] = useState(false);\n  const { lastOpenedDropdown, setLastOpenedDropdown } = useAppHeaderContext();\n  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);\n\n  const tabIndex = isOpen === false ? -1 : 0;\n  const dropdownId = item?.text ?? item?.type;\n\n  const toggleIsOpen = () => setIsOpen((prev) => !prev);\n\n  const handleOnClick = (event: React.MouseEvent) => {\n    toggleIsOpen();\n    if (!isOpen) {\n      if (action) action(event, item);\n      if (setLastOpenedDropdown) {\n        setLastOpenedDropdown(dropdownId);\n      }\n    }\n  };\n\n  const handleKeyboardClose = useCallback(() => {\n    setIsOpen(false);\n    buttonRef.current?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (lastOpenedDropdown !== dropdownId && isOpen) {\n      setIsOpen(false);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [lastOpenedDropdown]);\n\n  useIsomorphicLayoutEffect(() => {\n    if (listRef.current) {\n      const { height, width } = listRef.current.getBoundingClientRect();\n      setDimensions({ height, width });\n    }\n  }, [listRef, isOpen]);\n\n  useEffect(() => {\n    function handleClickOutside(event: MouseEvent | Event) {\n      const list = listRef?.current;\n      const button = buttonRef?.current;\n      if (\n        isOpen &&\n        list &&\n        !list.contains(event.target as Node) &&\n        button &&\n        !button.contains(event.target as Node)\n      ) {\n        setIsOpen(false);\n      }\n    }\n\n    document.addEventListener('mousedown', handleClickOutside);\n    document.addEventListener('blur', handleClickOutside);\n    return () => {\n      document.removeEventListener('mousedown', handleClickOutside);\n      document.removeEventListener('blur', handleClickOutside);\n    };\n  }, [listRef, isOpen]);\n\n  const isProfileDropdown = item.type === 'profile-dropdown';\n  const isGenericDropdown = ['dropdown', 'resources-simple-dropdown'].includes(\n    item.type\n  );\n  const isIconDropdown = 'isIconOnly' in item && item.isIconOnly;\n\n  const Provider = isProfileDropdown\n    ? AppHeaderMenuProvider\n    : AppHeaderDropdownProvider;\n\n  const topSpacing = useMemo(\n    () =>\n      getSpacing({\n        isProfileDropdown,\n        isGenericDropdown,\n        isIconDropdown,\n        isStandalone: standalone,\n      }),\n    [isProfileDropdown, isGenericDropdown, isIconDropdown, standalone]\n  );\n\n  const commonMenuProps = {\n    displayGlobalNavRedesign,\n    minWidth: '202px',\n    ref: listRef,\n    role: isProfileDropdown ? 'menu' : undefined,\n    spacing,\n    variant: 'popover' as const,\n    width: 'fit-content' as const,\n    zIndex: 1,\n    py:\n      spacing === 'normal'\n        ? 12\n        : (0 as Scale<{\n            readonly property: 'padding';\n            readonly scale: 'spacing';\n          }>),\n  };\n\n  const commonLinkSectionProps = {\n    action,\n    item,\n    id: `menu-container${item.text}`,\n    'aria-controls': `menu-container${item.text}`,\n    'aria-label': item.text,\n    'aria-hidden': !isOpen,\n    tabIndex,\n  };\n\n  return (\n    <Provider handleClose={handleKeyboardClose}>\n      <AppHeaderNavButton\n        buttonRef={buttonRef}\n        handleOnClick={handleOnClick}\n        isOpen={isOpen}\n        item={item}\n      />\n      <AnimatedSimpleDropdown\n        style={{\n          right: isProfileDropdown ? '0.5rem' : '',\n          top: topSpacing,\n          width: dimensions.width,\n        }}\n        animate={{\n          height: isOpen ? dimensions.height : 0,\n        }}\n        initial={{ borderWidth: 0, height: 0 }}\n        transition={{ duration: 0.175 }}\n        aria-hidden={!isOpen}\n      >\n        {displayGlobalNavRedesign ? (\n          <ColorMode mode=\"light\">\n            <StyledDropdownMenu {...commonMenuProps}>\n              <StyledLinkSection {...commonLinkSectionProps} />\n            </StyledDropdownMenu>\n          </ColorMode>\n        ) : (\n          <StyledDropdownMenu {...commonMenuProps}>\n            <StyledLinkSection {...commonLinkSectionProps} />\n          </StyledDropdownMenu>\n        )}\n      </AnimatedSimpleDropdown>\n    </Provider>\n  );\n};\n"]} */");
39
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx"],"names":[],"mappings":"AAqD0B","file":"../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx","sourcesContent":["import { Menu } from '@codecademy/gamut';\nimport { ColorMode, css } from '@codecademy/gamut-styles';\nimport { Scale } from '@codecademy/variance/dist/types/config';\nimport styled from '@emotion/styled';\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react';\nimport * as React from 'react';\nimport { useIsomorphicLayoutEffect } from 'react-use';\n\nimport { GlobalNavRedesignContext } from '../../../GlobalHeader';\nimport {\n  AnimatedSimpleDropdown,\n  AppHeaderAction,\n  AppHeaderDropdownItem,\n} from '../../shared';\nimport { AppHeaderDropdownProvider } from '../AppHeaderDropdownProvider';\nimport { AppHeaderLinkSections } from '../AppHeaderLinkSections';\nimport { AppHeaderMenuProvider } from '../AppHeaderMenuProvider';\nimport { AppHeaderNavButton } from '../AppHeaderNavButton';\nimport { useAppHeaderContext } from '../AppHeaderProvider';\n\nconst topSpacing = {\n  xl: '3.5rem',\n  lg: '3.2rem',\n  md: '2.75rem',\n  sm: '2.25rem',\n};\n\nconst getSpacing = ({\n  displayGlobalNavRedesign,\n  isProfileDropdown,\n  isGenericDropdown,\n  isIconDropdown,\n  isStandalone,\n}: {\n  displayGlobalNavRedesign?: boolean;\n  isProfileDropdown?: boolean;\n  isGenericDropdown?: boolean;\n  isIconDropdown?: boolean;\n  isStandalone?: boolean;\n}) => {\n  if (displayGlobalNavRedesign) return topSpacing.xl;\n  if (isProfileDropdown || isIconDropdown) return topSpacing.lg;\n  if (isGenericDropdown && isStandalone) return topSpacing.md;\n  return topSpacing.sm;\n};\n\nconst StyledLinkSection = styled(AppHeaderLinkSections)(\n  css({\n    padding: `0.75rem 0`,\n    position: `absolute`,\n  })\n);\n\nconst StyledDropdownMenu = styled(Menu)<{\n  displayGlobalNavRedesign: boolean;\n}>`\n  ${({ displayGlobalNavRedesign }) =>\n    displayGlobalNavRedesign &&\n    css({\n      border: 1,\n      borderColor: 'border-tertiary',\n      borderRadius: 'lg',\n    })}\n`;\n\nexport type AppHeaderDropdownProps = AppHeaderAction &\n  Pick<React.ComponentProps<typeof Menu>, 'spacing'> & {\n    item: AppHeaderDropdownItem;\n    onKeyDown?: (event: React.KeyboardEvent) => void;\n    /**\n     * If true, the dropdown is being rendered as a standalone component, rather than\n     * part of the global header.\n     */\n    standalone?: boolean;\n  };\n\nexport const AppHeaderDropdown: React.FC<AppHeaderDropdownProps> = ({\n  action,\n  item,\n  spacing = 'normal',\n  standalone,\n}) => {\n  const listRef = useRef<HTMLUListElement>(null);\n  const buttonRef = useRef<HTMLButtonElement>(null);\n\n  const [dimensions, setDimensions] = useState({ height: 0, width: 0 });\n  const [isOpen, setIsOpen] = useState(false);\n  const { lastOpenedDropdown, setLastOpenedDropdown } = useAppHeaderContext();\n  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);\n\n  const tabIndex = isOpen === false ? -1 : 0;\n  const dropdownId = item?.text ?? item?.type;\n\n  const toggleIsOpen = () => setIsOpen((prev) => !prev);\n\n  const handleOnClick = (event: React.MouseEvent) => {\n    toggleIsOpen();\n    if (!isOpen) {\n      if (action) action(event, item);\n      if (setLastOpenedDropdown) {\n        setLastOpenedDropdown(dropdownId);\n      }\n    }\n  };\n\n  const handleKeyboardClose = useCallback(() => {\n    setIsOpen(false);\n    buttonRef.current?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (lastOpenedDropdown !== dropdownId && isOpen) {\n      setIsOpen(false);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [lastOpenedDropdown]);\n\n  useIsomorphicLayoutEffect(() => {\n    if (listRef.current) {\n      const { height, width } = listRef.current.getBoundingClientRect();\n      setDimensions({ height, width });\n    }\n  }, [listRef, isOpen]);\n\n  useEffect(() => {\n    function handleClickOutside(event: MouseEvent | Event) {\n      const list = listRef?.current;\n      const button = buttonRef?.current;\n      if (\n        isOpen &&\n        list &&\n        !list.contains(event.target as Node) &&\n        button &&\n        !button.contains(event.target as Node)\n      ) {\n        setIsOpen(false);\n      }\n    }\n\n    document.addEventListener('mousedown', handleClickOutside);\n    document.addEventListener('blur', handleClickOutside);\n    return () => {\n      document.removeEventListener('mousedown', handleClickOutside);\n      document.removeEventListener('blur', handleClickOutside);\n    };\n  }, [listRef, isOpen]);\n\n  const isProfileDropdown = item.type === 'profile-dropdown';\n  const isGenericDropdown = ['dropdown', 'resources-simple-dropdown'].includes(\n    item.type\n  );\n  const isIconDropdown = 'isIconOnly' in item && item.isIconOnly;\n\n  const Provider = isProfileDropdown\n    ? AppHeaderMenuProvider\n    : AppHeaderDropdownProvider;\n\n  const topSpacing = useMemo(\n    () =>\n      getSpacing({\n        displayGlobalNavRedesign,\n        isProfileDropdown,\n        isGenericDropdown,\n        isIconDropdown,\n        isStandalone: standalone,\n      }),\n    [\n      isProfileDropdown,\n      isGenericDropdown,\n      isIconDropdown,\n      standalone,\n      displayGlobalNavRedesign,\n    ]\n  );\n\n  const commonMenuProps = {\n    displayGlobalNavRedesign,\n    minWidth: '202px',\n    ref: listRef,\n    role: isProfileDropdown ? 'menu' : undefined,\n    spacing,\n    variant: 'popover' as const,\n    width: 'fit-content' as const,\n    zIndex: 1,\n    py:\n      spacing === 'normal'\n        ? 12\n        : (0 as Scale<{\n            readonly property: 'padding';\n            readonly scale: 'spacing';\n          }>),\n  };\n\n  const commonLinkSectionProps = {\n    action,\n    item,\n    id: `menu-container${item.text}`,\n    'aria-controls': `menu-container${item.text}`,\n    'aria-label': item.text,\n    'aria-hidden': !isOpen,\n    tabIndex,\n  };\n\n  return (\n    <Provider handleClose={handleKeyboardClose}>\n      <AppHeaderNavButton\n        buttonRef={buttonRef}\n        handleOnClick={handleOnClick}\n        isOpen={isOpen}\n        item={item}\n      />\n      <AnimatedSimpleDropdown\n        isOpen={isOpen}\n        style={{\n          right: isProfileDropdown ? '0.5rem' : '',\n          top: topSpacing,\n          width: dimensions.width,\n        }}\n        displayGlobalNavRedesign={displayGlobalNavRedesign}\n      >\n        {displayGlobalNavRedesign ? (\n          <ColorMode mode=\"light\">\n            <StyledDropdownMenu {...commonMenuProps}>\n              <StyledLinkSection {...commonLinkSectionProps} />\n            </StyledDropdownMenu>\n          </ColorMode>\n        ) : (\n          <StyledDropdownMenu {...commonMenuProps}>\n            <StyledLinkSection {...commonLinkSectionProps} />\n          </StyledDropdownMenu>\n        )}\n      </AnimatedSimpleDropdown>\n    </Provider>\n  );\n};\n"]} */");
37
40
  const StyledDropdownMenu = /*#__PURE__*/_styled(Menu, {
38
41
  target: "efmcgmc0",
39
42
  label: "StyledDropdownMenu"
@@ -43,7 +46,7 @@ const StyledDropdownMenu = /*#__PURE__*/_styled(Menu, {
43
46
  border: 1,
44
47
  borderColor: 'border-tertiary',
45
48
  borderRadius: 'lg'
46
- }), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx"],"names":[],"mappings":"AA0DE","file":"../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx","sourcesContent":["import { Menu } from '@codecademy/gamut';\nimport { ColorMode, css } from '@codecademy/gamut-styles';\nimport { Scale } from '@codecademy/variance/dist/types/config';\nimport styled from '@emotion/styled';\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react';\nimport * as React from 'react';\nimport { useIsomorphicLayoutEffect } from 'react-use';\n\nimport { GlobalNavRedesignContext } from '../../../GlobalHeader';\nimport {\n  AnimatedSimpleDropdown,\n  AppHeaderAction,\n  AppHeaderDropdownItem,\n} from '../../shared';\nimport { AppHeaderDropdownProvider } from '../AppHeaderDropdownProvider';\nimport { AppHeaderLinkSections } from '../AppHeaderLinkSections';\nimport { AppHeaderMenuProvider } from '../AppHeaderMenuProvider';\nimport { AppHeaderNavButton } from '../AppHeaderNavButton';\nimport { useAppHeaderContext } from '../AppHeaderProvider';\n\nconst topSpacing = {\n  lg: '3.2rem',\n  md: '2.75rem',\n  sm: '2.25rem',\n};\n\nconst getSpacing = ({\n  isProfileDropdown,\n  isGenericDropdown,\n  isIconDropdown,\n  isStandalone,\n}: {\n  isProfileDropdown?: boolean;\n  isGenericDropdown?: boolean;\n  isIconDropdown?: boolean;\n  isStandalone?: boolean;\n}) => {\n  if (isProfileDropdown || isIconDropdown) return topSpacing.lg;\n  if (isGenericDropdown && isStandalone) return topSpacing.md;\n  return topSpacing.sm;\n};\n\nconst StyledLinkSection = styled(AppHeaderLinkSections)(\n  css({\n    padding: `0.75rem 0`,\n    position: `absolute`,\n  })\n);\n\nconst StyledDropdownMenu = styled(Menu)<{\n  displayGlobalNavRedesign: boolean;\n}>`\n  ${({ displayGlobalNavRedesign }) =>\n    displayGlobalNavRedesign &&\n    css({\n      border: 1,\n      borderColor: 'border-tertiary',\n      borderRadius: 'lg',\n    })}\n`;\n\nexport type AppHeaderDropdownProps = AppHeaderAction &\n  Pick<React.ComponentProps<typeof Menu>, 'spacing'> & {\n    item: AppHeaderDropdownItem;\n    onKeyDown?: (event: React.KeyboardEvent) => void;\n    /**\n     * If true, the dropdown is being rendered as a standalone component, rather than\n     * part of the global header.\n     */\n    standalone?: boolean;\n  };\n\nexport const AppHeaderDropdown: React.FC<AppHeaderDropdownProps> = ({\n  action,\n  item,\n  spacing = 'normal',\n  standalone,\n}) => {\n  const listRef = useRef<HTMLUListElement>(null);\n  const buttonRef = useRef<HTMLButtonElement>(null);\n\n  const [dimensions, setDimensions] = useState({ height: 0, width: 0 });\n  const [isOpen, setIsOpen] = useState(false);\n  const { lastOpenedDropdown, setLastOpenedDropdown } = useAppHeaderContext();\n  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);\n\n  const tabIndex = isOpen === false ? -1 : 0;\n  const dropdownId = item?.text ?? item?.type;\n\n  const toggleIsOpen = () => setIsOpen((prev) => !prev);\n\n  const handleOnClick = (event: React.MouseEvent) => {\n    toggleIsOpen();\n    if (!isOpen) {\n      if (action) action(event, item);\n      if (setLastOpenedDropdown) {\n        setLastOpenedDropdown(dropdownId);\n      }\n    }\n  };\n\n  const handleKeyboardClose = useCallback(() => {\n    setIsOpen(false);\n    buttonRef.current?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (lastOpenedDropdown !== dropdownId && isOpen) {\n      setIsOpen(false);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [lastOpenedDropdown]);\n\n  useIsomorphicLayoutEffect(() => {\n    if (listRef.current) {\n      const { height, width } = listRef.current.getBoundingClientRect();\n      setDimensions({ height, width });\n    }\n  }, [listRef, isOpen]);\n\n  useEffect(() => {\n    function handleClickOutside(event: MouseEvent | Event) {\n      const list = listRef?.current;\n      const button = buttonRef?.current;\n      if (\n        isOpen &&\n        list &&\n        !list.contains(event.target as Node) &&\n        button &&\n        !button.contains(event.target as Node)\n      ) {\n        setIsOpen(false);\n      }\n    }\n\n    document.addEventListener('mousedown', handleClickOutside);\n    document.addEventListener('blur', handleClickOutside);\n    return () => {\n      document.removeEventListener('mousedown', handleClickOutside);\n      document.removeEventListener('blur', handleClickOutside);\n    };\n  }, [listRef, isOpen]);\n\n  const isProfileDropdown = item.type === 'profile-dropdown';\n  const isGenericDropdown = ['dropdown', 'resources-simple-dropdown'].includes(\n    item.type\n  );\n  const isIconDropdown = 'isIconOnly' in item && item.isIconOnly;\n\n  const Provider = isProfileDropdown\n    ? AppHeaderMenuProvider\n    : AppHeaderDropdownProvider;\n\n  const topSpacing = useMemo(\n    () =>\n      getSpacing({\n        isProfileDropdown,\n        isGenericDropdown,\n        isIconDropdown,\n        isStandalone: standalone,\n      }),\n    [isProfileDropdown, isGenericDropdown, isIconDropdown, standalone]\n  );\n\n  const commonMenuProps = {\n    displayGlobalNavRedesign,\n    minWidth: '202px',\n    ref: listRef,\n    role: isProfileDropdown ? 'menu' : undefined,\n    spacing,\n    variant: 'popover' as const,\n    width: 'fit-content' as const,\n    zIndex: 1,\n    py:\n      spacing === 'normal'\n        ? 12\n        : (0 as Scale<{\n            readonly property: 'padding';\n            readonly scale: 'spacing';\n          }>),\n  };\n\n  const commonLinkSectionProps = {\n    action,\n    item,\n    id: `menu-container${item.text}`,\n    'aria-controls': `menu-container${item.text}`,\n    'aria-label': item.text,\n    'aria-hidden': !isOpen,\n    tabIndex,\n  };\n\n  return (\n    <Provider handleClose={handleKeyboardClose}>\n      <AppHeaderNavButton\n        buttonRef={buttonRef}\n        handleOnClick={handleOnClick}\n        isOpen={isOpen}\n        item={item}\n      />\n      <AnimatedSimpleDropdown\n        style={{\n          right: isProfileDropdown ? '0.5rem' : '',\n          top: topSpacing,\n          width: dimensions.width,\n        }}\n        animate={{\n          height: isOpen ? dimensions.height : 0,\n        }}\n        initial={{ borderWidth: 0, height: 0 }}\n        transition={{ duration: 0.175 }}\n        aria-hidden={!isOpen}\n      >\n        {displayGlobalNavRedesign ? (\n          <ColorMode mode=\"light\">\n            <StyledDropdownMenu {...commonMenuProps}>\n              <StyledLinkSection {...commonLinkSectionProps} />\n            </StyledDropdownMenu>\n          </ColorMode>\n        ) : (\n          <StyledDropdownMenu {...commonMenuProps}>\n            <StyledLinkSection {...commonLinkSectionProps} />\n          </StyledDropdownMenu>\n        )}\n      </AnimatedSimpleDropdown>\n    </Provider>\n  );\n};\n"]} */"));
49
+ }), ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx"],"names":[],"mappings":"AA8DE","file":"../../../../src/AppHeader/AppHeaderElements/AppHeaderDropdown/index.tsx","sourcesContent":["import { Menu } from '@codecademy/gamut';\nimport { ColorMode, css } from '@codecademy/gamut-styles';\nimport { Scale } from '@codecademy/variance/dist/types/config';\nimport styled from '@emotion/styled';\nimport {\n  useCallback,\n  useContext,\n  useEffect,\n  useMemo,\n  useRef,\n  useState,\n} from 'react';\nimport * as React from 'react';\nimport { useIsomorphicLayoutEffect } from 'react-use';\n\nimport { GlobalNavRedesignContext } from '../../../GlobalHeader';\nimport {\n  AnimatedSimpleDropdown,\n  AppHeaderAction,\n  AppHeaderDropdownItem,\n} from '../../shared';\nimport { AppHeaderDropdownProvider } from '../AppHeaderDropdownProvider';\nimport { AppHeaderLinkSections } from '../AppHeaderLinkSections';\nimport { AppHeaderMenuProvider } from '../AppHeaderMenuProvider';\nimport { AppHeaderNavButton } from '../AppHeaderNavButton';\nimport { useAppHeaderContext } from '../AppHeaderProvider';\n\nconst topSpacing = {\n  xl: '3.5rem',\n  lg: '3.2rem',\n  md: '2.75rem',\n  sm: '2.25rem',\n};\n\nconst getSpacing = ({\n  displayGlobalNavRedesign,\n  isProfileDropdown,\n  isGenericDropdown,\n  isIconDropdown,\n  isStandalone,\n}: {\n  displayGlobalNavRedesign?: boolean;\n  isProfileDropdown?: boolean;\n  isGenericDropdown?: boolean;\n  isIconDropdown?: boolean;\n  isStandalone?: boolean;\n}) => {\n  if (displayGlobalNavRedesign) return topSpacing.xl;\n  if (isProfileDropdown || isIconDropdown) return topSpacing.lg;\n  if (isGenericDropdown && isStandalone) return topSpacing.md;\n  return topSpacing.sm;\n};\n\nconst StyledLinkSection = styled(AppHeaderLinkSections)(\n  css({\n    padding: `0.75rem 0`,\n    position: `absolute`,\n  })\n);\n\nconst StyledDropdownMenu = styled(Menu)<{\n  displayGlobalNavRedesign: boolean;\n}>`\n  ${({ displayGlobalNavRedesign }) =>\n    displayGlobalNavRedesign &&\n    css({\n      border: 1,\n      borderColor: 'border-tertiary',\n      borderRadius: 'lg',\n    })}\n`;\n\nexport type AppHeaderDropdownProps = AppHeaderAction &\n  Pick<React.ComponentProps<typeof Menu>, 'spacing'> & {\n    item: AppHeaderDropdownItem;\n    onKeyDown?: (event: React.KeyboardEvent) => void;\n    /**\n     * If true, the dropdown is being rendered as a standalone component, rather than\n     * part of the global header.\n     */\n    standalone?: boolean;\n  };\n\nexport const AppHeaderDropdown: React.FC<AppHeaderDropdownProps> = ({\n  action,\n  item,\n  spacing = 'normal',\n  standalone,\n}) => {\n  const listRef = useRef<HTMLUListElement>(null);\n  const buttonRef = useRef<HTMLButtonElement>(null);\n\n  const [dimensions, setDimensions] = useState({ height: 0, width: 0 });\n  const [isOpen, setIsOpen] = useState(false);\n  const { lastOpenedDropdown, setLastOpenedDropdown } = useAppHeaderContext();\n  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);\n\n  const tabIndex = isOpen === false ? -1 : 0;\n  const dropdownId = item?.text ?? item?.type;\n\n  const toggleIsOpen = () => setIsOpen((prev) => !prev);\n\n  const handleOnClick = (event: React.MouseEvent) => {\n    toggleIsOpen();\n    if (!isOpen) {\n      if (action) action(event, item);\n      if (setLastOpenedDropdown) {\n        setLastOpenedDropdown(dropdownId);\n      }\n    }\n  };\n\n  const handleKeyboardClose = useCallback(() => {\n    setIsOpen(false);\n    buttonRef.current?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (lastOpenedDropdown !== dropdownId && isOpen) {\n      setIsOpen(false);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [lastOpenedDropdown]);\n\n  useIsomorphicLayoutEffect(() => {\n    if (listRef.current) {\n      const { height, width } = listRef.current.getBoundingClientRect();\n      setDimensions({ height, width });\n    }\n  }, [listRef, isOpen]);\n\n  useEffect(() => {\n    function handleClickOutside(event: MouseEvent | Event) {\n      const list = listRef?.current;\n      const button = buttonRef?.current;\n      if (\n        isOpen &&\n        list &&\n        !list.contains(event.target as Node) &&\n        button &&\n        !button.contains(event.target as Node)\n      ) {\n        setIsOpen(false);\n      }\n    }\n\n    document.addEventListener('mousedown', handleClickOutside);\n    document.addEventListener('blur', handleClickOutside);\n    return () => {\n      document.removeEventListener('mousedown', handleClickOutside);\n      document.removeEventListener('blur', handleClickOutside);\n    };\n  }, [listRef, isOpen]);\n\n  const isProfileDropdown = item.type === 'profile-dropdown';\n  const isGenericDropdown = ['dropdown', 'resources-simple-dropdown'].includes(\n    item.type\n  );\n  const isIconDropdown = 'isIconOnly' in item && item.isIconOnly;\n\n  const Provider = isProfileDropdown\n    ? AppHeaderMenuProvider\n    : AppHeaderDropdownProvider;\n\n  const topSpacing = useMemo(\n    () =>\n      getSpacing({\n        displayGlobalNavRedesign,\n        isProfileDropdown,\n        isGenericDropdown,\n        isIconDropdown,\n        isStandalone: standalone,\n      }),\n    [\n      isProfileDropdown,\n      isGenericDropdown,\n      isIconDropdown,\n      standalone,\n      displayGlobalNavRedesign,\n    ]\n  );\n\n  const commonMenuProps = {\n    displayGlobalNavRedesign,\n    minWidth: '202px',\n    ref: listRef,\n    role: isProfileDropdown ? 'menu' : undefined,\n    spacing,\n    variant: 'popover' as const,\n    width: 'fit-content' as const,\n    zIndex: 1,\n    py:\n      spacing === 'normal'\n        ? 12\n        : (0 as Scale<{\n            readonly property: 'padding';\n            readonly scale: 'spacing';\n          }>),\n  };\n\n  const commonLinkSectionProps = {\n    action,\n    item,\n    id: `menu-container${item.text}`,\n    'aria-controls': `menu-container${item.text}`,\n    'aria-label': item.text,\n    'aria-hidden': !isOpen,\n    tabIndex,\n  };\n\n  return (\n    <Provider handleClose={handleKeyboardClose}>\n      <AppHeaderNavButton\n        buttonRef={buttonRef}\n        handleOnClick={handleOnClick}\n        isOpen={isOpen}\n        item={item}\n      />\n      <AnimatedSimpleDropdown\n        isOpen={isOpen}\n        style={{\n          right: isProfileDropdown ? '0.5rem' : '',\n          top: topSpacing,\n          width: dimensions.width,\n        }}\n        displayGlobalNavRedesign={displayGlobalNavRedesign}\n      >\n        {displayGlobalNavRedesign ? (\n          <ColorMode mode=\"light\">\n            <StyledDropdownMenu {...commonMenuProps}>\n              <StyledLinkSection {...commonLinkSectionProps} />\n            </StyledDropdownMenu>\n          </ColorMode>\n        ) : (\n          <StyledDropdownMenu {...commonMenuProps}>\n            <StyledLinkSection {...commonLinkSectionProps} />\n          </StyledDropdownMenu>\n        )}\n      </AnimatedSimpleDropdown>\n    </Provider>\n  );\n};\n"]} */"));
47
50
  export const AppHeaderDropdown = ({
48
51
  action,
49
52
  item,
@@ -116,11 +119,12 @@ export const AppHeaderDropdown = ({
116
119
  const isIconDropdown = 'isIconOnly' in item && item.isIconOnly;
117
120
  const Provider = isProfileDropdown ? AppHeaderMenuProvider : AppHeaderDropdownProvider;
118
121
  const topSpacing = useMemo(() => getSpacing({
122
+ displayGlobalNavRedesign,
119
123
  isProfileDropdown,
120
124
  isGenericDropdown,
121
125
  isIconDropdown,
122
126
  isStandalone: standalone
123
- }), [isProfileDropdown, isGenericDropdown, isIconDropdown, standalone]);
127
+ }), [isProfileDropdown, isGenericDropdown, isIconDropdown, standalone, displayGlobalNavRedesign]);
124
128
  const commonMenuProps = {
125
129
  displayGlobalNavRedesign,
126
130
  minWidth: '202px',
@@ -149,22 +153,13 @@ export const AppHeaderDropdown = ({
149
153
  isOpen: isOpen,
150
154
  item: item
151
155
  }), /*#__PURE__*/_jsx(AnimatedSimpleDropdown, {
156
+ isOpen: isOpen,
152
157
  style: {
153
158
  right: isProfileDropdown ? '0.5rem' : '',
154
159
  top: topSpacing,
155
160
  width: dimensions.width
156
161
  },
157
- animate: {
158
- height: isOpen ? dimensions.height : 0
159
- },
160
- initial: {
161
- borderWidth: 0,
162
- height: 0
163
- },
164
- transition: {
165
- duration: 0.175
166
- },
167
- "aria-hidden": !isOpen,
162
+ displayGlobalNavRedesign: displayGlobalNavRedesign,
168
163
  children: displayGlobalNavRedesign ? /*#__PURE__*/_jsx(ColorMode, {
169
164
  mode: "light",
170
165
  children: /*#__PURE__*/_jsx(StyledDropdownMenu, {
@@ -14,8 +14,9 @@ export const AppHeaderDropdownLink = ({
14
14
  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);
15
15
  const ref = useRef(null);
16
16
  useEffect(() => {
17
- if (isFirstElem && setFirstItemRef && ref?.current) {
18
- setFirstItemRef(ref.current);
17
+ if (isFirstElem && ref?.current) {
18
+ setFirstItemRef?.(ref.current);
19
+ ref.current.focus();
19
20
  }
20
21
  }, [setFirstItemRef, ref, isFirstElem]);
21
22
  return /*#__PURE__*/_jsx(StyledAppHeaderLink, {
@@ -5,6 +5,7 @@ import { StyledAppHeaderLink } from './StyledAppHeaderLink';
5
5
  import { jsx as _jsx } from "react/jsx-runtime";
6
6
  export const AppHeaderMenuItem = ({
7
7
  role,
8
+ isFirstElem,
8
9
  ...props
9
10
  }) => {
10
11
  const {
@@ -17,6 +18,10 @@ export const AppHeaderMenuItem = ({
17
18
  if (setMenuItems && ref?.current) {
18
19
  setMenuItems(menuItems.add(ref.current));
19
20
  }
21
+ if (isFirstElem && ref?.current) {
22
+ ref.current.focus();
23
+ }
24
+ return () => menuItems.clear();
20
25
  // eslint-disable-next-line react-hooks/exhaustive-deps
21
26
  }, []);
22
27
  return /*#__PURE__*/_jsx(StyledAppHeaderLink, {
@@ -16,7 +16,10 @@ export const StyledAppHeaderLink = /*#__PURE__*/_styled(AppHeaderLink, {
16
16
  content: '""',
17
17
  position: 'absolute',
18
18
  width: 'calc(100% - 24px)',
19
- left: '12px',
19
+ left: {
20
+ _: 0,
21
+ lg: 12
22
+ },
20
23
  borderRadius: 'lg',
21
24
  transition: 'background-color 0.15s ease',
22
25
  pointerEvents: 'none',
@@ -32,4 +35,4 @@ export const StyledAppHeaderLink = /*#__PURE__*/_styled(AppHeaderLink, {
32
35
  '&:active::before': {
33
36
  backgroundColor: theme.colors['navy-200']
34
37
  }
35
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvQXBwSGVhZGVyRWxlbWVudHMvQXBwSGVhZGVyRHJvcGRvd25MaW5rcy9lbGVtZW50cy9TdHlsZWRBcHBIZWFkZXJMaW5rLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLbUMiLCJmaWxlIjoiLi4vLi4vLi4vLi4vLi4vc3JjL0FwcEhlYWRlci9BcHBIZWFkZXJFbGVtZW50cy9BcHBIZWFkZXJEcm9wZG93bkxpbmtzL2VsZW1lbnRzL1N0eWxlZEFwcEhlYWRlckxpbmsudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzLCB0aGVtZSB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5cbmltcG9ydCB7IEFwcEhlYWRlckxpbmsgfSBmcm9tICcuLi8uLi9BcHBIZWFkZXJMaW5rJztcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEhlYWRlckxpbmsgPSBzdHlsZWQoQXBwSGVhZGVyTGluayk8e1xuICBkaXNwbGF5R2xvYmFsTmF2UmVkZXNpZ24/OiBib29sZWFuO1xufT4oXG4gICh7IGRpc3BsYXlHbG9iYWxOYXZSZWRlc2lnbiB9KSA9PlxuICAgIGRpc3BsYXlHbG9iYWxOYXZSZWRlc2lnbiAmJlxuICAgIGNzcyh7XG4gICAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgICBwb3NpdGlvbjogJ3JlbGF0aXZlJyxcbiAgICAgICdAbWVkaWEgKG1heC13aWR0aDogMTE5OXB4KSc6IHtcbiAgICAgICAgcGw6IDAsXG4gICAgICB9LFxuICAgICAgJyY6OmJlZm9yZSc6IHtcbiAgICAgICAgY29udGVudDogJ1wiXCInLFxuICAgICAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICAgICAgd2lkdGg6ICdjYWxjKDEwMCUgLSAyNHB4KScsXG4gICAgICAgIGxlZnQ6ICcxMnB4JyxcbiAgICAgICAgYm9yZGVyUmFkaXVzOiAnbGcnLFxuICAgICAgICB0cmFuc2l0aW9uOiAnYmFja2dyb3VuZC1jb2xvciAwLjE1cyBlYXNlJyxcbiAgICAgICAgcG9pbnRlckV2ZW50czogJ25vbmUnLFxuICAgICAgICB6SW5kZXg6IC0xLFxuICAgICAgfSxcbiAgICAgICcmOmhvdmVyOjpiZWZvcmUsICY6Zm9jdXMtdmlzaWJsZTo6YmVmb3JlJzoge1xuICAgICAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLmNvbG9yc1snbmF2eS0xMDAnXSxcbiAgICAgIH0sXG4gICAgICAnJjpob3ZlciwgJjpmb2N1cy12aXNpYmxlLCAmOmFjdGl2ZSc6IHtcbiAgICAgICAgY29sb3I6ICdoeXBlci00MDAnLFxuICAgICAgICBib3JkZXJSYWRpdXM6ICdsZycsXG4gICAgICB9LFxuICAgICAgJyY6YWN0aXZlOjpiZWZvcmUnOiB7XG4gICAgICAgIGJhY2tncm91bmRDb2xvcjogdGhlbWUuY29sb3JzWyduYXZ5LTIwMCddLFxuICAgICAgfSxcbiAgICB9KVxuKTtcbiJdfQ== */");
38
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvQXBwSGVhZGVyRWxlbWVudHMvQXBwSGVhZGVyRHJvcGRvd25MaW5rcy9lbGVtZW50cy9TdHlsZWRBcHBIZWFkZXJMaW5rLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLbUMiLCJmaWxlIjoiLi4vLi4vLi4vLi4vLi4vc3JjL0FwcEhlYWRlci9BcHBIZWFkZXJFbGVtZW50cy9BcHBIZWFkZXJEcm9wZG93bkxpbmtzL2VsZW1lbnRzL1N0eWxlZEFwcEhlYWRlckxpbmsudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzLCB0aGVtZSB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5cbmltcG9ydCB7IEFwcEhlYWRlckxpbmsgfSBmcm9tICcuLi8uLi9BcHBIZWFkZXJMaW5rJztcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEhlYWRlckxpbmsgPSBzdHlsZWQoQXBwSGVhZGVyTGluayk8e1xuICBkaXNwbGF5R2xvYmFsTmF2UmVkZXNpZ24/OiBib29sZWFuO1xufT4oXG4gICh7IGRpc3BsYXlHbG9iYWxOYXZSZWRlc2lnbiB9KSA9PlxuICAgIGRpc3BsYXlHbG9iYWxOYXZSZWRlc2lnbiAmJlxuICAgIGNzcyh7XG4gICAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgICBwb3NpdGlvbjogJ3JlbGF0aXZlJyxcbiAgICAgICdAbWVkaWEgKG1heC13aWR0aDogMTE5OXB4KSc6IHtcbiAgICAgICAgcGw6IDAsXG4gICAgICB9LFxuICAgICAgJyY6OmJlZm9yZSc6IHtcbiAgICAgICAgY29udGVudDogJ1wiXCInLFxuICAgICAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICAgICAgd2lkdGg6ICdjYWxjKDEwMCUgLSAyNHB4KScsXG4gICAgICAgIGxlZnQ6IHsgXzogMCwgbGc6IDEyIH0sXG4gICAgICAgIGJvcmRlclJhZGl1czogJ2xnJyxcbiAgICAgICAgdHJhbnNpdGlvbjogJ2JhY2tncm91bmQtY29sb3IgMC4xNXMgZWFzZScsXG4gICAgICAgIHBvaW50ZXJFdmVudHM6ICdub25lJyxcbiAgICAgICAgekluZGV4OiAtMSxcbiAgICAgIH0sXG4gICAgICAnJjpob3Zlcjo6YmVmb3JlLCAmOmZvY3VzLXZpc2libGU6OmJlZm9yZSc6IHtcbiAgICAgICAgYmFja2dyb3VuZENvbG9yOiB0aGVtZS5jb2xvcnNbJ25hdnktMTAwJ10sXG4gICAgICB9LFxuICAgICAgJyY6aG92ZXIsICY6Zm9jdXMtdmlzaWJsZSwgJjphY3RpdmUnOiB7XG4gICAgICAgIGNvbG9yOiAnaHlwZXItNDAwJyxcbiAgICAgICAgYm9yZGVyUmFkaXVzOiAnbGcnLFxuICAgICAgfSxcbiAgICAgICcmOmFjdGl2ZTo6YmVmb3JlJzoge1xuICAgICAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLmNvbG9yc1snbmF2eS0yMDAnXSxcbiAgICAgIH0sXG4gICAgfSlcbik7XG4iXX0= */");
@@ -17,7 +17,7 @@ const LinkComponent = ({
17
17
  return /*#__PURE__*/_jsxs(_Fragment, {
18
18
  children: [showLineBreak && /*#__PURE__*/_jsx(MenuSeparator, {}), /*#__PURE__*/_jsx(DropdownLink, {
19
19
  action: action,
20
- isFirstElem: !isMenuItem ? isFirstElem : undefined,
20
+ isFirstElem: isFirstElem,
21
21
  item: link,
22
22
  mobile: mobile,
23
23
  onKeyDown: onKeyDown,
@@ -40,7 +40,8 @@ export const AppHeaderLinkSections = ({
40
40
  showLineBreak: sectionIndex !== 0 && linkIndex === 0,
41
41
  tabIndex: tabIndex,
42
42
  isMenuItem: true,
43
- mobile: mobile
43
+ mobile: mobile,
44
+ isFirstElem: sectionIndex === 0 && linkIndex === 0
44
45
  }, link.id))) : item.popover.map((link, linkIndex) => /*#__PURE__*/_jsx(LinkComponent, {
45
46
  onKeyDown: onKeyDown,
46
47
  action: action,
@@ -1,4 +1,5 @@
1
1
  import { IconButton } from '@codecademy/gamut';
2
+ import { useEffect } from 'react';
2
3
  import { DropdownAnchor, DropdownIcon, StyledText } from '../../shared';
3
4
  import { useAppHeaderDropdownContext } from '../AppHeaderDropdownProvider';
4
5
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
@@ -13,15 +14,14 @@ export const AppHeaderDropdownNavButton = ({
13
14
  const {
14
15
  firstItemRef
15
16
  } = useAppHeaderDropdownContext();
16
- const dropdownNavOnClick = e => {
17
- handleOnClick(e);
18
- if (!isOpen) {
17
+ useEffect(() => {
18
+ if (isOpen) {
19
19
  firstItemRef?.focus();
20
20
  }
21
- };
21
+ }, [isOpen, firstItemRef]);
22
22
  const sharedProps = {
23
23
  'aria-expanded': isOpen,
24
- onClick: dropdownNavOnClick,
24
+ onClick: handleOnClick,
25
25
  ref: buttonRef
26
26
  };
27
27
  if (icon && isIconOnly) return /*#__PURE__*/_jsx(IconButton, {
@@ -1,4 +1,5 @@
1
1
  import { TextButton } from '@codecademy/gamut';
2
+ import { useEffect } from 'react';
2
3
  import { Avatar } from '../../../Avatar';
3
4
  import { useAppHeaderMenuContext } from '../AppHeaderMenuProvider';
4
5
  import { jsx as _jsx } from "react/jsx-runtime";
@@ -12,17 +13,18 @@ export const AppHeaderMenuNavButton = ({
12
13
  menuItems,
13
14
  resetIndex
14
15
  } = useAppHeaderMenuContext();
15
- const profileOnClick = e => {
16
- const menuItemsArray = menuItems.values();
17
- const firstMenuItem = menuItemsArray.next().value;
18
- handleOnClick(e);
19
- firstMenuItem?.focus();
20
- };
16
+ useEffect(() => {
17
+ if (isOpen) {
18
+ const menuItemsArray = menuItems.values();
19
+ const firstMenuItem = menuItemsArray.next().value;
20
+ firstMenuItem?.focus();
21
+ }
22
+ }, [isOpen, menuItems]);
21
23
  return /*#__PURE__*/_jsx(TextButton, {
22
24
  "aria-expanded": isOpen,
23
25
  "aria-haspopup": true,
24
26
  "data-testid": "avatar-dropdown-button",
25
- onClick: profileOnClick
27
+ onClick: handleOnClick
26
28
  // this resets our programatic menu index to 0 in case someone clicks out of the dropdown
27
29
  ,
28
30
  onFocus: resetIndex,
@@ -14,6 +14,9 @@ export const DocsPanel = () => {
14
14
  const {
15
15
  tabIndex
16
16
  } = useAppHeaderSectionContext();
17
+ const {
18
+ handleClose
19
+ } = useAppHeaderDropdownContext();
17
20
  const description = /*#__PURE__*/_jsxs(_Fragment, {
18
21
  children: [/*#__PURE__*/_jsx(Box, {
19
22
  mb: 8,
@@ -21,12 +24,15 @@ export const DocsPanel = () => {
21
24
  }), "Interested in helping build it?", ' ', /*#__PURE__*/_jsx(Anchor, {
22
25
  tabIndex: tabIndex,
23
26
  href: "/pages/contribute-docs",
24
- onClick: event => globalHeaderItemClick(event, {
25
- type: 'link',
26
- href: '/pages/contribute-docs',
27
- text: 'Contribute to Docs',
28
- id: 'contribute-to-docs'
29
- }),
27
+ onClick: event => {
28
+ globalHeaderItemClick(event, {
29
+ type: 'link',
30
+ href: '/pages/contribute-docs',
31
+ text: 'Contribute to Docs',
32
+ id: 'contribute-to-docs'
33
+ });
34
+ handleClose();
35
+ },
30
36
  children: "Contribute to Docs"
31
37
  })]
32
38
  });
@@ -1,12 +1,5 @@
1
1
  import React from 'react';
2
- import { AppHeaderLinkItem, AppHeaderResourcesDropdownItem } from '../../shared';
3
- export type NavSectionConfig = {
4
- item: AppHeaderResourcesDropdownItem;
5
- panel: React.ComponentType<{
6
- tabIndex?: number;
7
- }>;
8
- };
9
- export declare const RESOURCES_NAV_SECTIONS: NavSectionConfig[];
2
+ import { AppHeaderLinkItem } from '../../shared';
10
3
  export declare const docsLinks: AppHeaderLinkItem[];
11
4
  export type AppHeaderCardItem = {
12
5
  linkItem: AppHeaderLinkItem;
@@ -1,44 +1,6 @@
1
- import { ArticleIcon, BriefcaseIcon, BulbIcon, CIcon, CommentsIcon, ComputerScienceIcon, CPlusIcon, FileIcon, GameDevelopmentIcon, GaugeDashboardIcon, GitIcon, GoIcon, HammerWrenchIcon, HtmlCssIcon, InfoCircleIcon, JavaIcon, JavascriptIcon, LearnIcon, MarkdownIcon, NotebookIcon, PeopleIcon, PhpIcon, ProgrammingBrowserIcon, ProjectsIcon, PythonIcon, ReactIcon, RulerTriangleIcon, SearchIcon, SqlIcon, SwiftIcon, TerminalIcon, TypescriptIcon, VideoPlayerMovieIcon } from '@codecademy/gamut-icons';
1
+ import { ArticleIcon, BriefcaseIcon, CIcon, CommentsIcon, ComputerScienceIcon, CPlusIcon, GameDevelopmentIcon, GaugeDashboardIcon, GitIcon, GoIcon, HtmlCssIcon, InfoCircleIcon, JavaIcon, JavascriptIcon, LearnIcon, MarkdownIcon, NotebookIcon, PeopleIcon, PhpIcon, ProgrammingBrowserIcon, ProjectsIcon, PythonIcon, ReactIcon, SearchIcon, SqlIcon, SwiftIcon, TerminalIcon, TypescriptIcon, VideoPlayerMovieIcon } from '@codecademy/gamut-icons';
2
2
  import React from 'react';
3
- import { DocsPanel, InspirationPanel, LearningToolsPanel, PracticeToolsPanel } from './NavPanels';
4
3
  import { jsx as _jsx } from "react/jsx-runtime";
5
- export const RESOURCES_NAV_SECTIONS = [{
6
- item: {
7
- icon: FileIcon,
8
- text: 'Docs',
9
- id: 'docs',
10
- type: 'resources-dropdown',
11
- trackingTarget: 'topnav_resources_tab_docs'
12
- },
13
- panel: DocsPanel
14
- }, {
15
- item: {
16
- icon: RulerTriangleIcon,
17
- text: 'Learning tools',
18
- id: 'learning-tools',
19
- type: 'resources-dropdown',
20
- trackingTarget: 'topnav_resources_tab_learning_tools'
21
- },
22
- panel: LearningToolsPanel
23
- }, {
24
- item: {
25
- icon: HammerWrenchIcon,
26
- text: 'Practice tools',
27
- id: 'practice-tools',
28
- type: 'resources-dropdown',
29
- trackingTarget: 'topnav_resources_tab_practice_tools'
30
- },
31
- panel: PracticeToolsPanel
32
- }, {
33
- item: {
34
- icon: BulbIcon,
35
- text: 'Inspiration',
36
- id: 'inspiration',
37
- type: 'resources-dropdown',
38
- trackingTarget: 'topnav_resources_tab_inspiration'
39
- },
40
- panel: InspirationPanel
41
- }];
42
4
  export const docsLinks = [{
43
5
  id: 'c',
44
6
  href: '/resources/docs/c',
@@ -11,11 +11,11 @@ import { AppHeaderDropdownNavButton } from '../AppHeaderNavButton/AppHeaderDropd
11
11
  import { useAppHeaderContext } from '../AppHeaderProvider';
12
12
  import { AppHeaderResourcesSection } from '../AppHeaderResourcesSection';
13
13
  import { AppHeaderSection } from '../AppHeaderSection';
14
- import { RESOURCES_NAV_SECTIONS } from './consts';
14
+ import { RESOURCES_TAB_SECTIONS } from './tabConfig';
15
15
  import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
16
16
  const StyledAnimatedMegaMenuDropdown = /*#__PURE__*/_styled(AnimatedMegaMenuDropdown, {
17
17
  // prevent the isAnon and displayGlobalNavRedesign props from being passed to the DOM element, which would cause a React warning.
18
- shouldForwardProp: prop => prop !== 'isAnon' && prop !== 'displayGlobalNavRedesign' && prop !== 'narrowResourceDropdownPosition' && prop !== 'wideResourceDropdownPosition',
18
+ shouldForwardProp: prop => prop !== 'isAnon' && prop !== 'narrowResourceDropdownPosition' && prop !== 'wideResourceDropdownPosition',
19
19
  target: "ekdg1bw0",
20
20
  label: "StyledAnimatedMegaMenuDropdown"
21
21
  })("left:", ({
@@ -32,7 +32,7 @@ const StyledAnimatedMegaMenuDropdown = /*#__PURE__*/_styled(AnimatedMegaMenuDrop
32
32
  }) => {
33
33
  if (displayGlobalNavRedesign) return wideResourceDropdownPosition;
34
34
  return isAnon ? '-14.5rem' : '-19.5rem';
35
- }, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.tsx"],"names":[],"mappings":"AAwCE","file":"../../../../src/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.tsx","sourcesContent":["import { ColorMode } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useCallback, useContext, useEffect, useRef, useState } from 'react';\nimport * as React from 'react';\n\nimport { GlobalNavRedesignContext } from '../../../GlobalHeader';\nimport {\n  DropdownPositions,\n  useHeaderDisplayContext,\n} from '../../../GlobalHeader/context';\nimport { getHeaderResourcesList } from '../../../lib/resourcesList';\nimport {\n  AnimatedMegaMenuDropdown,\n  AppHeaderAction,\n  AppHeaderResourcesDropdownItem,\n} from '../../shared';\nimport { AppHeaderDropdownProvider } from '../AppHeaderDropdownProvider';\nimport { AppHeaderDropdownNavButton } from '../AppHeaderNavButton/AppHeaderDropdownNavButton';\nimport { useAppHeaderContext } from '../AppHeaderProvider';\nimport { AppHeaderResourcesSection } from '../AppHeaderResourcesSection';\nimport { AppHeaderSection } from '../AppHeaderSection';\nimport { RESOURCES_NAV_SECTIONS } from './consts';\n\nexport type AppHeaderResourceDropdownProps = AppHeaderAction & {\n  item: AppHeaderResourcesDropdownItem;\n  isAnon: boolean;\n};\n\nconst StyledAnimatedMegaMenuDropdown = styled(AnimatedMegaMenuDropdown, {\n  // prevent the isAnon and displayGlobalNavRedesign props from being passed to the DOM element, which would cause a React warning.\n  shouldForwardProp: (prop) =>\n    prop !== 'isAnon' &&\n    prop !== 'displayGlobalNavRedesign' &&\n    prop !== 'narrowResourceDropdownPosition' &&\n    prop !== 'wideResourceDropdownPosition',\n})<{\n  isAnon: boolean;\n  displayGlobalNavRedesign: boolean;\n  narrowResourceDropdownPosition: string;\n  wideResourceDropdownPosition: string;\n}>`\n  left: ${({\n    isAnon,\n    displayGlobalNavRedesign,\n    narrowResourceDropdownPosition,\n  }) => {\n    if (displayGlobalNavRedesign) return narrowResourceDropdownPosition;\n    return isAnon ? '-14.5rem' : '-19.5rem';\n  }};\n\n  @media (min-width: 1261px) {\n    left: ${({\n      isAnon,\n      displayGlobalNavRedesign,\n      wideResourceDropdownPosition,\n    }) => {\n      if (displayGlobalNavRedesign) return wideResourceDropdownPosition;\n      return isAnon ? '-14.5rem' : '-19.5rem';\n    }};\n  }\n`;\n\nexport const AppHeaderResourcesDropdown: React.FC<\n  AppHeaderResourceDropdownProps\n> = ({ action, item, isAnon }) => {\n  const { text } = item;\n  const containerRef = useRef<HTMLDivElement | null>(null);\n  const buttonRef = useRef<HTMLButtonElement>(null);\n  const dropdownRef = useRef<HTMLDivElement | null>(null);\n  const [isOpen, setIsOpen] = useState(false);\n  const { lastOpenedDropdown, setLastOpenedDropdown } = useAppHeaderContext();\n  const { headerType } = useHeaderDisplayContext();\n  const { resourceDropdown } = DropdownPositions[headerType];\n  const narrowResourceDropdownPosition = resourceDropdown.narrow;\n  const wideResourceDropdownPosition = resourceDropdown.wide;\n\n  // Add scrollbar for short screens after transition (otherwise scrollbar appears and dissapears for adequate heights during transition)\n  const [animationCompleteStyles, setAnimationCompleteStyles] = useState(false);\n\n  const headerResourcesList = getHeaderResourcesList();\n\n  const animationFinished = () => {\n    setAnimationCompleteStyles(isOpen);\n    if (isOpen && dropdownRef.current) {\n      dropdownRef.current.scrollTop = 0;\n    }\n  };\n\n  const toggleIsOpen = () => setIsOpen((prev) => !prev);\n\n  const handleOnClick = (event: React.MouseEvent) => {\n    toggleIsOpen();\n    if (!isOpen) {\n      action(event, item);\n      if (setLastOpenedDropdown) {\n        setLastOpenedDropdown(text);\n      }\n    }\n  };\n\n  const handleClose = useCallback(() => {\n    setIsOpen(false);\n    buttonRef?.current?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (lastOpenedDropdown !== text && isOpen) {\n      setIsOpen(false);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [lastOpenedDropdown]);\n\n  useEffect(() => {\n    function handleClickOutside(event: MouseEvent | Event) {\n      const container = containerRef?.current;\n      const button = buttonRef?.current;\n      if (\n        isOpen &&\n        container &&\n        !container.contains(event.target as Node) &&\n        button &&\n        !button.contains(event.target as Node)\n      ) {\n        handleClose();\n      }\n    }\n\n    document.addEventListener('mousedown', handleClickOutside);\n    document.addEventListener('blur', handleClickOutside);\n    return () => {\n      document.removeEventListener('mousedown', handleClickOutside);\n      document.removeEventListener('blur', handleClickOutside);\n    };\n  }, [containerRef, handleClose, isOpen]);\n\n  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);\n\n  return (\n    <AppHeaderDropdownProvider handleClose={handleClose}>\n      <AppHeaderDropdownNavButton\n        buttonRef={buttonRef}\n        handleOnClick={handleOnClick}\n        isOpen={isOpen}\n        title={text}\n      />\n      <StyledAnimatedMegaMenuDropdown\n        isAnon={isAnon}\n        displayGlobalNavRedesign={displayGlobalNavRedesign}\n        narrowResourceDropdownPosition={narrowResourceDropdownPosition}\n        wideResourceDropdownPosition={wideResourceDropdownPosition}\n        tabIndex={-1} // prevent from interfering with AppHeader focus logic\n        ref={dropdownRef}\n        style={{\n          top: '3.5rem',\n          minWidth: '64rem',\n          overflowY: 'auto',\n          maxHeight: animationCompleteStyles ? '85vh' : 0,\n          backgroundColor: displayGlobalNavRedesign ? 'transparent' : '',\n        }}\n        initial={{ borderWidth: 0, height: 0 }}\n        animate={{\n          borderWidth: isOpen ? (displayGlobalNavRedesign ? 0 : 1) : 0,\n          height: isOpen ? 'fit-content' : 0,\n        }}\n        transition={{ duration: 0.175 }}\n        aria-hidden={!isOpen}\n        data-testid=\"resources-menu-dropdown\"\n        onAnimationComplete={animationFinished}\n      >\n        {displayGlobalNavRedesign ? (\n          <ColorMode mode=\"light\">\n            <AppHeaderSection\n              isOpen={isOpen}\n              ref={containerRef}\n              navSections={RESOURCES_NAV_SECTIONS}\n              handleClose={() => setIsOpen(false)}\n              type=\"resources-dropdown\"\n            />\n          </ColorMode>\n        ) : (\n          <AppHeaderResourcesSection\n            action={action}\n            ref={containerRef}\n            id={`menu-container${item.text}`}\n            isOpen={isOpen}\n            handleClose={() => setIsOpen(false)}\n            resources={headerResourcesList}\n          />\n        )}\n      </StyledAnimatedMegaMenuDropdown>\n    </AppHeaderDropdownProvider>\n  );\n};\n"]} */"));
35
+ }, ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.tsx"],"names":[],"mappings":"AAuCE","file":"../../../../src/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.tsx","sourcesContent":["import { ColorMode } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useCallback, useContext, useEffect, useRef, useState } from 'react';\nimport * as React from 'react';\n\nimport { GlobalNavRedesignContext } from '../../../GlobalHeader';\nimport {\n  DropdownPositions,\n  useHeaderDisplayContext,\n} from '../../../GlobalHeader/context';\nimport { getHeaderResourcesList } from '../../../lib/resourcesList';\nimport {\n  AnimatedMegaMenuDropdown,\n  AppHeaderAction,\n  AppHeaderResourcesDropdownItem,\n} from '../../shared';\nimport { AppHeaderDropdownProvider } from '../AppHeaderDropdownProvider';\nimport { AppHeaderDropdownNavButton } from '../AppHeaderNavButton/AppHeaderDropdownNavButton';\nimport { useAppHeaderContext } from '../AppHeaderProvider';\nimport { AppHeaderResourcesSection } from '../AppHeaderResourcesSection';\nimport { AppHeaderSection } from '../AppHeaderSection';\nimport { RESOURCES_TAB_SECTIONS } from './tabConfig';\n\nexport type AppHeaderResourceDropdownProps = AppHeaderAction & {\n  item: AppHeaderResourcesDropdownItem;\n  isAnon: boolean;\n};\n\nconst StyledAnimatedMegaMenuDropdown = styled(AnimatedMegaMenuDropdown, {\n  // prevent the isAnon and displayGlobalNavRedesign props from being passed to the DOM element, which would cause a React warning.\n  shouldForwardProp: (prop) =>\n    prop !== 'isAnon' &&\n    prop !== 'narrowResourceDropdownPosition' &&\n    prop !== 'wideResourceDropdownPosition',\n})<{\n  isAnon: boolean;\n  displayGlobalNavRedesign: boolean;\n  narrowResourceDropdownPosition: string;\n  wideResourceDropdownPosition: string;\n}>`\n  left: ${({\n    isAnon,\n    displayGlobalNavRedesign,\n    narrowResourceDropdownPosition,\n  }) => {\n    if (displayGlobalNavRedesign) return narrowResourceDropdownPosition;\n    return isAnon ? '-14.5rem' : '-19.5rem';\n  }};\n\n  @media (min-width: 1261px) {\n    left: ${({\n      isAnon,\n      displayGlobalNavRedesign,\n      wideResourceDropdownPosition,\n    }) => {\n      if (displayGlobalNavRedesign) return wideResourceDropdownPosition;\n      return isAnon ? '-14.5rem' : '-19.5rem';\n    }};\n  }\n`;\n\nexport const AppHeaderResourcesDropdown: React.FC<\n  AppHeaderResourceDropdownProps\n> = ({ action, item, isAnon }) => {\n  const { text } = item;\n  const containerRef = useRef<HTMLDivElement | null>(null);\n  const buttonRef = useRef<HTMLButtonElement>(null);\n  const dropdownRef = useRef<HTMLDivElement | null>(null);\n  const [isOpen, setIsOpen] = useState(false);\n  const { lastOpenedDropdown, setLastOpenedDropdown } = useAppHeaderContext();\n  const { headerType } = useHeaderDisplayContext();\n  const { resourceDropdown } = DropdownPositions[headerType];\n  const narrowResourceDropdownPosition = resourceDropdown.narrow;\n  const wideResourceDropdownPosition = resourceDropdown.wide;\n\n  const headerResourcesList = getHeaderResourcesList();\n\n  const animationFinished = () => {\n    if (isOpen && dropdownRef.current) {\n      dropdownRef.current.scrollTop = 0;\n    }\n  };\n\n  const toggleIsOpen = () => setIsOpen((prev) => !prev);\n\n  const handleOnClick = (event: React.MouseEvent) => {\n    toggleIsOpen();\n    if (!isOpen) {\n      action(event, item);\n      if (setLastOpenedDropdown) {\n        setLastOpenedDropdown(text);\n      }\n    }\n  };\n\n  const handleClose = useCallback(() => {\n    setIsOpen(false);\n    buttonRef?.current?.focus();\n  }, []);\n\n  useEffect(() => {\n    if (lastOpenedDropdown !== text && isOpen) {\n      setIsOpen(false);\n    }\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [lastOpenedDropdown]);\n\n  useEffect(() => {\n    function handleClickOutside(event: MouseEvent | Event) {\n      const container = containerRef?.current;\n      const button = buttonRef?.current;\n      if (\n        isOpen &&\n        container &&\n        !container.contains(event.target as Node) &&\n        button &&\n        !button.contains(event.target as Node)\n      ) {\n        handleClose();\n      }\n    }\n\n    document.addEventListener('mousedown', handleClickOutside);\n    document.addEventListener('blur', handleClickOutside);\n    return () => {\n      document.removeEventListener('mousedown', handleClickOutside);\n      document.removeEventListener('blur', handleClickOutside);\n    };\n  }, [containerRef, handleClose, isOpen]);\n\n  const displayGlobalNavRedesign = useContext(GlobalNavRedesignContext);\n\n  return (\n    <AppHeaderDropdownProvider handleClose={handleClose}>\n      <AppHeaderDropdownNavButton\n        buttonRef={buttonRef}\n        handleOnClick={handleOnClick}\n        isOpen={isOpen}\n        title={text}\n      />\n      <StyledAnimatedMegaMenuDropdown\n        isOpen={isOpen}\n        isAnon={isAnon}\n        displayGlobalNavRedesign={displayGlobalNavRedesign}\n        narrowResourceDropdownPosition={narrowResourceDropdownPosition}\n        wideResourceDropdownPosition={wideResourceDropdownPosition}\n        tabIndex={-1} // prevent from interfering with AppHeader focus logic\n        ref={dropdownRef}\n        style={{\n          top: '3.5rem',\n          minWidth: '64rem',\n          overflowY: 'auto',\n          maxHeight: '85vh',\n          backgroundColor: displayGlobalNavRedesign ? 'transparent' : '',\n        }}\n        data-testid=\"resources-menu-dropdown\"\n        onAnimationComplete={animationFinished}\n      >\n        {displayGlobalNavRedesign ? (\n          <ColorMode mode=\"light\">\n            <AppHeaderSection\n              isOpen={isOpen}\n              ref={containerRef}\n              handleClose={() => setIsOpen(false)}\n              type=\"resources-dropdown\"\n              tabSections={RESOURCES_TAB_SECTIONS}\n            />\n          </ColorMode>\n        ) : (\n          <AppHeaderResourcesSection\n            action={action}\n            ref={containerRef}\n            id={`menu-container${item.text}`}\n            isOpen={isOpen}\n            handleClose={() => setIsOpen(false)}\n            resources={headerResourcesList}\n          />\n        )}\n      </StyledAnimatedMegaMenuDropdown>\n    </AppHeaderDropdownProvider>\n  );\n};\n"]} */"));
36
36
  export const AppHeaderResourcesDropdown = ({
37
37
  action,
38
38
  item,
@@ -57,12 +57,8 @@ export const AppHeaderResourcesDropdown = ({
57
57
  } = DropdownPositions[headerType];
58
58
  const narrowResourceDropdownPosition = resourceDropdown.narrow;
59
59
  const wideResourceDropdownPosition = resourceDropdown.wide;
60
-
61
- // Add scrollbar for short screens after transition (otherwise scrollbar appears and dissapears for adequate heights during transition)
62
- const [animationCompleteStyles, setAnimationCompleteStyles] = useState(false);
63
60
  const headerResourcesList = getHeaderResourcesList();
64
61
  const animationFinished = () => {
65
- setAnimationCompleteStyles(isOpen);
66
62
  if (isOpen && dropdownRef.current) {
67
63
  dropdownRef.current.scrollTop = 0;
68
64
  }
@@ -111,6 +107,7 @@ export const AppHeaderResourcesDropdown = ({
111
107
  isOpen: isOpen,
112
108
  title: text
113
109
  }), /*#__PURE__*/_jsx(StyledAnimatedMegaMenuDropdown, {
110
+ isOpen: isOpen,
114
111
  isAnon: isAnon,
115
112
  displayGlobalNavRedesign: displayGlobalNavRedesign,
116
113
  narrowResourceDropdownPosition: narrowResourceDropdownPosition,
@@ -122,21 +119,9 @@ export const AppHeaderResourcesDropdown = ({
122
119
  top: '3.5rem',
123
120
  minWidth: '64rem',
124
121
  overflowY: 'auto',
125
- maxHeight: animationCompleteStyles ? '85vh' : 0,
122
+ maxHeight: '85vh',
126
123
  backgroundColor: displayGlobalNavRedesign ? 'transparent' : ''
127
124
  },
128
- initial: {
129
- borderWidth: 0,
130
- height: 0
131
- },
132
- animate: {
133
- borderWidth: isOpen ? displayGlobalNavRedesign ? 0 : 1 : 0,
134
- height: isOpen ? 'fit-content' : 0
135
- },
136
- transition: {
137
- duration: 0.175
138
- },
139
- "aria-hidden": !isOpen,
140
125
  "data-testid": "resources-menu-dropdown",
141
126
  onAnimationComplete: animationFinished,
142
127
  children: displayGlobalNavRedesign ? /*#__PURE__*/_jsx(ColorMode, {
@@ -144,9 +129,9 @@ export const AppHeaderResourcesDropdown = ({
144
129
  children: /*#__PURE__*/_jsx(AppHeaderSection, {
145
130
  isOpen: isOpen,
146
131
  ref: containerRef,
147
- navSections: RESOURCES_NAV_SECTIONS,
148
132
  handleClose: () => setIsOpen(false),
149
- type: "resources-dropdown"
133
+ type: "resources-dropdown",
134
+ tabSections: RESOURCES_TAB_SECTIONS
150
135
  })
151
136
  }) : /*#__PURE__*/_jsx(AppHeaderResourcesSection, {
152
137
  action: action,
@@ -0,0 +1,2 @@
1
+ import { TabSectionConfig } from '../AppHeaderSection';
2
+ export declare const RESOURCES_TAB_SECTIONS: TabSectionConfig[];
@@ -0,0 +1,39 @@
1
+ import { BulbIcon, FileIcon, HammerWrenchIcon, RulerTriangleIcon } from '@codecademy/gamut-icons';
2
+ import { DocsPanel, InspirationPanel, LearningToolsPanel, PracticeToolsPanel } from './NavPanels';
3
+ export const RESOURCES_TAB_SECTIONS = [{
4
+ item: {
5
+ id: 'docs',
6
+ text: 'Docs',
7
+ type: 'resources-dropdown',
8
+ trackingTarget: 'topnav_resources_tab_docs',
9
+ icon: FileIcon
10
+ },
11
+ panel: DocsPanel
12
+ }, {
13
+ item: {
14
+ id: 'learning-tools',
15
+ text: 'Learning tools',
16
+ type: 'resources-dropdown',
17
+ trackingTarget: 'topnav_resources_tab_learning_tools',
18
+ icon: RulerTriangleIcon
19
+ },
20
+ panel: LearningToolsPanel
21
+ }, {
22
+ item: {
23
+ id: 'practice-tools',
24
+ text: 'Practice tools',
25
+ type: 'resources-dropdown',
26
+ trackingTarget: 'topnav_resources_tab_practice_tools',
27
+ icon: HammerWrenchIcon
28
+ },
29
+ panel: PracticeToolsPanel
30
+ }, {
31
+ item: {
32
+ id: 'inspiration',
33
+ text: 'Inspiration',
34
+ type: 'resources-dropdown',
35
+ trackingTarget: 'topnav_resources_tab_inspiration',
36
+ icon: BulbIcon
37
+ },
38
+ panel: InspirationPanel
39
+ }];