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