@codecademy/brand 3.26.0-alpha.fa0b72ae29.0 → 3.26.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 (42) hide show
  1. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/index.js +6 -28
  2. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdown/index.d.ts +0 -1
  3. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdown/index.js +10 -67
  4. package/dist/AppHeader/AppHeaderElements/{AppHeaderDropdownLinks/elements/StyledAppHeaderLink.d.ts → AppHeaderLinkSections/elements.d.ts} +6 -6
  5. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/elements.js +36 -0
  6. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/index.d.ts +1 -8
  7. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/index.js +32 -25
  8. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/index.d.ts +5 -2
  9. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/index.js +91 -24
  10. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.js +5 -27
  11. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/NavSection.d.ts +2 -2
  12. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/NavSection.js +38 -20
  13. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/elements.js +8 -8
  14. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/index.js +5 -13
  15. package/dist/AppHeader/Search/SearchPane.js +49 -24
  16. package/dist/AppHeader/shared/elements.d.ts +9 -3
  17. package/dist/AppHeader/shared/elements.js +126 -26
  18. package/dist/AppHeader/shared/types.d.ts +1 -1
  19. package/dist/AppHeader/shared/utils.d.ts +1 -2
  20. package/dist/AppHeader/shared/utils.js +0 -2
  21. package/dist/AppHeaderMobile/index.js +96 -100
  22. package/dist/PlanCard/PricingAmount.js +6 -6
  23. package/dist/PlanCard/types.d.ts +1 -0
  24. package/dist/PlanCard/types.js +13 -1
  25. package/package.json +6 -1
  26. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderDropdownLink.d.ts +0 -3
  27. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderDropdownLink.js +0 -25
  28. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderMenuItem.d.ts +0 -3
  29. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderMenuItem.js +0 -26
  30. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/StyledAppHeaderLink.js +0 -36
  31. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/types.d.ts +0 -4
  32. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/types.js +0 -1
  33. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/index.d.ts +0 -2
  34. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/index.js +0 -2
  35. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderDropdownNavButton.d.ts +0 -18
  36. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderDropdownNavButton.js +0 -50
  37. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderMenuNavButton.d.ts +0 -9
  38. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderMenuNavButton.js +0 -38
  39. package/dist/AppHeader/Search/hooks/useSearchTracking.d.ts +0 -5
  40. package/dist/AppHeader/Search/hooks/useSearchTracking.js +0 -46
  41. package/dist/AppHeader/utils/string-similarity.d.ts +0 -17
  42. package/dist/AppHeader/utils/string-similarity.js +0 -30
@@ -5,7 +5,6 @@ import { Background, css, states, theme } from '@codecademy/gamut-styles';
5
5
  import { useEffect, useRef, useState } from 'react';
6
6
  import * as React from 'react';
7
7
  import { AppHeaderListItem } from '../AppHeader/AppHeaderElements/AppHeaderListItem';
8
- import { SearchTrackingProvider } from '../AppHeader/Search/SearchTrackingProvider';
9
8
  import { useHeaderSearch } from '../AppHeader/Search/useHeaderSearch';
10
9
  import { appHeaderMobileBreakpoint, StyledAppBar } from '../AppHeader/shared';
11
10
  import { mapAppHeaderItemToElement } from '../AppHeader/shared/utils';
@@ -30,7 +29,7 @@ const StyledOverlay = /*#__PURE__*/_styled(Overlay, {
30
29
  left: 0,
31
30
  top: 0,
32
31
  overflowX: `hidden`
33
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/AppHeaderMobile/index.tsx"],"names":[],"mappings":"AA8CsB","file":"../../src/AppHeaderMobile/index.tsx","sourcesContent":["import {\n  Box,\n  ButtonBaseElements,\n  IconButton,\n  Overlay,\n} from '@codecademy/gamut';\nimport { MenuIcon } from '@codecademy/gamut-icons';\nimport { Background, css, states, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useEffect, useRef, useState } from 'react';\nimport * as React from 'react';\n\nimport { AppHeaderSearch } from '..';\nimport { AppHeaderListItem } from '../AppHeader/AppHeaderElements/AppHeaderListItem';\nimport { SearchTrackingProvider } from '../AppHeader/Search/SearchTrackingProvider';\nimport { useHeaderSearch } from '../AppHeader/Search/useHeaderSearch';\nimport {\n  AppHeaderAction,\n  AppHeaderItem,\n  appHeaderMobileBreakpoint,\n  FormattedMobileAppHeaderItems,\n  StyledAppBar,\n} from '../AppHeader/shared';\nimport { mapAppHeaderItemToElement } from '../AppHeader/shared/utils';\nimport { AppHeaderMainMenuMobile } from '../AppHeaderMobile/AppHeaderMainMenuMobile';\nimport { NavigationMenuFormattedLabel } from '../GlobalHeader';\nimport { HeaderHeightArea } from '../HeaderHeightArea';\nimport { NotificationsContents } from '../Notifications/NotificationsContents';\nimport { AppHeaderNotificationSettings } from '../Notifications/types';\nimport { useHeaderNotifications } from '../Notifications/useHeaderNotifications';\n\nexport type AppHeaderMobileProps = AppHeaderAction & {\n  items: FormattedMobileAppHeaderItems;\n  notifications?: AppHeaderNotificationSettings;\n  redirectParam?: string;\n  search: AppHeaderSearch;\n  isAnon: boolean;\n  /**\n   * used to conditonally hide the default search icon and notification bell\n   */\n  isEnterprise?: boolean;\n  isSimple?: boolean;\n  hideRightMenuButton?: boolean;\n  navigationMenuFormattedLabel?: NavigationMenuFormattedLabel;\n};\n\nconst StyledOverlay = styled(Overlay)(\n  css({\n    display: { _: `block`, [appHeaderMobileBreakpoint]: `none` },\n    width: `100vw`,\n    height: `100vh`,\n    opacity: 1,\n    bg: `beige`,\n    position: `fixed`,\n    left: 0,\n    top: 0,\n    overflowX: `hidden`,\n  })\n);\n\nconst StyledNavBar = styled.ul<{ center?: boolean }>(\n  css({\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n    alignItems: 'center',\n  }),\n  states({\n    center: {\n      justifyContent: {\n        _: 'center',\n        sm: 'flex-start',\n      },\n    },\n  })\n);\n\nexport const AppHeaderMobile: React.FC<AppHeaderMobileProps> = ({\n  action,\n  items,\n  notifications,\n  search,\n  redirectParam,\n  isAnon,\n  isEnterprise,\n  isSimple,\n  hideRightMenuButton,\n  navigationMenuFormattedLabel,\n}) => {\n  const [mobileMenuOpen, setMobileMenuOpen] = useState<boolean | undefined>(\n    undefined\n  );\n  const [allowScroll, setAllowScroll] = useState<boolean>(false);\n  const openButtonRef = useRef<ButtonBaseElements>(null);\n  const closeButtonRef = useRef<ButtonBaseElements>(null);\n\n  const [notificationsBell, notificationsView] = useHeaderNotifications({\n    settings: notifications,\n    Renderer: NotificationsContents,\n  });\n\n  const [searchButton, searchPane] = useHeaderSearch({\n    ...search,\n  });\n\n  const openMobileMenu = () => {\n    setMobileMenuOpen(true);\n\n    if (closeButtonRef.current) {\n      closeButtonRef.current.focus();\n    }\n  };\n\n  useEffect(() => {\n    if (mobileMenuOpen === false && openButtonRef.current) {\n      setTimeout(() => {\n        if (openButtonRef.current) openButtonRef.current.focus();\n      }, 0);\n    }\n  }, [mobileMenuOpen]);\n\n  const mapItemsToElement = <T extends AppHeaderItem[]>(\n    items: T,\n    side: 'left' | 'right',\n    hideExtraItems?: boolean\n  ) => {\n    const shouldHideItems = hideExtraItems === true && items.length > 1;\n    return items.map((item, index) => {\n      const isLastItem = index + 1 === items.length;\n      const isHidable = !isLastItem && shouldHideItems;\n      return (\n        <AppHeaderListItem\n          key={item.id}\n          ml={side === 'right' && index === 0 ? 'auto' : 0}\n          display={{\n            _: isHidable ? 'none' : 'flex',\n            xs: 'flex',\n          }}\n        >\n          {mapAppHeaderItemToElement({\n            action,\n            isStandalone: undefined,\n            isTeams: undefined,\n            item,\n            mobile: true,\n            onKeyDown: undefined,\n            redirectParam,\n          })}\n        </AppHeaderListItem>\n      );\n    });\n  };\n\n  const right = [\n    ...(!isEnterprise ? [searchButton] : []),\n    ...(notificationsBell && !isEnterprise ? [notificationsBell] : []),\n    ...items.right,\n  ];\n\n  const onItemType = (type: string | undefined) => {\n    if (\n      type &&\n      (type === 'catalog-dropdown' || type === 'resources-dropdown')\n    ) {\n      setAllowScroll(true);\n    } else {\n      setAllowScroll(false);\n    }\n  };\n\n  return (\n    <>\n      <SearchTrackingProvider>\n        {!mobileMenuOpen && ( // need this bc AppBar has a hardcoded z-Index of 15\n          <HeaderHeightArea\n            display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n          >\n            <StyledAppBar as=\"nav\" aria-label=\"Main\">\n              <StyledNavBar center={!!isSimple}>\n                {mapItemsToElement(items.left, 'left')}\n                {mapItemsToElement(right, 'right', true)}\n                {!hideRightMenuButton && (\n                  <AppHeaderListItem ml={right.length === 0 ? 'auto' : 0}>\n                    <IconButton\n                      aria-expanded={mobileMenuOpen}\n                      aria-label={\n                        navigationMenuFormattedLabel?.siteNavigation ||\n                        'Site navigation'\n                      }\n                      tip={'Site\\nnavigation'}\n                      tipProps={{\n                        alignment: 'bottom-center',\n                        placement: 'floating',\n                      }}\n                      data-testid=\"header-mobile-menu\"\n                      onClick={() => {\n                        openMobileMenu();\n                      }}\n                      icon={MenuIcon}\n                      variant=\"interface\"\n                      ref={openButtonRef}\n                    />\n                  </AppHeaderListItem>\n                )}\n              </StyledNavBar>\n            </StyledAppBar>\n          </HeaderHeightArea>\n        )}\n        <StyledOverlay\n          clickOutsideCloses\n          escapeCloses\n          isOpen={mobileMenuOpen}\n          onRequestClose={() => setMobileMenuOpen(false)}\n          allowScroll={allowScroll}\n        >\n          <Background bg=\"beige\">\n            <HeaderHeightArea\n              display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n              as=\"nav\"\n              ariaLabel=\"Main\"\n              data-testid=\"header-mobile-menu-dropdown\"\n            >\n              <StyledAppBar>\n                <StyledNavBar>\n                  {mapItemsToElement(items.left, 'left')}\n                  <AppHeaderListItem ml=\"auto\">\n                    <IconButton\n                      aria-expanded={mobileMenuOpen}\n                      onClick={() => {\n                        setMobileMenuOpen(false);\n                      }}\n                      icon={MenuIcon}\n                      aria-label={\n                        navigationMenuFormattedLabel?.siteNavigation ||\n                        'Site navigation'\n                      }\n                      tip={\n                        navigationMenuFormattedLabel?.siteNavigation ||\n                        'Site navigation'\n                      }\n                      tipProps={{\n                        alignment: 'bottom-center',\n                        placement: 'floating',\n                      }}\n                      ref={closeButtonRef}\n                    />\n                  </AppHeaderListItem>\n                </StyledNavBar>\n              </StyledAppBar>\n              <Box background={theme.colors.beige} height=\"auto\">\n                <AppHeaderMainMenuMobile\n                  action={action}\n                  items={items.mainMenu}\n                  getItemType={onItemType}\n                  isAnon={isAnon}\n                />\n              </Box>\n            </HeaderHeightArea>\n          </Background>\n        </StyledOverlay>\n        {!isEnterprise && (\n          <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n            {searchPane}\n          </Box>\n        )}\n        <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n          {notificationsView}\n        </Box>\n      </SearchTrackingProvider>\n    </>\n  );\n};\n"]} */");
32
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/AppHeaderMobile/index.tsx"],"names":[],"mappings":"AA6CsB","file":"../../src/AppHeaderMobile/index.tsx","sourcesContent":["import {\n  Box,\n  ButtonBaseElements,\n  IconButton,\n  Overlay,\n} from '@codecademy/gamut';\nimport { MenuIcon } from '@codecademy/gamut-icons';\nimport { Background, css, states, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useEffect, useRef, useState } from 'react';\nimport * as React from 'react';\n\nimport { AppHeaderSearch } from '..';\nimport { AppHeaderListItem } from '../AppHeader/AppHeaderElements/AppHeaderListItem';\nimport { useHeaderSearch } from '../AppHeader/Search/useHeaderSearch';\nimport {\n  AppHeaderAction,\n  AppHeaderItem,\n  appHeaderMobileBreakpoint,\n  FormattedMobileAppHeaderItems,\n  StyledAppBar,\n} from '../AppHeader/shared';\nimport { mapAppHeaderItemToElement } from '../AppHeader/shared/utils';\nimport { AppHeaderMainMenuMobile } from '../AppHeaderMobile/AppHeaderMainMenuMobile';\nimport { NavigationMenuFormattedLabel } from '../GlobalHeader';\nimport { HeaderHeightArea } from '../HeaderHeightArea';\nimport { NotificationsContents } from '../Notifications/NotificationsContents';\nimport { AppHeaderNotificationSettings } from '../Notifications/types';\nimport { useHeaderNotifications } from '../Notifications/useHeaderNotifications';\n\nexport type AppHeaderMobileProps = AppHeaderAction & {\n  items: FormattedMobileAppHeaderItems;\n  notifications?: AppHeaderNotificationSettings;\n  redirectParam?: string;\n  search: AppHeaderSearch;\n  isAnon: boolean;\n  /**\n   * used to conditonally hide the default search icon and notification bell\n   */\n  isEnterprise?: boolean;\n  isSimple?: boolean;\n  hideRightMenuButton?: boolean;\n  navigationMenuFormattedLabel?: NavigationMenuFormattedLabel;\n};\n\nconst StyledOverlay = styled(Overlay)(\n  css({\n    display: { _: `block`, [appHeaderMobileBreakpoint]: `none` },\n    width: `100vw`,\n    height: `100vh`,\n    opacity: 1,\n    bg: `beige`,\n    position: `fixed`,\n    left: 0,\n    top: 0,\n    overflowX: `hidden`,\n  })\n);\n\nconst StyledNavBar = styled.ul<{ center?: boolean }>(\n  css({\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n    alignItems: 'center',\n  }),\n  states({\n    center: {\n      justifyContent: {\n        _: 'center',\n        sm: 'flex-start',\n      },\n    },\n  })\n);\n\nexport const AppHeaderMobile: React.FC<AppHeaderMobileProps> = ({\n  action,\n  items,\n  notifications,\n  search,\n  redirectParam,\n  isAnon,\n  isEnterprise,\n  isSimple,\n  hideRightMenuButton,\n  navigationMenuFormattedLabel,\n}) => {\n  const [mobileMenuOpen, setMobileMenuOpen] = useState<boolean | undefined>(\n    undefined\n  );\n  const [allowScroll, setAllowScroll] = useState<boolean>(false);\n  const openButtonRef = useRef<ButtonBaseElements>(null);\n  const closeButtonRef = useRef<ButtonBaseElements>(null);\n\n  const [notificationsBell, notificationsView] = useHeaderNotifications({\n    settings: notifications,\n    Renderer: NotificationsContents,\n  });\n\n  const [searchButton, searchPane] = useHeaderSearch({\n    ...search,\n  });\n\n  const openMobileMenu = () => {\n    setMobileMenuOpen(true);\n\n    if (closeButtonRef.current) {\n      closeButtonRef.current.focus();\n    }\n  };\n\n  useEffect(() => {\n    if (mobileMenuOpen === false && openButtonRef.current) {\n      setTimeout(() => {\n        if (openButtonRef.current) openButtonRef.current.focus();\n      }, 0);\n    }\n  }, [mobileMenuOpen]);\n\n  const mapItemsToElement = <T extends AppHeaderItem[]>(\n    items: T,\n    side: 'left' | 'right',\n    hideExtraItems?: boolean\n  ) => {\n    const shouldHideItems = hideExtraItems === true && items.length > 1;\n    return items.map((item, index) => {\n      const isLastItem = index + 1 === items.length;\n      const isHidable = !isLastItem && shouldHideItems;\n      return (\n        <AppHeaderListItem\n          key={item.id}\n          ml={side === 'right' && index === 0 ? 'auto' : 0}\n          display={{\n            _: isHidable ? 'none' : 'flex',\n            xs: 'flex',\n          }}\n        >\n          {mapAppHeaderItemToElement({\n            action,\n            isStandalone: undefined,\n            isTeams: undefined,\n            item,\n            mobile: true,\n            redirectParam,\n          })}\n        </AppHeaderListItem>\n      );\n    });\n  };\n\n  const right = [\n    ...(!isEnterprise ? [searchButton] : []),\n    ...(notificationsBell && !isEnterprise ? [notificationsBell] : []),\n    ...items.right,\n  ];\n\n  const onItemType = (type: string | undefined) => {\n    if (\n      type &&\n      (type === 'catalog-dropdown' || type === 'resources-dropdown')\n    ) {\n      setAllowScroll(true);\n    } else {\n      setAllowScroll(false);\n    }\n  };\n\n  return (\n    <>\n      {!mobileMenuOpen && ( // need this bc AppBar has a hardcoded z-Index of 15\n        <HeaderHeightArea\n          display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n        >\n          <StyledAppBar as=\"nav\" aria-label=\"Main\">\n            <StyledNavBar center={!!isSimple}>\n              {mapItemsToElement(items.left, 'left')}\n              {mapItemsToElement(right, 'right', true)}\n              {!hideRightMenuButton && (\n                <AppHeaderListItem ml={right.length === 0 ? 'auto' : 0}>\n                  <IconButton\n                    aria-expanded={mobileMenuOpen}\n                    aria-label={\n                      navigationMenuFormattedLabel?.siteNavigation ||\n                      'Site navigation'\n                    }\n                    tip={'Site\\nnavigation'}\n                    tipProps={{\n                      alignment: 'bottom-center',\n                      placement: 'floating',\n                    }}\n                    data-testid=\"header-mobile-menu\"\n                    onClick={() => {\n                      openMobileMenu();\n                    }}\n                    icon={MenuIcon}\n                    variant=\"interface\"\n                    ref={openButtonRef}\n                  />\n                </AppHeaderListItem>\n              )}\n            </StyledNavBar>\n          </StyledAppBar>\n        </HeaderHeightArea>\n      )}\n      <StyledOverlay\n        clickOutsideCloses\n        escapeCloses\n        isOpen={mobileMenuOpen}\n        onRequestClose={() => setMobileMenuOpen(false)}\n        allowScroll={allowScroll}\n      >\n        <Background bg=\"beige\">\n          <HeaderHeightArea\n            display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n            as=\"nav\"\n            ariaLabel=\"Main\"\n            data-testid=\"header-mobile-menu-dropdown\"\n          >\n            <StyledAppBar>\n              <StyledNavBar>\n                {mapItemsToElement(items.left, 'left')}\n                <AppHeaderListItem ml=\"auto\">\n                  <IconButton\n                    aria-expanded={mobileMenuOpen}\n                    onClick={() => {\n                      setMobileMenuOpen(false);\n                    }}\n                    icon={MenuIcon}\n                    aria-label={\n                      navigationMenuFormattedLabel?.siteNavigation ||\n                      'Site navigation'\n                    }\n                    tip={\n                      navigationMenuFormattedLabel?.siteNavigation ||\n                      'Site navigation'\n                    }\n                    tipProps={{\n                      alignment: 'bottom-center',\n                      placement: 'floating',\n                    }}\n                    ref={closeButtonRef}\n                  />\n                </AppHeaderListItem>\n              </StyledNavBar>\n            </StyledAppBar>\n            <Box background={theme.colors.beige} height=\"auto\">\n              <AppHeaderMainMenuMobile\n                action={action}\n                items={items.mainMenu}\n                getItemType={onItemType}\n                isAnon={isAnon}\n              />\n            </Box>\n          </HeaderHeightArea>\n        </Background>\n      </StyledOverlay>\n      {!isEnterprise && (\n        <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n          {searchPane}\n        </Box>\n      )}\n      <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n        {notificationsView}\n      </Box>\n    </>\n  );\n};\n"]} */");
34
33
  const StyledNavBar = /*#__PURE__*/_styled("ul", {
35
34
  target: "e14c9jns0",
36
35
  label: "StyledNavBar"
@@ -48,7 +47,7 @@ const StyledNavBar = /*#__PURE__*/_styled("ul", {
48
47
  sm: 'flex-start'
49
48
  }
50
49
  }
51
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/AppHeaderMobile/index.tsx"],"names":[],"mappings":"AA4DqB","file":"../../src/AppHeaderMobile/index.tsx","sourcesContent":["import {\n  Box,\n  ButtonBaseElements,\n  IconButton,\n  Overlay,\n} from '@codecademy/gamut';\nimport { MenuIcon } from '@codecademy/gamut-icons';\nimport { Background, css, states, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useEffect, useRef, useState } from 'react';\nimport * as React from 'react';\n\nimport { AppHeaderSearch } from '..';\nimport { AppHeaderListItem } from '../AppHeader/AppHeaderElements/AppHeaderListItem';\nimport { SearchTrackingProvider } from '../AppHeader/Search/SearchTrackingProvider';\nimport { useHeaderSearch } from '../AppHeader/Search/useHeaderSearch';\nimport {\n  AppHeaderAction,\n  AppHeaderItem,\n  appHeaderMobileBreakpoint,\n  FormattedMobileAppHeaderItems,\n  StyledAppBar,\n} from '../AppHeader/shared';\nimport { mapAppHeaderItemToElement } from '../AppHeader/shared/utils';\nimport { AppHeaderMainMenuMobile } from '../AppHeaderMobile/AppHeaderMainMenuMobile';\nimport { NavigationMenuFormattedLabel } from '../GlobalHeader';\nimport { HeaderHeightArea } from '../HeaderHeightArea';\nimport { NotificationsContents } from '../Notifications/NotificationsContents';\nimport { AppHeaderNotificationSettings } from '../Notifications/types';\nimport { useHeaderNotifications } from '../Notifications/useHeaderNotifications';\n\nexport type AppHeaderMobileProps = AppHeaderAction & {\n  items: FormattedMobileAppHeaderItems;\n  notifications?: AppHeaderNotificationSettings;\n  redirectParam?: string;\n  search: AppHeaderSearch;\n  isAnon: boolean;\n  /**\n   * used to conditonally hide the default search icon and notification bell\n   */\n  isEnterprise?: boolean;\n  isSimple?: boolean;\n  hideRightMenuButton?: boolean;\n  navigationMenuFormattedLabel?: NavigationMenuFormattedLabel;\n};\n\nconst StyledOverlay = styled(Overlay)(\n  css({\n    display: { _: `block`, [appHeaderMobileBreakpoint]: `none` },\n    width: `100vw`,\n    height: `100vh`,\n    opacity: 1,\n    bg: `beige`,\n    position: `fixed`,\n    left: 0,\n    top: 0,\n    overflowX: `hidden`,\n  })\n);\n\nconst StyledNavBar = styled.ul<{ center?: boolean }>(\n  css({\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n    alignItems: 'center',\n  }),\n  states({\n    center: {\n      justifyContent: {\n        _: 'center',\n        sm: 'flex-start',\n      },\n    },\n  })\n);\n\nexport const AppHeaderMobile: React.FC<AppHeaderMobileProps> = ({\n  action,\n  items,\n  notifications,\n  search,\n  redirectParam,\n  isAnon,\n  isEnterprise,\n  isSimple,\n  hideRightMenuButton,\n  navigationMenuFormattedLabel,\n}) => {\n  const [mobileMenuOpen, setMobileMenuOpen] = useState<boolean | undefined>(\n    undefined\n  );\n  const [allowScroll, setAllowScroll] = useState<boolean>(false);\n  const openButtonRef = useRef<ButtonBaseElements>(null);\n  const closeButtonRef = useRef<ButtonBaseElements>(null);\n\n  const [notificationsBell, notificationsView] = useHeaderNotifications({\n    settings: notifications,\n    Renderer: NotificationsContents,\n  });\n\n  const [searchButton, searchPane] = useHeaderSearch({\n    ...search,\n  });\n\n  const openMobileMenu = () => {\n    setMobileMenuOpen(true);\n\n    if (closeButtonRef.current) {\n      closeButtonRef.current.focus();\n    }\n  };\n\n  useEffect(() => {\n    if (mobileMenuOpen === false && openButtonRef.current) {\n      setTimeout(() => {\n        if (openButtonRef.current) openButtonRef.current.focus();\n      }, 0);\n    }\n  }, [mobileMenuOpen]);\n\n  const mapItemsToElement = <T extends AppHeaderItem[]>(\n    items: T,\n    side: 'left' | 'right',\n    hideExtraItems?: boolean\n  ) => {\n    const shouldHideItems = hideExtraItems === true && items.length > 1;\n    return items.map((item, index) => {\n      const isLastItem = index + 1 === items.length;\n      const isHidable = !isLastItem && shouldHideItems;\n      return (\n        <AppHeaderListItem\n          key={item.id}\n          ml={side === 'right' && index === 0 ? 'auto' : 0}\n          display={{\n            _: isHidable ? 'none' : 'flex',\n            xs: 'flex',\n          }}\n        >\n          {mapAppHeaderItemToElement({\n            action,\n            isStandalone: undefined,\n            isTeams: undefined,\n            item,\n            mobile: true,\n            onKeyDown: undefined,\n            redirectParam,\n          })}\n        </AppHeaderListItem>\n      );\n    });\n  };\n\n  const right = [\n    ...(!isEnterprise ? [searchButton] : []),\n    ...(notificationsBell && !isEnterprise ? [notificationsBell] : []),\n    ...items.right,\n  ];\n\n  const onItemType = (type: string | undefined) => {\n    if (\n      type &&\n      (type === 'catalog-dropdown' || type === 'resources-dropdown')\n    ) {\n      setAllowScroll(true);\n    } else {\n      setAllowScroll(false);\n    }\n  };\n\n  return (\n    <>\n      <SearchTrackingProvider>\n        {!mobileMenuOpen && ( // need this bc AppBar has a hardcoded z-Index of 15\n          <HeaderHeightArea\n            display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n          >\n            <StyledAppBar as=\"nav\" aria-label=\"Main\">\n              <StyledNavBar center={!!isSimple}>\n                {mapItemsToElement(items.left, 'left')}\n                {mapItemsToElement(right, 'right', true)}\n                {!hideRightMenuButton && (\n                  <AppHeaderListItem ml={right.length === 0 ? 'auto' : 0}>\n                    <IconButton\n                      aria-expanded={mobileMenuOpen}\n                      aria-label={\n                        navigationMenuFormattedLabel?.siteNavigation ||\n                        'Site navigation'\n                      }\n                      tip={'Site\\nnavigation'}\n                      tipProps={{\n                        alignment: 'bottom-center',\n                        placement: 'floating',\n                      }}\n                      data-testid=\"header-mobile-menu\"\n                      onClick={() => {\n                        openMobileMenu();\n                      }}\n                      icon={MenuIcon}\n                      variant=\"interface\"\n                      ref={openButtonRef}\n                    />\n                  </AppHeaderListItem>\n                )}\n              </StyledNavBar>\n            </StyledAppBar>\n          </HeaderHeightArea>\n        )}\n        <StyledOverlay\n          clickOutsideCloses\n          escapeCloses\n          isOpen={mobileMenuOpen}\n          onRequestClose={() => setMobileMenuOpen(false)}\n          allowScroll={allowScroll}\n        >\n          <Background bg=\"beige\">\n            <HeaderHeightArea\n              display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n              as=\"nav\"\n              ariaLabel=\"Main\"\n              data-testid=\"header-mobile-menu-dropdown\"\n            >\n              <StyledAppBar>\n                <StyledNavBar>\n                  {mapItemsToElement(items.left, 'left')}\n                  <AppHeaderListItem ml=\"auto\">\n                    <IconButton\n                      aria-expanded={mobileMenuOpen}\n                      onClick={() => {\n                        setMobileMenuOpen(false);\n                      }}\n                      icon={MenuIcon}\n                      aria-label={\n                        navigationMenuFormattedLabel?.siteNavigation ||\n                        'Site navigation'\n                      }\n                      tip={\n                        navigationMenuFormattedLabel?.siteNavigation ||\n                        'Site navigation'\n                      }\n                      tipProps={{\n                        alignment: 'bottom-center',\n                        placement: 'floating',\n                      }}\n                      ref={closeButtonRef}\n                    />\n                  </AppHeaderListItem>\n                </StyledNavBar>\n              </StyledAppBar>\n              <Box background={theme.colors.beige} height=\"auto\">\n                <AppHeaderMainMenuMobile\n                  action={action}\n                  items={items.mainMenu}\n                  getItemType={onItemType}\n                  isAnon={isAnon}\n                />\n              </Box>\n            </HeaderHeightArea>\n          </Background>\n        </StyledOverlay>\n        {!isEnterprise && (\n          <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n            {searchPane}\n          </Box>\n        )}\n        <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n          {notificationsView}\n        </Box>\n      </SearchTrackingProvider>\n    </>\n  );\n};\n"]} */");
50
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/AppHeaderMobile/index.tsx"],"names":[],"mappings":"AA2DqB","file":"../../src/AppHeaderMobile/index.tsx","sourcesContent":["import {\n  Box,\n  ButtonBaseElements,\n  IconButton,\n  Overlay,\n} from '@codecademy/gamut';\nimport { MenuIcon } from '@codecademy/gamut-icons';\nimport { Background, css, states, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { useEffect, useRef, useState } from 'react';\nimport * as React from 'react';\n\nimport { AppHeaderSearch } from '..';\nimport { AppHeaderListItem } from '../AppHeader/AppHeaderElements/AppHeaderListItem';\nimport { useHeaderSearch } from '../AppHeader/Search/useHeaderSearch';\nimport {\n  AppHeaderAction,\n  AppHeaderItem,\n  appHeaderMobileBreakpoint,\n  FormattedMobileAppHeaderItems,\n  StyledAppBar,\n} from '../AppHeader/shared';\nimport { mapAppHeaderItemToElement } from '../AppHeader/shared/utils';\nimport { AppHeaderMainMenuMobile } from '../AppHeaderMobile/AppHeaderMainMenuMobile';\nimport { NavigationMenuFormattedLabel } from '../GlobalHeader';\nimport { HeaderHeightArea } from '../HeaderHeightArea';\nimport { NotificationsContents } from '../Notifications/NotificationsContents';\nimport { AppHeaderNotificationSettings } from '../Notifications/types';\nimport { useHeaderNotifications } from '../Notifications/useHeaderNotifications';\n\nexport type AppHeaderMobileProps = AppHeaderAction & {\n  items: FormattedMobileAppHeaderItems;\n  notifications?: AppHeaderNotificationSettings;\n  redirectParam?: string;\n  search: AppHeaderSearch;\n  isAnon: boolean;\n  /**\n   * used to conditonally hide the default search icon and notification bell\n   */\n  isEnterprise?: boolean;\n  isSimple?: boolean;\n  hideRightMenuButton?: boolean;\n  navigationMenuFormattedLabel?: NavigationMenuFormattedLabel;\n};\n\nconst StyledOverlay = styled(Overlay)(\n  css({\n    display: { _: `block`, [appHeaderMobileBreakpoint]: `none` },\n    width: `100vw`,\n    height: `100vh`,\n    opacity: 1,\n    bg: `beige`,\n    position: `fixed`,\n    left: 0,\n    top: 0,\n    overflowX: `hidden`,\n  })\n);\n\nconst StyledNavBar = styled.ul<{ center?: boolean }>(\n  css({\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n    alignItems: 'center',\n  }),\n  states({\n    center: {\n      justifyContent: {\n        _: 'center',\n        sm: 'flex-start',\n      },\n    },\n  })\n);\n\nexport const AppHeaderMobile: React.FC<AppHeaderMobileProps> = ({\n  action,\n  items,\n  notifications,\n  search,\n  redirectParam,\n  isAnon,\n  isEnterprise,\n  isSimple,\n  hideRightMenuButton,\n  navigationMenuFormattedLabel,\n}) => {\n  const [mobileMenuOpen, setMobileMenuOpen] = useState<boolean | undefined>(\n    undefined\n  );\n  const [allowScroll, setAllowScroll] = useState<boolean>(false);\n  const openButtonRef = useRef<ButtonBaseElements>(null);\n  const closeButtonRef = useRef<ButtonBaseElements>(null);\n\n  const [notificationsBell, notificationsView] = useHeaderNotifications({\n    settings: notifications,\n    Renderer: NotificationsContents,\n  });\n\n  const [searchButton, searchPane] = useHeaderSearch({\n    ...search,\n  });\n\n  const openMobileMenu = () => {\n    setMobileMenuOpen(true);\n\n    if (closeButtonRef.current) {\n      closeButtonRef.current.focus();\n    }\n  };\n\n  useEffect(() => {\n    if (mobileMenuOpen === false && openButtonRef.current) {\n      setTimeout(() => {\n        if (openButtonRef.current) openButtonRef.current.focus();\n      }, 0);\n    }\n  }, [mobileMenuOpen]);\n\n  const mapItemsToElement = <T extends AppHeaderItem[]>(\n    items: T,\n    side: 'left' | 'right',\n    hideExtraItems?: boolean\n  ) => {\n    const shouldHideItems = hideExtraItems === true && items.length > 1;\n    return items.map((item, index) => {\n      const isLastItem = index + 1 === items.length;\n      const isHidable = !isLastItem && shouldHideItems;\n      return (\n        <AppHeaderListItem\n          key={item.id}\n          ml={side === 'right' && index === 0 ? 'auto' : 0}\n          display={{\n            _: isHidable ? 'none' : 'flex',\n            xs: 'flex',\n          }}\n        >\n          {mapAppHeaderItemToElement({\n            action,\n            isStandalone: undefined,\n            isTeams: undefined,\n            item,\n            mobile: true,\n            redirectParam,\n          })}\n        </AppHeaderListItem>\n      );\n    });\n  };\n\n  const right = [\n    ...(!isEnterprise ? [searchButton] : []),\n    ...(notificationsBell && !isEnterprise ? [notificationsBell] : []),\n    ...items.right,\n  ];\n\n  const onItemType = (type: string | undefined) => {\n    if (\n      type &&\n      (type === 'catalog-dropdown' || type === 'resources-dropdown')\n    ) {\n      setAllowScroll(true);\n    } else {\n      setAllowScroll(false);\n    }\n  };\n\n  return (\n    <>\n      {!mobileMenuOpen && ( // need this bc AppBar has a hardcoded z-Index of 15\n        <HeaderHeightArea\n          display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n        >\n          <StyledAppBar as=\"nav\" aria-label=\"Main\">\n            <StyledNavBar center={!!isSimple}>\n              {mapItemsToElement(items.left, 'left')}\n              {mapItemsToElement(right, 'right', true)}\n              {!hideRightMenuButton && (\n                <AppHeaderListItem ml={right.length === 0 ? 'auto' : 0}>\n                  <IconButton\n                    aria-expanded={mobileMenuOpen}\n                    aria-label={\n                      navigationMenuFormattedLabel?.siteNavigation ||\n                      'Site navigation'\n                    }\n                    tip={'Site\\nnavigation'}\n                    tipProps={{\n                      alignment: 'bottom-center',\n                      placement: 'floating',\n                    }}\n                    data-testid=\"header-mobile-menu\"\n                    onClick={() => {\n                      openMobileMenu();\n                    }}\n                    icon={MenuIcon}\n                    variant=\"interface\"\n                    ref={openButtonRef}\n                  />\n                </AppHeaderListItem>\n              )}\n            </StyledNavBar>\n          </StyledAppBar>\n        </HeaderHeightArea>\n      )}\n      <StyledOverlay\n        clickOutsideCloses\n        escapeCloses\n        isOpen={mobileMenuOpen}\n        onRequestClose={() => setMobileMenuOpen(false)}\n        allowScroll={allowScroll}\n      >\n        <Background bg=\"beige\">\n          <HeaderHeightArea\n            display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}\n            as=\"nav\"\n            ariaLabel=\"Main\"\n            data-testid=\"header-mobile-menu-dropdown\"\n          >\n            <StyledAppBar>\n              <StyledNavBar>\n                {mapItemsToElement(items.left, 'left')}\n                <AppHeaderListItem ml=\"auto\">\n                  <IconButton\n                    aria-expanded={mobileMenuOpen}\n                    onClick={() => {\n                      setMobileMenuOpen(false);\n                    }}\n                    icon={MenuIcon}\n                    aria-label={\n                      navigationMenuFormattedLabel?.siteNavigation ||\n                      'Site navigation'\n                    }\n                    tip={\n                      navigationMenuFormattedLabel?.siteNavigation ||\n                      'Site navigation'\n                    }\n                    tipProps={{\n                      alignment: 'bottom-center',\n                      placement: 'floating',\n                    }}\n                    ref={closeButtonRef}\n                  />\n                </AppHeaderListItem>\n              </StyledNavBar>\n            </StyledAppBar>\n            <Box background={theme.colors.beige} height=\"auto\">\n              <AppHeaderMainMenuMobile\n                action={action}\n                items={items.mainMenu}\n                getItemType={onItemType}\n                isAnon={isAnon}\n              />\n            </Box>\n          </HeaderHeightArea>\n        </Background>\n      </StyledOverlay>\n      {!isEnterprise && (\n        <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n          {searchPane}\n        </Box>\n      )}\n      <Box display={{ _: `block`, [appHeaderMobileBreakpoint]: `none` }}>\n        {notificationsView}\n      </Box>\n    </>\n  );\n};\n"]} */");
52
51
  export const AppHeaderMobile = ({
53
52
  action,
54
53
  items,
@@ -102,7 +101,6 @@ export const AppHeaderMobile = ({
102
101
  isTeams: undefined,
103
102
  item,
104
103
  mobile: true,
105
- onKeyDown: undefined,
106
104
  redirectParam
107
105
  })
108
106
  }, item.id);
@@ -116,103 +114,101 @@ export const AppHeaderMobile = ({
116
114
  setAllowScroll(false);
117
115
  }
118
116
  };
119
- return /*#__PURE__*/_jsx(_Fragment, {
120
- children: /*#__PURE__*/_jsxs(SearchTrackingProvider, {
121
- children: [!mobileMenuOpen &&
122
- /*#__PURE__*/
123
- // need this bc AppBar has a hardcoded z-Index of 15
124
- _jsx(HeaderHeightArea, {
125
- display: {
126
- _: `block`,
127
- [appHeaderMobileBreakpoint]: `none`
128
- },
129
- children: /*#__PURE__*/_jsx(StyledAppBar, {
130
- as: "nav",
131
- "aria-label": "Main",
132
- children: /*#__PURE__*/_jsxs(StyledNavBar, {
133
- center: !!isSimple,
134
- children: [mapItemsToElement(items.left, 'left'), mapItemsToElement(right, 'right', true), !hideRightMenuButton && /*#__PURE__*/_jsx(AppHeaderListItem, {
135
- ml: right.length === 0 ? 'auto' : 0,
136
- children: /*#__PURE__*/_jsx(IconButton, {
137
- "aria-expanded": mobileMenuOpen,
138
- "aria-label": navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
139
- tip: 'Site\nnavigation',
140
- tipProps: {
141
- alignment: 'bottom-center',
142
- placement: 'floating'
143
- },
144
- "data-testid": "header-mobile-menu",
145
- onClick: () => {
146
- openMobileMenu();
147
- },
148
- icon: MenuIcon,
149
- variant: "interface",
150
- ref: openButtonRef
151
- })
152
- })]
153
- })
117
+ return /*#__PURE__*/_jsxs(_Fragment, {
118
+ children: [!mobileMenuOpen &&
119
+ /*#__PURE__*/
120
+ // need this bc AppBar has a hardcoded z-Index of 15
121
+ _jsx(HeaderHeightArea, {
122
+ display: {
123
+ _: `block`,
124
+ [appHeaderMobileBreakpoint]: `none`
125
+ },
126
+ children: /*#__PURE__*/_jsx(StyledAppBar, {
127
+ as: "nav",
128
+ "aria-label": "Main",
129
+ children: /*#__PURE__*/_jsxs(StyledNavBar, {
130
+ center: !!isSimple,
131
+ children: [mapItemsToElement(items.left, 'left'), mapItemsToElement(right, 'right', true), !hideRightMenuButton && /*#__PURE__*/_jsx(AppHeaderListItem, {
132
+ ml: right.length === 0 ? 'auto' : 0,
133
+ children: /*#__PURE__*/_jsx(IconButton, {
134
+ "aria-expanded": mobileMenuOpen,
135
+ "aria-label": navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
136
+ tip: 'Site\nnavigation',
137
+ tipProps: {
138
+ alignment: 'bottom-center',
139
+ placement: 'floating'
140
+ },
141
+ "data-testid": "header-mobile-menu",
142
+ onClick: () => {
143
+ openMobileMenu();
144
+ },
145
+ icon: MenuIcon,
146
+ variant: "interface",
147
+ ref: openButtonRef
148
+ })
149
+ })]
154
150
  })
155
- }), /*#__PURE__*/_jsx(StyledOverlay, {
156
- clickOutsideCloses: true,
157
- escapeCloses: true,
158
- isOpen: mobileMenuOpen,
159
- onRequestClose: () => setMobileMenuOpen(false),
160
- allowScroll: allowScroll,
161
- children: /*#__PURE__*/_jsx(Background, {
162
- bg: "beige",
163
- children: /*#__PURE__*/_jsxs(HeaderHeightArea, {
164
- display: {
165
- _: `block`,
166
- [appHeaderMobileBreakpoint]: `none`
167
- },
168
- as: "nav",
169
- ariaLabel: "Main",
170
- "data-testid": "header-mobile-menu-dropdown",
171
- children: [/*#__PURE__*/_jsx(StyledAppBar, {
172
- children: /*#__PURE__*/_jsxs(StyledNavBar, {
173
- children: [mapItemsToElement(items.left, 'left'), /*#__PURE__*/_jsx(AppHeaderListItem, {
174
- ml: "auto",
175
- children: /*#__PURE__*/_jsx(IconButton, {
176
- "aria-expanded": mobileMenuOpen,
177
- onClick: () => {
178
- setMobileMenuOpen(false);
179
- },
180
- icon: MenuIcon,
181
- "aria-label": navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
182
- tip: navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
183
- tipProps: {
184
- alignment: 'bottom-center',
185
- placement: 'floating'
186
- },
187
- ref: closeButtonRef
188
- })
189
- })]
190
- })
191
- }), /*#__PURE__*/_jsx(Box, {
192
- background: theme.colors.beige,
193
- height: "auto",
194
- children: /*#__PURE__*/_jsx(AppHeaderMainMenuMobile, {
195
- action: action,
196
- items: items.mainMenu,
197
- getItemType: onItemType,
198
- isAnon: isAnon
199
- })
200
- })]
201
- })
151
+ })
152
+ }), /*#__PURE__*/_jsx(StyledOverlay, {
153
+ clickOutsideCloses: true,
154
+ escapeCloses: true,
155
+ isOpen: mobileMenuOpen,
156
+ onRequestClose: () => setMobileMenuOpen(false),
157
+ allowScroll: allowScroll,
158
+ children: /*#__PURE__*/_jsx(Background, {
159
+ bg: "beige",
160
+ children: /*#__PURE__*/_jsxs(HeaderHeightArea, {
161
+ display: {
162
+ _: `block`,
163
+ [appHeaderMobileBreakpoint]: `none`
164
+ },
165
+ as: "nav",
166
+ ariaLabel: "Main",
167
+ "data-testid": "header-mobile-menu-dropdown",
168
+ children: [/*#__PURE__*/_jsx(StyledAppBar, {
169
+ children: /*#__PURE__*/_jsxs(StyledNavBar, {
170
+ children: [mapItemsToElement(items.left, 'left'), /*#__PURE__*/_jsx(AppHeaderListItem, {
171
+ ml: "auto",
172
+ children: /*#__PURE__*/_jsx(IconButton, {
173
+ "aria-expanded": mobileMenuOpen,
174
+ onClick: () => {
175
+ setMobileMenuOpen(false);
176
+ },
177
+ icon: MenuIcon,
178
+ "aria-label": navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
179
+ tip: navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
180
+ tipProps: {
181
+ alignment: 'bottom-center',
182
+ placement: 'floating'
183
+ },
184
+ ref: closeButtonRef
185
+ })
186
+ })]
187
+ })
188
+ }), /*#__PURE__*/_jsx(Box, {
189
+ background: theme.colors.beige,
190
+ height: "auto",
191
+ children: /*#__PURE__*/_jsx(AppHeaderMainMenuMobile, {
192
+ action: action,
193
+ items: items.mainMenu,
194
+ getItemType: onItemType,
195
+ isAnon: isAnon
196
+ })
197
+ })]
202
198
  })
203
- }), !isEnterprise && /*#__PURE__*/_jsx(Box, {
204
- display: {
205
- _: `block`,
206
- [appHeaderMobileBreakpoint]: `none`
207
- },
208
- children: searchPane
209
- }), /*#__PURE__*/_jsx(Box, {
210
- display: {
211
- _: `block`,
212
- [appHeaderMobileBreakpoint]: `none`
213
- },
214
- children: notificationsView
215
- })]
216
- })
199
+ })
200
+ }), !isEnterprise && /*#__PURE__*/_jsx(Box, {
201
+ display: {
202
+ _: `block`,
203
+ [appHeaderMobileBreakpoint]: `none`
204
+ },
205
+ children: searchPane
206
+ }), /*#__PURE__*/_jsx(Box, {
207
+ display: {
208
+ _: `block`,
209
+ [appHeaderMobileBreakpoint]: `none`
210
+ },
211
+ children: notificationsView
212
+ })]
217
213
  });
218
214
  };
@@ -2,7 +2,7 @@ import _styled from "@emotion/styled/base";
2
2
  import { FlexBox, Text } from '@codecademy/gamut';
3
3
  import { theme } from '@codecademy/gamut-styles';
4
4
  import React from 'react';
5
- import { getCurrencySymbol } from './types';
5
+ import { getCurrencyPrefix, getCurrencySymbol } from './types';
6
6
  import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
7
7
  const Amount = /*#__PURE__*/_styled(Text, {
8
8
  target: "elw3jo70",
@@ -11,7 +11,7 @@ const Amount = /*#__PURE__*/_styled(Text, {
11
11
  isMultiple
12
12
  }) => isMultiple ? '44px' : '64px', ";line-height:", ({
13
13
  isMultiple
14
- }) => isMultiple ? '64px' : '80px', ";", theme.breakpoints.xl, "{font-size:64px;line-height:80px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9QbGFuQ2FyZC9QcmljaW5nQW1vdW50LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFPb0QiLCJmaWxlIjoiLi4vLi4vc3JjL1BsYW5DYXJkL1ByaWNpbmdBbW91bnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRmxleEJveCwgVGV4dCB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0JztcbmltcG9ydCB7IHRoZW1lIH0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQtc3R5bGVzJztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEN1cnJlbmN5LCBnZXRDdXJyZW5jeVN5bWJvbCwgUGxhblR5cGUgfSBmcm9tICcuL3R5cGVzJztcblxuY29uc3QgQW1vdW50ID0gc3R5bGVkKFRleHQpPHsgaXNNdWx0aXBsZTogYm9vbGVhbiB9PmBcbiAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gIGNvbG9yOiAke3RoZW1lLmNvbG9ycy50ZXh0fTtcbiAgZm9udC1zaXplOiAkeyh7IGlzTXVsdGlwbGUgfSkgPT4gKGlzTXVsdGlwbGUgPyAnNDRweCcgOiAnNjRweCcpfTtcbiAgbGluZS1oZWlnaHQ6ICR7KHsgaXNNdWx0aXBsZSB9KSA9PiAoaXNNdWx0aXBsZSA/ICc2NHB4JyA6ICc4MHB4Jyl9O1xuICAke3RoZW1lLmJyZWFrcG9pbnRzLnhsfSB7XG4gICAgZm9udC1zaXplOiA2NHB4O1xuICAgIGxpbmUtaGVpZ2h0OiA4MHB4O1xuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJpY2luZ0Ftb3VudFByb3BzIHtcbiAgcHJvZHVjdDogUGxhblR5cGU7XG4gIHByaWNlOiBzdHJpbmc7XG4gIG1vbnRobHlQcmljZT86IHN0cmluZztcbiAgY3VycmVuY3k6IEN1cnJlbmN5O1xuICB0ZXJtTW9udGhzPzogbnVtYmVyO1xuICBjb21wYWN0PzogYm9vbGVhbjtcbiAgaXNNdWx0aXBsZTogYm9vbGVhbjtcbiAgaW5VcHNlbGxNb2RhbD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBQcmljaW5nQW1vdW50OiBSZWFjdC5GQzxQcmljaW5nQW1vdW50UHJvcHM+ID0gKHtcbiAgY3VycmVuY3ksXG4gIG1vbnRobHlQcmljZSxcbiAgcHJvZHVjdCxcbiAgcHJpY2UsXG4gIHRlcm1Nb250aHMsXG4gIGNvbXBhY3QgPSBmYWxzZSxcbiAgaXNNdWx0aXBsZSxcbiAgaW5VcHNlbGxNb2RhbCA9IGZhbHNlLFxufSkgPT4ge1xuICBjb25zdCBjeWNsZSA9IHRlcm1Nb250aHMgPT09IDEgPyAnbW9udGhseScgOiAnYW5udWFsbHknO1xuICBjb25zdCBbZG9sbGFycywgY2VudHNdID0gcHJpY2Uuc3BsaXQoJy4nKTtcbiAgbGV0IGFyaWFMYWJlbCA9IGAke2dldEN1cnJlbmN5U3ltYm9sKGN1cnJlbmN5KX0ke2RvbGxhcnN9JHtcbiAgICBjZW50cyA/IGAuJHtjZW50c31gIDogJydcbiAgfWA7XG4gIGlmIChwcm9kdWN0ICE9PSAnYmFzaWMnKSB7XG4gICAgYXJpYUxhYmVsICs9IGAgYSBtb250aCwgYmlsbGVkICR7Y3ljbGV9YDtcbiAgICBpZiAoaW5VcHNlbGxNb2RhbCkgYXJpYUxhYmVsICs9ICcsIG1vbnRobHkgcHJpY2luZyBhbHNvIGF2YWlsYWJsZSc7XG4gICAgaWYgKG1vbnRobHlQcmljZSkge1xuICAgICAgYXJpYUxhYmVsICs9IGAgb3IgJHtnZXRDdXJyZW5jeVN5bWJvbChcbiAgICAgICAgY3VycmVuY3lcbiAgICAgICl9JHttb250aGx5UHJpY2V9IGJpbGxlZCBtb250aGx5YDtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8RmxleEJveFxuICAgICAgICBkYXRhLXRlc3RpZD1cInByaWNpbmctYW1vdW50XCJcbiAgICAgICAgbGluZUhlaWdodD1cInRpdGxlXCJcbiAgICAgICAgbWI9e2NvbXBhY3QgPyAwIDogOH1cbiAgICAgICAgY29sdW1uXG4gICAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgPlxuICAgICAgICA8VGV4dCBzY3JlZW5yZWFkZXIgZGF0YS10ZXN0aWQ9XCJwcmljaW5nLWFtb3VudC1sYWJlbFwiPlxuICAgICAgICAgIHthcmlhTGFiZWx9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAgey8qIGZvciBzY3JlZW4gcmVhZGVyIG9wdGltaXphdGlvbiAqL31cbiAgICAgICAgPEZsZXhCb3hcbiAgICAgICAgICBhcmlhLWhpZGRlblxuICAgICAgICAgIHJvd1xuICAgICAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgICAgIGhlaWdodD17aXNNdWx0aXBsZSA/IHsgXzogNTAsIHhsOiAndW5zZXQnIH0gOiAndW5zZXQnfVxuICAgICAgICA+XG4gICAgICAgICAgPEZsZXhCb3ggbXI9ezR9IGFsaWduSXRlbXM9XCJzdGFydFwiPlxuICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgY29sb3I9XCJ0ZXh0XCJcbiAgICAgICAgICAgICAgbXQ9e3sgXzogMTIsIHhsOiAxNiB9fVxuICAgICAgICAgICAgICBmb250V2VpZ2h0PVwiYm9sZFwiXG4gICAgICAgICAgICAgIGZvbnRTaXplPXszNH1cbiAgICAgICAgICAgID57YCR7Z2V0Q3VycmVuY3lTeW1ib2woY3VycmVuY3kpfWB9PC9UZXh0PlxuICAgICAgICAgICAgPEFtb3VudCBpc011bHRpcGxlPXtpc011bHRpcGxlfT57YCR7ZG9sbGFyc31gfTwvQW1vdW50PlxuICAgICAgICAgICAge3Byb2R1Y3QgPT09ICdiYXNpYycgPyBudWxsIDogKFxuICAgICAgICAgICAgICA8RmxleEJveCBjb2x1bW4gYWxpZ25JdGVtcz1cImZsZXgtZW5kXCIgbWw9ezR9PlxuICAgICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgICBjb2xvcj1cInRleHRcIlxuICAgICAgICAgICAgICAgICAgZm9udFdlaWdodD1cImJvbGRcIlxuICAgICAgICAgICAgICAgICAgZm9udFNpemU9ezIyfVxuICAgICAgICAgICAgICAgICAgbXQ9e3sgXzogMTIsIHhsOiAxNiB9fVxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIHtjZW50cyA/IGAuJHtjZW50c31gIDogPHNwYW4+Jm5ic3A7PC9zcGFuPn1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPFRleHQgbXQ9e3sgXzogMCwgeGw6IDQgfX0gY29sb3I9XCJ0ZXh0LXNlY29uZGFyeVwiPlxuICAgICAgICAgICAgICAgICAgL21vXG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L0ZsZXhCb3g+XG4gICAgICAgICAgICApfVxuICAgICAgICAgIDwvRmxleEJveD5cbiAgICAgICAgPC9GbGV4Qm94PlxuICAgICAgICA8VGV4dCBjb2xvcj1cInRleHQtc2Vjb25kYXJ5XCIgY2VudGVyIG10PXtjb21wYWN0ID8gMCA6IHsgXzogOCwgeGw6IDAgfX0+XG4gICAgICAgICAge3Byb2R1Y3QgPT09ICdiYXNpYycgPyAoXG4gICAgICAgICAgICA8VGV4dCBtYj17MjR9PkFsd2F5cyBmcmVlPC9UZXh0PlxuICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICA8PlxuICAgICAgICAgICAgICA8VGV4dCBtYj17Y29tcGFjdCA/IDAgOiA4fSBhcmlhLWhpZGRlbj5cbiAgICAgICAgICAgICAgICB7YEJpbGxlZCAke2N5Y2xlfSR7aW5VcHNlbGxNb2RhbCA/ICcqJyA6ICcnfWB9XG4gICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPGJyIC8+XG4gICAgICAgICAgICAgIHttb250aGx5UHJpY2UgJiYgKFxuICAgICAgICAgICAgICAgIDxUZXh0IGFyaWEtaGlkZGVuPlxuICAgICAgICAgICAgICAgICAge2BvciAke2dldEN1cnJlbmN5U3ltYm9sKFxuICAgICAgICAgICAgICAgICAgICBjdXJyZW5jeVxuICAgICAgICAgICAgICAgICAgKX0ke21vbnRobHlQcmljZX0gYmlsbGVkIG1vbnRobHlgfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgIDwvPlxuICAgICAgICAgICl9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvRmxleEJveD5cbiAgICA8Lz5cbiAgKTtcbn07XG4iXX0= */"));
14
+ }) => isMultiple ? '64px' : '80px', ";", theme.breakpoints.xl, "{font-size:64px;line-height:80px;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9QbGFuQ2FyZC9QcmljaW5nQW1vdW50LnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFZb0QiLCJmaWxlIjoiLi4vLi4vc3JjL1BsYW5DYXJkL1ByaWNpbmdBbW91bnQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRmxleEJveCwgVGV4dCB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0JztcbmltcG9ydCB7IHRoZW1lIH0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQtc3R5bGVzJztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7XG4gIEN1cnJlbmN5LFxuICBnZXRDdXJyZW5jeVByZWZpeCxcbiAgZ2V0Q3VycmVuY3lTeW1ib2wsXG4gIFBsYW5UeXBlLFxufSBmcm9tICcuL3R5cGVzJztcblxuY29uc3QgQW1vdW50ID0gc3R5bGVkKFRleHQpPHsgaXNNdWx0aXBsZTogYm9vbGVhbiB9PmBcbiAgZm9udC13ZWlnaHQ6IGJvbGQ7XG4gIGNvbG9yOiAke3RoZW1lLmNvbG9ycy50ZXh0fTtcbiAgZm9udC1zaXplOiAkeyh7IGlzTXVsdGlwbGUgfSkgPT4gKGlzTXVsdGlwbGUgPyAnNDRweCcgOiAnNjRweCcpfTtcbiAgbGluZS1oZWlnaHQ6ICR7KHsgaXNNdWx0aXBsZSB9KSA9PiAoaXNNdWx0aXBsZSA/ICc2NHB4JyA6ICc4MHB4Jyl9O1xuICAke3RoZW1lLmJyZWFrcG9pbnRzLnhsfSB7XG4gICAgZm9udC1zaXplOiA2NHB4O1xuICAgIGxpbmUtaGVpZ2h0OiA4MHB4O1xuICB9XG5gO1xuXG5pbnRlcmZhY2UgUHJpY2luZ0Ftb3VudFByb3BzIHtcbiAgcHJvZHVjdDogUGxhblR5cGU7XG4gIHByaWNlOiBzdHJpbmc7XG4gIG1vbnRobHlQcmljZT86IHN0cmluZztcbiAgY3VycmVuY3k6IEN1cnJlbmN5O1xuICB0ZXJtTW9udGhzPzogbnVtYmVyO1xuICBjb21wYWN0PzogYm9vbGVhbjtcbiAgaXNNdWx0aXBsZTogYm9vbGVhbjtcbiAgaW5VcHNlbGxNb2RhbD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCBjb25zdCBQcmljaW5nQW1vdW50OiBSZWFjdC5GQzxQcmljaW5nQW1vdW50UHJvcHM+ID0gKHtcbiAgY3VycmVuY3ksXG4gIG1vbnRobHlQcmljZSxcbiAgcHJvZHVjdCxcbiAgcHJpY2UsXG4gIHRlcm1Nb250aHMsXG4gIGNvbXBhY3QgPSBmYWxzZSxcbiAgaXNNdWx0aXBsZSxcbiAgaW5VcHNlbGxNb2RhbCA9IGZhbHNlLFxufSkgPT4ge1xuICBjb25zdCBjeWNsZSA9IHRlcm1Nb250aHMgPT09IDEgPyAnbW9udGhseScgOiAnYW5udWFsbHknO1xuICBjb25zdCBbZG9sbGFycywgY2VudHNdID0gcHJpY2Uuc3BsaXQoJy4nKTtcbiAgbGV0IGFyaWFMYWJlbCA9IGAke2dldEN1cnJlbmN5UHJlZml4KGN1cnJlbmN5KX0ke2dldEN1cnJlbmN5U3ltYm9sKFxuICAgIGN1cnJlbmN5XG4gICl9JHtkb2xsYXJzfSR7Y2VudHMgPyBgLiR7Y2VudHN9YCA6ICcnfWA7XG4gIGlmIChwcm9kdWN0ICE9PSAnYmFzaWMnKSB7XG4gICAgYXJpYUxhYmVsICs9IGAgYSBtb250aCwgYmlsbGVkICR7Y3ljbGV9YDtcbiAgICBpZiAoaW5VcHNlbGxNb2RhbCkgYXJpYUxhYmVsICs9ICcsIG1vbnRobHkgcHJpY2luZyBhbHNvIGF2YWlsYWJsZSc7XG4gICAgaWYgKG1vbnRobHlQcmljZSkge1xuICAgICAgYXJpYUxhYmVsICs9IGAgb3IgJHtnZXRDdXJyZW5jeVByZWZpeChjdXJyZW5jeSl9JHtnZXRDdXJyZW5jeVN5bWJvbChcbiAgICAgICAgY3VycmVuY3lcbiAgICAgICl9JHttb250aGx5UHJpY2V9IGJpbGxlZCBtb250aGx5YDtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gKFxuICAgIDw+XG4gICAgICA8RmxleEJveFxuICAgICAgICBkYXRhLXRlc3RpZD1cInByaWNpbmctYW1vdW50XCJcbiAgICAgICAgbGluZUhlaWdodD1cInRpdGxlXCJcbiAgICAgICAgbWI9e2NvbXBhY3QgPyAwIDogOH1cbiAgICAgICAgY29sdW1uXG4gICAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgPlxuICAgICAgICA8VGV4dCBzY3JlZW5yZWFkZXIgZGF0YS10ZXN0aWQ9XCJwcmljaW5nLWFtb3VudC1sYWJlbFwiPlxuICAgICAgICAgIHthcmlhTGFiZWx9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgICAgey8qIGZvciBzY3JlZW4gcmVhZGVyIG9wdGltaXphdGlvbiAqL31cbiAgICAgICAgPEZsZXhCb3hcbiAgICAgICAgICBhcmlhLWhpZGRlblxuICAgICAgICAgIHJvd1xuICAgICAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgICAgIGhlaWdodD17aXNNdWx0aXBsZSA/IHsgXzogNTAsIHhsOiAndW5zZXQnIH0gOiAndW5zZXQnfVxuICAgICAgICA+XG4gICAgICAgICAgPEZsZXhCb3ggbXI9ezR9IGFsaWduSXRlbXM9XCJzdGFydFwiPlxuICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgY29sb3I9XCJ0ZXh0XCJcbiAgICAgICAgICAgICAgbXQ9e3sgXzogMTIsIHhsOiAxNiB9fVxuICAgICAgICAgICAgICBmb250V2VpZ2h0PVwiYm9sZFwiXG4gICAgICAgICAgICAgIGZvbnRTaXplPXszNH1cbiAgICAgICAgICAgID57YCR7Z2V0Q3VycmVuY3lQcmVmaXgoY3VycmVuY3kpfSR7Z2V0Q3VycmVuY3lTeW1ib2woXG4gICAgICAgICAgICAgIGN1cnJlbmN5XG4gICAgICAgICAgICApfWB9PC9UZXh0PlxuICAgICAgICAgICAgPEFtb3VudCBpc011bHRpcGxlPXtpc011bHRpcGxlfT57YCR7ZG9sbGFyc31gfTwvQW1vdW50PlxuICAgICAgICAgICAge3Byb2R1Y3QgPT09ICdiYXNpYycgPyBudWxsIDogKFxuICAgICAgICAgICAgICA8RmxleEJveCBjb2x1bW4gYWxpZ25JdGVtcz1cImZsZXgtZW5kXCIgbWw9ezR9PlxuICAgICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgICBjb2xvcj1cInRleHRcIlxuICAgICAgICAgICAgICAgICAgZm9udFdlaWdodD1cImJvbGRcIlxuICAgICAgICAgICAgICAgICAgZm9udFNpemU9ezIyfVxuICAgICAgICAgICAgICAgICAgbXQ9e3sgXzogMTIsIHhsOiAxNiB9fVxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIHtjZW50cyA/IGAuJHtjZW50c31gIDogPHNwYW4+Jm5ic3A7PC9zcGFuPn1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPFRleHQgbXQ9e3sgXzogMCwgeGw6IDQgfX0gY29sb3I9XCJ0ZXh0LXNlY29uZGFyeVwiPlxuICAgICAgICAgICAgICAgICAgL21vXG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L0ZsZXhCb3g+XG4gICAgICAgICAgICApfVxuICAgICAgICAgIDwvRmxleEJveD5cbiAgICAgICAgPC9GbGV4Qm94PlxuICAgICAgICA8VGV4dCBjb2xvcj1cInRleHQtc2Vjb25kYXJ5XCIgY2VudGVyIG10PXtjb21wYWN0ID8gMCA6IHsgXzogOCwgeGw6IDAgfX0+XG4gICAgICAgICAge3Byb2R1Y3QgPT09ICdiYXNpYycgPyAoXG4gICAgICAgICAgICA8VGV4dCBtYj17MjR9PkFsd2F5cyBmcmVlPC9UZXh0PlxuICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICA8PlxuICAgICAgICAgICAgICA8VGV4dCBtYj17Y29tcGFjdCA/IDAgOiA4fSBhcmlhLWhpZGRlbj5cbiAgICAgICAgICAgICAgICB7YEJpbGxlZCAke2N5Y2xlfSR7aW5VcHNlbGxNb2RhbCA/ICcqJyA6ICcnfWB9XG4gICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPGJyIC8+XG4gICAgICAgICAgICAgIHttb250aGx5UHJpY2UgJiYgKFxuICAgICAgICAgICAgICAgIDxUZXh0IGFyaWEtaGlkZGVuPlxuICAgICAgICAgICAgICAgICAge2BvciAke2dldEN1cnJlbmN5UHJlZml4KGN1cnJlbmN5KX0ke2dldEN1cnJlbmN5U3ltYm9sKFxuICAgICAgICAgICAgICAgICAgICBjdXJyZW5jeVxuICAgICAgICAgICAgICAgICAgKX0ke21vbnRobHlQcmljZX0gYmlsbGVkIG1vbnRobHlgfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgIDwvPlxuICAgICAgICAgICl9XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvRmxleEJveD5cbiAgICA8Lz5cbiAgKTtcbn07XG4iXX0= */"));
15
15
  export const PricingAmount = ({
16
16
  currency,
17
17
  monthlyPrice,
@@ -24,12 +24,12 @@ export const PricingAmount = ({
24
24
  }) => {
25
25
  const cycle = termMonths === 1 ? 'monthly' : 'annually';
26
26
  const [dollars, cents] = price.split('.');
27
- let ariaLabel = `${getCurrencySymbol(currency)}${dollars}${cents ? `.${cents}` : ''}`;
27
+ let ariaLabel = `${getCurrencyPrefix(currency)}${getCurrencySymbol(currency)}${dollars}${cents ? `.${cents}` : ''}`;
28
28
  if (product !== 'basic') {
29
29
  ariaLabel += ` a month, billed ${cycle}`;
30
30
  if (inUpsellModal) ariaLabel += ', monthly pricing also available';
31
31
  if (monthlyPrice) {
32
- ariaLabel += ` or ${getCurrencySymbol(currency)}${monthlyPrice} billed monthly`;
32
+ ariaLabel += ` or ${getCurrencyPrefix(currency)}${getCurrencySymbol(currency)}${monthlyPrice} billed monthly`;
33
33
  }
34
34
  }
35
35
  return /*#__PURE__*/_jsx(_Fragment, {
@@ -62,7 +62,7 @@ export const PricingAmount = ({
62
62
  },
63
63
  fontWeight: "bold",
64
64
  fontSize: 34,
65
- children: `${getCurrencySymbol(currency)}`
65
+ children: `${getCurrencyPrefix(currency)}${getCurrencySymbol(currency)}`
66
66
  }), /*#__PURE__*/_jsx(Amount, {
67
67
  isMultiple: isMultiple,
68
68
  children: `${dollars}`
@@ -108,7 +108,7 @@ export const PricingAmount = ({
108
108
  children: `Billed ${cycle}${inUpsellModal ? '*' : ''}`
109
109
  }), /*#__PURE__*/_jsx("br", {}), monthlyPrice && /*#__PURE__*/_jsx(Text, {
110
110
  "aria-hidden": true,
111
- children: `or ${getCurrencySymbol(currency)}${monthlyPrice} billed monthly`
111
+ children: `or ${getCurrencyPrefix(currency)}${getCurrencySymbol(currency)}${monthlyPrice} billed monthly`
112
112
  })]
113
113
  })
114
114
  })]
@@ -37,3 +37,4 @@ export declare enum Currency {
37
37
  RON = "RON"
38
38
  }
39
39
  export declare const getCurrencySymbol: (currency: Currency) => CurrencySymbol;
40
+ export declare const getCurrencyPrefix: (currency: Currency) => "" | "AU" | "CA";
@@ -59,4 +59,16 @@ const currencySymbols = {
59
59
  [Currency.MXN]: CurrencySymbol.PESO,
60
60
  [Currency.RON]: CurrencySymbol.LEU
61
61
  };
62
- export const getCurrencySymbol = currency => currencySymbols[currency];
62
+ export const getCurrencySymbol = currency => currencySymbols[currency];
63
+
64
+ // for currencies that have common symbols eg: Dollar - US=$, AU=AU$, CA=CA$
65
+ export const getCurrencyPrefix = currency => {
66
+ switch (currency) {
67
+ case Currency.AUD:
68
+ return 'AU';
69
+ case Currency.CAD:
70
+ return 'CA';
71
+ default:
72
+ return '';
73
+ }
74
+ };
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@codecademy/brand",
3
3
  "description": "Brand component library for Codecademy",
4
- "version": "3.26.0-alpha.fa0b72ae29.0",
4
+ "version": "3.26.0",
5
5
  "author": "Codecademy Engineering <dev@codecademy.com>",
6
6
  "dependencies": {
7
7
  "@emotion/is-prop-valid": "^1.2.1",
@@ -27,7 +27,12 @@
27
27
  "main": "./dist/index.js",
28
28
  "peerDependencies": {
29
29
  "@codecademy/gamut": "*",
30
+ "@codecademy/gamut-icons": "*",
31
+ "@codecademy/gamut-illustrations": "*",
32
+ "@codecademy/gamut-patterns": "*",
33
+ "@codecademy/gamut-styles": "*",
30
34
  "@codecademy/tracking": "1.2.0",
35
+ "@codecademy/variance": "*",
31
36
  "@emotion/react": "^11.4.0",
32
37
  "@emotion/styled": "^11.3.0",
33
38
  "react": "^17.0.2 || ^18.2.0"
@@ -1,3 +0,0 @@
1
- import React from 'react';
2
- import { AppHeaderDropdownLinkProps } from './types';
3
- export declare const AppHeaderDropdownLink: React.FC<AppHeaderDropdownLinkProps>;
@@ -1,25 +0,0 @@
1
- import React, { useEffect, useRef } from 'react';
2
- import { useAppHeaderDropdownContext } from '../../AppHeaderDropdownProvider';
3
- import { StyledAppHeaderLink } from './StyledAppHeaderLink';
4
- import { jsx as _jsx } from "react/jsx-runtime";
5
- export const AppHeaderDropdownLink = ({
6
- isFirstElem,
7
- role,
8
- ...props
9
- }) => {
10
- const {
11
- setFirstItemRef
12
- } = useAppHeaderDropdownContext();
13
- const ref = useRef(null);
14
- useEffect(() => {
15
- if (isFirstElem && setFirstItemRef && ref?.current) {
16
- setFirstItemRef(ref.current);
17
- }
18
- }, [setFirstItemRef, ref, isFirstElem]);
19
- return /*#__PURE__*/_jsx(StyledAppHeaderLink, {
20
- isInDropdown: true,
21
- role: role,
22
- ref: ref,
23
- ...props
24
- });
25
- };
@@ -1,3 +0,0 @@
1
- import React from 'react';
2
- import { AppHeaderDropdownLinkProps } from './types';
3
- export declare const AppHeaderMenuItem: React.FC<AppHeaderDropdownLinkProps>;
@@ -1,26 +0,0 @@
1
- import React, { useEffect, useRef } from 'react';
2
- import { useAppHeaderMenuContext } from '../../AppHeaderMenuProvider';
3
- import { StyledAppHeaderLink } from './StyledAppHeaderLink';
4
- import { jsx as _jsx } from "react/jsx-runtime";
5
- export const AppHeaderMenuItem = ({
6
- role,
7
- ...props
8
- }) => {
9
- const {
10
- menuItems,
11
- setMenuItems
12
- } = useAppHeaderMenuContext();
13
- const ref = useRef(null);
14
- useEffect(() => {
15
- if (setMenuItems && ref?.current) {
16
- setMenuItems(menuItems.add(ref.current));
17
- }
18
- // eslint-disable-next-line react-hooks/exhaustive-deps
19
- }, []);
20
- return /*#__PURE__*/_jsx(StyledAppHeaderLink, {
21
- isInDropdown: true,
22
- role: role,
23
- ref: ref,
24
- ...props
25
- });
26
- };
@@ -1,36 +0,0 @@
1
- import _styled from "@emotion/styled/base";
2
- import { css, theme } from '@codecademy/gamut-styles';
3
- import { AppHeaderLink } from '../../AppHeaderLink';
4
- export const StyledAppHeaderLink = /*#__PURE__*/_styled(AppHeaderLink, {
5
- target: "e1oc8aqt0",
6
- label: "StyledAppHeaderLink"
7
- })(css({
8
- display: 'flex',
9
- position: 'relative',
10
- '@media (max-width: 1199px)': {
11
- pl: 0
12
- },
13
- '&::before': {
14
- content: '""',
15
- position: 'absolute',
16
- width: 'calc(100% - 24px)',
17
- left: {
18
- _: 0,
19
- lg: 12
20
- },
21
- borderRadius: 'lg',
22
- transition: 'background-color 0.15s ease',
23
- pointerEvents: 'none',
24
- zIndex: -1
25
- },
26
- '&:hover::before, &:focus-visible::before': {
27
- backgroundColor: theme.colors['navy-100']
28
- },
29
- '&:hover, &:focus-visible, &:active': {
30
- color: 'hyper-400',
31
- borderRadius: 'lg'
32
- },
33
- '&:active::before': {
34
- backgroundColor: theme.colors['navy-200']
35
- }
36
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvQXBwSGVhZGVyRWxlbWVudHMvQXBwSGVhZGVyRHJvcGRvd25MaW5rcy9lbGVtZW50cy9TdHlsZWRBcHBIZWFkZXJMaW5rLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFLbUMiLCJmaWxlIjoiLi4vLi4vLi4vLi4vLi4vc3JjL0FwcEhlYWRlci9BcHBIZWFkZXJFbGVtZW50cy9BcHBIZWFkZXJEcm9wZG93bkxpbmtzL2VsZW1lbnRzL1N0eWxlZEFwcEhlYWRlckxpbmsudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgY3NzLCB0aGVtZSB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5cbmltcG9ydCB7IEFwcEhlYWRlckxpbmsgfSBmcm9tICcuLi8uLi9BcHBIZWFkZXJMaW5rJztcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEhlYWRlckxpbmsgPSBzdHlsZWQoQXBwSGVhZGVyTGluaykoXG4gIGNzcyh7XG4gICAgZGlzcGxheTogJ2ZsZXgnLFxuICAgIHBvc2l0aW9uOiAncmVsYXRpdmUnLFxuICAgICdAbWVkaWEgKG1heC13aWR0aDogMTE5OXB4KSc6IHtcbiAgICAgIHBsOiAwLFxuICAgIH0sXG4gICAgJyY6OmJlZm9yZSc6IHtcbiAgICAgIGNvbnRlbnQ6ICdcIlwiJyxcbiAgICAgIHBvc2l0aW9uOiAnYWJzb2x1dGUnLFxuICAgICAgd2lkdGg6ICdjYWxjKDEwMCUgLSAyNHB4KScsXG4gICAgICBsZWZ0OiB7IF86IDAsIGxnOiAxMiB9LFxuICAgICAgYm9yZGVyUmFkaXVzOiAnbGcnLFxuICAgICAgdHJhbnNpdGlvbjogJ2JhY2tncm91bmQtY29sb3IgMC4xNXMgZWFzZScsXG4gICAgICBwb2ludGVyRXZlbnRzOiAnbm9uZScsXG4gICAgICB6SW5kZXg6IC0xLFxuICAgIH0sXG4gICAgJyY6aG92ZXI6OmJlZm9yZSwgJjpmb2N1cy12aXNpYmxlOjpiZWZvcmUnOiB7XG4gICAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLmNvbG9yc1snbmF2eS0xMDAnXSxcbiAgICB9LFxuICAgICcmOmhvdmVyLCAmOmZvY3VzLXZpc2libGUsICY6YWN0aXZlJzoge1xuICAgICAgY29sb3I6ICdoeXBlci00MDAnLFxuICAgICAgYm9yZGVyUmFkaXVzOiAnbGcnLFxuICAgIH0sXG4gICAgJyY6YWN0aXZlOjpiZWZvcmUnOiB7XG4gICAgICBiYWNrZ3JvdW5kQ29sb3I6IHRoZW1lLmNvbG9yc1snbmF2eS0yMDAnXSxcbiAgICB9LFxuICB9KVxuKTtcbiJdfQ== */");
@@ -1,4 +0,0 @@
1
- import { AppHeaderLinkProps } from '../../AppHeaderLink';
2
- export type AppHeaderDropdownLinkProps = AppHeaderLinkProps & {
3
- isFirstElem?: boolean;
4
- };
@@ -1,2 +0,0 @@
1
- export * from './elements/AppHeaderDropdownLink';
2
- export * from './elements/AppHeaderMenuItem';
@@ -1,2 +0,0 @@
1
- export * from './elements/AppHeaderDropdownLink';
2
- export * from './elements/AppHeaderMenuItem';
@@ -1,18 +0,0 @@
1
- import { GamutIconProps } from '@codecademy/gamut-icons';
2
- interface AppHeaderNavButtonBaseProps {
3
- buttonRef: React.RefObject<HTMLButtonElement>;
4
- handleOnClick: (event: React.MouseEvent) => void;
5
- isOpen: boolean;
6
- title?: string;
7
- }
8
- interface AppHeaderTextNavButtonProps extends AppHeaderNavButtonBaseProps {
9
- icon?: never;
10
- isIconOnly?: never;
11
- }
12
- interface AppHeaderIconNavButtonProps extends AppHeaderNavButtonBaseProps {
13
- icon: React.ComponentType<GamutIconProps>;
14
- isIconOnly?: boolean;
15
- }
16
- type AppHeaderNavButtonProps = AppHeaderTextNavButtonProps | AppHeaderIconNavButtonProps;
17
- export declare const AppHeaderDropdownNavButton: React.FC<AppHeaderNavButtonProps>;
18
- export {};