@codecademy/brand 3.25.0 → 3.26.0-alpha.193077b42d.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.
- package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/MarketingBanner.js +4 -3
- package/dist/AppHeader/AppHeaderElements/AppHeaderSection/AppHeaderSection.test.js +107 -5
- package/dist/AppHeader/AppHeaderElements/AppHeaderSection/index.js +4 -2
- package/dist/AppHeader/Search/SearchPane.js +24 -49
- package/dist/AppHeader/Search/hooks/useSearchTracking.d.ts +5 -0
- package/dist/AppHeader/Search/hooks/useSearchTracking.js +46 -0
- package/dist/AppHeader/utils/string-similarity.d.ts +17 -0
- package/dist/AppHeader/utils/string-similarity.js +30 -0
- package/dist/AppHeaderMobile/index.js +99 -96
- package/package.json +1 -6
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { useEffect, useRef } from 'react';
|
|
2
|
+
import { usePrevious } from 'react-use';
|
|
3
|
+
import { isSimilarSetOfStrings } from '../../utils/string-similarity';
|
|
4
|
+
import { useSearchTrackingContext } from '../SearchTrackingProvider';
|
|
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
|
+
}, 1500);
|
|
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
|
+
};
|
|
@@ -5,6 +5,7 @@ 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';
|
|
8
9
|
import { useHeaderSearch } from '../AppHeader/Search/useHeaderSearch';
|
|
9
10
|
import { appHeaderMobileBreakpoint, StyledAppBar } from '../AppHeader/shared';
|
|
10
11
|
import { mapAppHeaderItemToElement } from '../AppHeader/shared/utils';
|
|
@@ -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 { 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"]} */");
|
|
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 { 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"]} */");
|
|
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__*/
|
|
119
|
-
children:
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
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
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
children: /*#__PURE__*/
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
})
|
|
188
|
-
})
|
|
189
|
-
})
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
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
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
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.
|
|
4
|
+
"version": "3.26.0-alpha.193077b42d.0",
|
|
5
5
|
"author": "Codecademy Engineering <dev@codecademy.com>",
|
|
6
6
|
"dependencies": {
|
|
7
7
|
"@emotion/is-prop-valid": "^1.2.1",
|
|
@@ -27,12 +27,7 @@
|
|
|
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": "*",
|
|
34
30
|
"@codecademy/tracking": "1.2.0",
|
|
35
|
-
"@codecademy/variance": "*",
|
|
36
31
|
"@emotion/react": "^11.4.0",
|
|
37
32
|
"@emotion/styled": "^11.3.0",
|
|
38
33
|
"react": "^17.0.2 || ^18.2.0"
|