@codecademy/brand 3.25.0 → 3.26.0-alpha.9d629b471a.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.
@@ -0,0 +1,5 @@
1
+ interface SearchTracking {
2
+ onSearchAsYouType?: (query: string, searchId: string, resultsCount: number, queryLoadTime: number) => void;
3
+ }
4
+ export declare const useSearchTracking: ({ onSearchAsYouType }: SearchTracking) => void;
5
+ export {};
@@ -0,0 +1,46 @@
1
+ import { useEffect, useRef } from 'react';
2
+ import { useSearchTrackingContext } from '../SearchTrackingProvider';
3
+ import { isSimilarSetOfStrings } from '../../utils/string-similarity';
4
+ import { usePrevious } from 'react-use';
5
+ export const useSearchTracking = ({
6
+ onSearchAsYouType
7
+ }) => {
8
+ const {
9
+ onSearchAsYouTypeParams
10
+ } = useSearchTrackingContext();
11
+
12
+ // store the previous search event params
13
+ const prevSearchEventParams = usePrevious(onSearchAsYouTypeParams);
14
+ const lastTrackedQueryRef = useRef('');
15
+ useEffect(() => {
16
+ // skip search tracking entirely if tracking function is not provided
17
+ if (!onSearchAsYouType) {
18
+ return;
19
+ }
20
+
21
+ // skip empty queries
22
+ if (onSearchAsYouTypeParams?.query.length === 0) {
23
+ return;
24
+ }
25
+ let searchEventDelay;
26
+
27
+ // scenario 1: track when user's search has no results
28
+ if (onSearchAsYouTypeParams?.resultsCount === 0) {
29
+ searchEventDelay = setTimeout(() => {
30
+ onSearchAsYouType(onSearchAsYouTypeParams.query, onSearchAsYouTypeParams.searchId, 0, onSearchAsYouTypeParams.queryLoadTime);
31
+ lastTrackedQueryRef.current = onSearchAsYouTypeParams.query;
32
+ }, 1000);
33
+ }
34
+
35
+ // scenario 2: track when user's search has meaningful edits
36
+ if (prevSearchEventParams && onSearchAsYouTypeParams?.query !== prevSearchEventParams.query && prevSearchEventParams.query !== '' && prevSearchEventParams.resultsCount > 0 && prevSearchEventParams.query !== lastTrackedQueryRef.current && !isSimilarSetOfStrings(onSearchAsYouTypeParams?.query ?? '', prevSearchEventParams.query)) {
37
+ searchEventDelay = setTimeout(() => {
38
+ onSearchAsYouType(prevSearchEventParams.query, prevSearchEventParams.searchId, prevSearchEventParams.resultsCount, prevSearchEventParams.queryLoadTime);
39
+ lastTrackedQueryRef.current = prevSearchEventParams.query;
40
+ }, 1500);
41
+ }
42
+ return () => {
43
+ clearTimeout(searchEventDelay);
44
+ };
45
+ }, [onSearchAsYouType, onSearchAsYouTypeParams, prevSearchEventParams]);
46
+ };
@@ -0,0 +1,17 @@
1
+ /**
2
+ * Determines whether two sets of words are similar based on the Jaccard index and
3
+ * a specified similarity threshold.
4
+ *
5
+ * @param a - A set of strings to compare against the second set of strings.
6
+ * @param b - A set of strings to compare against the first set of strings.
7
+ * @param threshold - A float between 0 and 1, where 1 means the strings are
8
+ * identical and 0 means the strings are completely different.
9
+ *
10
+ * The threshold is used to determine if the strings are similar.
11
+ * By default, if the threshold is 0.8, then the strings are considered similar
12
+ * if at least 80% of the words in the first set of strings are also in the
13
+ * second set of strings.
14
+ *
15
+ * @returns - It returns true if the strings are similar, false otherwise.
16
+ */
17
+ export declare const isSimilarSetOfStrings: (a: string, b: string, threshold?: number) => boolean;
@@ -0,0 +1,30 @@
1
+ const transformSanitizedStringToArray = s => s.trim().toLowerCase().split(/\s+/);
2
+
3
+ /**
4
+ * Determines whether two sets of words are similar based on the Jaccard index and
5
+ * a specified similarity threshold.
6
+ *
7
+ * @param a - A set of strings to compare against the second set of strings.
8
+ * @param b - A set of strings to compare against the first set of strings.
9
+ * @param threshold - A float between 0 and 1, where 1 means the strings are
10
+ * identical and 0 means the strings are completely different.
11
+ *
12
+ * The threshold is used to determine if the strings are similar.
13
+ * By default, if the threshold is 0.8, then the strings are considered similar
14
+ * if at least 80% of the words in the first set of strings are also in the
15
+ * second set of strings.
16
+ *
17
+ * @returns - It returns true if the strings are similar, false otherwise.
18
+ */
19
+ export const isSimilarSetOfStrings = (a, b, threshold = 0.8) => {
20
+ const setA = Array.from(new Set(transformSanitizedStringToArray(a)));
21
+ const setB = Array.from(new Set(transformSanitizedStringToArray(b)));
22
+
23
+ // manually calculating intersection and union to support older browsers that may not have
24
+ // the Set.intersection and Set.union methods
25
+ const intersection = new Set([...setA].filter(s => setB.includes(s)));
26
+ const union = new Set([...setA, ...setB]);
27
+
28
+ // the Jaccard similarity is the ratio between the intersection and union of the two sets of words
29
+ return threshold <= intersection.size / union.size;
30
+ };
@@ -12,6 +12,7 @@ import { AppHeaderMainMenuMobile } from '../AppHeaderMobile/AppHeaderMainMenuMob
12
12
  import { HeaderHeightArea } from '../HeaderHeightArea';
13
13
  import { NotificationsContents } from '../Notifications/NotificationsContents';
14
14
  import { useHeaderNotifications } from '../Notifications/useHeaderNotifications';
15
+ import { SearchTrackingProvider } from '../AppHeader/Search/SearchTrackingProvider';
15
16
  import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
16
17
  const StyledOverlay = /*#__PURE__*/_styled(Overlay, {
17
18
  target: "e14c9jns1",
@@ -29,7 +30,7 @@ const StyledOverlay = /*#__PURE__*/_styled(Overlay, {
29
30
  left: 0,
30
31
  top: 0,
31
32
  overflowX: `hidden`
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            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      {!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"]} */");
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 { 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';\nimport { SearchTrackingProvider } from '../AppHeader/Search/SearchTrackingProvider';\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"]} */");
33
34
  const StyledNavBar = /*#__PURE__*/_styled("ul", {
34
35
  target: "e14c9jns0",
35
36
  label: "StyledNavBar"
@@ -47,7 +48,7 @@ const StyledNavBar = /*#__PURE__*/_styled("ul", {
47
48
  sm: 'flex-start'
48
49
  }
49
50
  }
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            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      {!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"]} */");
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 { 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';\nimport { SearchTrackingProvider } from '../AppHeader/Search/SearchTrackingProvider';\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"]} */");
51
52
  export const AppHeaderMobile = ({
52
53
  action,
53
54
  items,
@@ -115,101 +116,103 @@ export const AppHeaderMobile = ({
115
116
  setAllowScroll(false);
116
117
  }
117
118
  };
118
- return /*#__PURE__*/_jsxs(_Fragment, {
119
- children: [!mobileMenuOpen &&
120
- /*#__PURE__*/
121
- // need this bc AppBar has a hardcoded z-Index of 15
122
- _jsx(HeaderHeightArea, {
123
- display: {
124
- _: `block`,
125
- [appHeaderMobileBreakpoint]: `none`
126
- },
127
- children: /*#__PURE__*/_jsx(StyledAppBar, {
128
- as: "nav",
129
- "aria-label": "Main",
130
- children: /*#__PURE__*/_jsxs(StyledNavBar, {
131
- center: !!isSimple,
132
- children: [mapItemsToElement(items.left, 'left'), mapItemsToElement(right, 'right', true), !hideRightMenuButton && /*#__PURE__*/_jsx(AppHeaderListItem, {
133
- ml: right.length === 0 ? 'auto' : 0,
134
- children: /*#__PURE__*/_jsx(IconButton, {
135
- "aria-expanded": mobileMenuOpen,
136
- "aria-label": navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
137
- tip: 'Site\nnavigation',
138
- tipProps: {
139
- alignment: 'bottom-center',
140
- placement: 'floating'
141
- },
142
- "data-testid": "header-mobile-menu",
143
- onClick: () => {
144
- openMobileMenu();
145
- },
146
- icon: MenuIcon,
147
- variant: "interface",
148
- ref: openButtonRef
149
- })
150
- })]
151
- })
152
- })
153
- }), /*#__PURE__*/_jsx(StyledOverlay, {
154
- clickOutsideCloses: true,
155
- escapeCloses: true,
156
- isOpen: mobileMenuOpen,
157
- onRequestClose: () => setMobileMenuOpen(false),
158
- allowScroll: allowScroll,
159
- children: /*#__PURE__*/_jsx(Background, {
160
- bg: "beige",
161
- children: /*#__PURE__*/_jsxs(HeaderHeightArea, {
162
- display: {
163
- _: `block`,
164
- [appHeaderMobileBreakpoint]: `none`
165
- },
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, {
166
130
  as: "nav",
167
- ariaLabel: "Main",
168
- "data-testid": "header-mobile-menu-dropdown",
169
- children: [/*#__PURE__*/_jsx(StyledAppBar, {
170
- children: /*#__PURE__*/_jsxs(StyledNavBar, {
171
- children: [mapItemsToElement(items.left, 'left'), /*#__PURE__*/_jsx(AppHeaderListItem, {
172
- ml: "auto",
173
- children: /*#__PURE__*/_jsx(IconButton, {
174
- "aria-expanded": mobileMenuOpen,
175
- onClick: () => {
176
- setMobileMenuOpen(false);
177
- },
178
- icon: MenuIcon,
179
- "aria-label": navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
180
- tip: navigationMenuFormattedLabel?.siteNavigation || 'Site navigation',
181
- tipProps: {
182
- alignment: 'bottom-center',
183
- placement: 'floating'
184
- },
185
- ref: closeButtonRef
186
- })
187
- })]
188
- })
189
- }), /*#__PURE__*/_jsx(Box, {
190
- background: theme.colors.beige,
191
- height: "auto",
192
- children: /*#__PURE__*/_jsx(AppHeaderMainMenuMobile, {
193
- action: action,
194
- items: items.mainMenu,
195
- getItemType: onItemType,
196
- isAnon: isAnon
197
- })
198
- })]
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
+ })
154
+ })
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
+ })
199
202
  })
200
- })
201
- }), !isEnterprise && /*#__PURE__*/_jsx(Box, {
202
- display: {
203
- _: `block`,
204
- [appHeaderMobileBreakpoint]: `none`
205
- },
206
- children: searchPane
207
- }), /*#__PURE__*/_jsx(Box, {
208
- display: {
209
- _: `block`,
210
- [appHeaderMobileBreakpoint]: `none`
211
- },
212
- children: notificationsView
213
- })]
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
+ })
214
217
  });
215
218
  };
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.25.0",
4
+ "version": "3.26.0-alpha.9d629b471a.0",
5
5
  "author": "Codecademy Engineering <dev@codecademy.com>",
6
6
  "dependencies": {
7
7
  "@emotion/is-prop-valid": "^1.2.1",