@codecademy/brand 3.26.0-alpha.fa0b72ae29.0 → 3.27.0-alpha.745751754b.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/index.js +6 -28
  2. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdown/index.d.ts +0 -1
  3. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdown/index.js +10 -67
  4. package/dist/AppHeader/AppHeaderElements/{AppHeaderDropdownLinks/elements/StyledAppHeaderLink.d.ts → AppHeaderLinkSections/elements.d.ts} +6 -6
  5. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/elements.js +36 -0
  6. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/index.d.ts +1 -8
  7. package/dist/AppHeader/AppHeaderElements/AppHeaderLinkSections/index.js +32 -25
  8. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/index.d.ts +5 -2
  9. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/index.js +91 -24
  10. package/dist/AppHeader/AppHeaderElements/AppHeaderResourcesDropdown/index.js +5 -27
  11. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/NavSection.d.ts +2 -2
  12. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/NavSection.js +38 -20
  13. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/elements.js +8 -8
  14. package/dist/AppHeader/AppHeaderElements/AppHeaderSection/index.js +5 -13
  15. package/dist/AppHeader/Search/SearchPane.js +49 -24
  16. package/dist/AppHeader/index.js +4 -1
  17. package/dist/AppHeader/shared/elements.d.ts +9 -3
  18. package/dist/AppHeader/shared/elements.js +126 -26
  19. package/dist/AppHeader/shared/types.d.ts +1 -1
  20. package/dist/AppHeader/shared/utils.d.ts +1 -2
  21. package/dist/AppHeader/shared/utils.js +0 -2
  22. package/dist/AppHeaderMobile/index.js +96 -100
  23. package/dist/PlanCard/PricingAmount.js +6 -6
  24. package/dist/PlanCard/types.d.ts +1 -0
  25. package/dist/PlanCard/types.js +13 -1
  26. package/package.json +6 -1
  27. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderDropdownLink.d.ts +0 -3
  28. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderDropdownLink.js +0 -25
  29. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderMenuItem.d.ts +0 -3
  30. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/AppHeaderMenuItem.js +0 -26
  31. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/StyledAppHeaderLink.js +0 -36
  32. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/types.d.ts +0 -4
  33. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/elements/types.js +0 -1
  34. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/index.d.ts +0 -2
  35. package/dist/AppHeader/AppHeaderElements/AppHeaderDropdownLinks/index.js +0 -2
  36. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderDropdownNavButton.d.ts +0 -18
  37. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderDropdownNavButton.js +0 -50
  38. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderMenuNavButton.d.ts +0 -9
  39. package/dist/AppHeader/AppHeaderElements/AppHeaderNavButton/AppHeaderMenuNavButton.js +0 -38
  40. package/dist/AppHeader/Search/hooks/useSearchTracking.d.ts +0 -5
  41. package/dist/AppHeader/Search/hooks/useSearchTracking.js +0 -46
  42. package/dist/AppHeader/utils/string-similarity.d.ts +0 -17
  43. package/dist/AppHeader/utils/string-similarity.js +0 -30
@@ -48,7 +48,10 @@ export const AppHeader = ({
48
48
  // logging error to help with debugging
49
49
  if (items && items.some(item => !item)) {
50
50
  try {
51
- const itemTypesString = items.map(item => item.type).join(', ');
51
+ const itemTypesString = items.map(item => {
52
+ if (!item) return '**UNDEFINED ITEM**';
53
+ return item.type;
54
+ }).join(', ');
52
55
  logError?.(`Error: Found an undefined app header item on the ${side} side of the header.`, {
53
56
  message: `The current array of items includes the following header item types: ${itemTypesString}`
54
57
  });
@@ -51,12 +51,17 @@ export declare const StyledText: import("@emotion/styled").StyledComponent<(((Om
51
51
  }) & {
52
52
  theme?: import("@emotion/react").Theme;
53
53
  }, {}, {}>;
54
- export declare const AnimatedMegaMenuDropdown: import("@emotion/styled").StyledComponent<Omit<import("framer-motion").HTMLMotionProps<"div">, "ref"> & React.RefAttributes<HTMLDivElement> & {
54
+ declare const BaseDropdown: import("@emotion/styled").StyledComponent<Omit<import("framer-motion").HTMLMotionProps<"div">, "ref"> & React.RefAttributes<HTMLDivElement> & {
55
55
  theme?: import("@emotion/react").Theme;
56
56
  }, {}, {}>;
57
- export declare const AnimatedSimpleDropdown: import("@emotion/styled").StyledComponent<Omit<import("framer-motion").HTMLMotionProps<"div">, "ref"> & React.RefAttributes<HTMLDivElement> & {
57
+ export declare const AnimatedMegaMenuDropdown: React.ForwardRefExoticComponent<Omit<Omit<import("framer-motion").HTMLMotionProps<"div">, "ref"> & React.RefAttributes<HTMLDivElement> & {
58
58
  theme?: import("@emotion/react").Theme;
59
- }, {}, {}>;
59
+ } & {
60
+ isOpen: boolean;
61
+ }, "ref"> & React.RefAttributes<HTMLDivElement>>;
62
+ export declare const AnimatedSimpleDropdown: ({ isOpen, ...props }: React.ComponentProps<typeof BaseDropdown> & {
63
+ isOpen: boolean;
64
+ }) => import("react/jsx-runtime").JSX.Element;
60
65
  export declare const LayoutGridAntiAliased: import("@emotion/styled").StyledComponent<{
61
66
  theme?: import("@emotion/react").Theme | undefined;
62
67
  as?: import("react").ElementType<any, keyof import("react").JSX.IntrinsicElements> | undefined;
@@ -82,3 +87,4 @@ export declare const appHeaderSpacing: {
82
87
  readonly standard: 8;
83
88
  readonly enterprise: 12;
84
89
  };
90
+ export {};
@@ -1,12 +1,13 @@
1
1
  import _styled from "@emotion/styled/base";
2
2
  function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
3
- import { Anchor, FlexBox, LayoutGrid, Text } from '@codecademy/gamut';
3
+ import { Anchor, Box, FlexBox, LayoutGrid, Text } from '@codecademy/gamut';
4
4
  import { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';
5
5
  import { css, pxRem, states, useCurrentMode } from '@codecademy/gamut-styles';
6
- import { AnimatePresence, motion } from 'framer-motion';
6
+ import { AnimatePresence, motion, useReducedMotion } from 'framer-motion';
7
7
  import * as React from 'react';
8
8
  import { AppBar } from '../../AppBar';
9
- import { jsx as _jsx } from "react/jsx-runtime";
9
+ import { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';
10
+ import { jsx as _jsx, Fragment as _Fragment, jsxs as _jsxs } from "react/jsx-runtime";
10
11
  export const appHeaderMobileBreakpoint = 'lg';
11
12
  const animatedPopoverVariants = {
12
13
  enter: {
@@ -22,7 +23,6 @@ const animatedPopoverVariants = {
22
23
  }
23
24
  }
24
25
  };
25
- const borderColor = 'border-primary';
26
26
  export const useMegaMenuHeaderResponsiveStyles = () => {
27
27
  const mode = useCurrentMode();
28
28
  const bg = mode === 'dark' ? 'white' : 'navy-800';
@@ -59,7 +59,7 @@ export const DropdownAnchor = /*#__PURE__*/_styled(Anchor, {
59
59
  '&:focus::before': {
60
60
  opacity: 1
61
61
  }
62
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFrRThCIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
62
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AAuE8B","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
63
63
  const dropdownStates = states({
64
64
  open: {
65
65
  transform: `rotate(-180deg)`
@@ -72,7 +72,7 @@ export const DropdownIcon = /*#__PURE__*/_styled(ArrowChevronDownFilledIcon, {
72
72
  marginLeft: pxRem(5),
73
73
  transition: `transform 0.35s ease-out`,
74
74
  transformOrigin: `center ${pxRem(5)}`
75
- }), dropdownStates, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFzRjRCIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
75
+ }), dropdownStates, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AA2F4B","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
76
76
  export const StyledText = /*#__PURE__*/_styled(Text, {
77
77
  target: "e1xddtpe6",
78
78
  label: "StyledText"
@@ -86,26 +86,126 @@ export const StyledText = /*#__PURE__*/_styled(Text, {
86
86
  overflow: `hidden`,
87
87
  visibility: `hidden`
88
88
  }
89
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFpRzBCIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
90
- const dropdownAnimations = {
91
- overflow: `hidden`,
92
- position: `absolute`
89
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AAsG0B","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
90
+ const getDropdownAnimations = ({
91
+ isOpen,
92
+ prefersReducedMotion
93
+ }) => {
94
+ const openState = {
95
+ transform: 'translateY(0)',
96
+ opacity: 1
97
+ };
98
+ const closedState = {
99
+ transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',
100
+ opacity: 0
101
+ };
102
+ return {
103
+ animate: isOpen ? openState : closedState,
104
+ initial: closedState,
105
+ transition: {
106
+ duration: 0.15,
107
+ ease: 'easeOut'
108
+ }
109
+ };
93
110
  };
94
- export const AnimatedMegaMenuDropdown = /*#__PURE__*/_styled(motion.create('div'), {
111
+ const BaseDropdown = /*#__PURE__*/_styled(motion.create('div'), {
95
112
  target: "e1xddtpe5",
96
- label: "AnimatedMegaMenuDropdown"
97
- })(css({
98
- bg: `background`,
99
- borderColor,
100
- borderStyle: `solid`,
101
- ...dropdownAnimations
102
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFvSHdDIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
103
- export const AnimatedSimpleDropdown = /*#__PURE__*/_styled(motion.create('div'), {
113
+ label: "BaseDropdown"
114
+ })(() => css({
115
+ position: `absolute`
116
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AA8IqB","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
117
+ const MegaMenuBackdrop = /*#__PURE__*/_styled(motion.create('div'), {
104
118
  target: "e1xddtpe4",
105
- label: "AnimatedSimpleDropdown"
119
+ label: "MegaMenuBackdrop"
106
120
  })(css({
107
- ...dropdownAnimations
108
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUE2SHNDIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
121
+ position: 'relative',
122
+ zIndex: -1
123
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AAkJyB","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
124
+ export const AnimatedMegaMenuDropdown = /*#__PURE__*/React.forwardRef(({
125
+ isOpen,
126
+ ...props
127
+ }, ref) => {
128
+ const [isAnimating, setIsAnimating] = React.useState(false);
129
+ const prefersReducedMotion = useReducedMotion();
130
+ return /*#__PURE__*/_jsxs(_Fragment, {
131
+ children: [/*#__PURE__*/_jsx(AnimatePresence, {
132
+ children: isOpen && /*#__PURE__*/_jsx(MegaMenuBackdrop, {
133
+ initial: {
134
+ opacity: 0
135
+ },
136
+ animate: {
137
+ opacity: 1
138
+ },
139
+ exit: {
140
+ opacity: 0
141
+ },
142
+ transition: {
143
+ duration: 0.15,
144
+ ease: 'easeOut'
145
+ },
146
+ children: /*#__PURE__*/_jsx(GradientBackground, {
147
+ position: "fixed",
148
+ top: "0",
149
+ left: "0",
150
+ right: "0",
151
+ bottom: "0"
152
+ }, "dropdown-backdrop")
153
+ })
154
+ }), /*#__PURE__*/_jsx(Box, {
155
+ display: isOpen || isAnimating ? 'block' : 'none',
156
+ children: /*#__PURE__*/_jsx(BaseDropdown, {
157
+ ref: ref,
158
+ ...getDropdownAnimations({
159
+ isOpen,
160
+ prefersReducedMotion
161
+ }),
162
+ ...props,
163
+ onAnimationComplete: anim => {
164
+ if (!isOpen) {
165
+ // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes
166
+ setIsAnimating(false);
167
+ }
168
+ props.onAnimationComplete?.(anim);
169
+ },
170
+ onAnimationStart: anim => {
171
+ setIsAnimating(true);
172
+ props.onAnimationStart?.(anim);
173
+ }
174
+ }, "dropdown-content")
175
+ })]
176
+ });
177
+ });
178
+ export const AnimatedSimpleDropdown = ({
179
+ isOpen,
180
+ ...props
181
+ }) => {
182
+ const [isAnimating, setIsAnimating] = React.useState(false);
183
+ const prefersReducedMotion = useReducedMotion();
184
+
185
+ // the preferred way to animate this would be via AnimatePresence, but it is unclear if
186
+ // keeping closed dropdown content in the DOM has a positive effect on SEO
187
+ return /*#__PURE__*/_jsx(Box, {
188
+ display: isOpen || isAnimating ? 'block' : 'none',
189
+ children: /*#__PURE__*/_jsx(BaseDropdown, {
190
+ ...getDropdownAnimations({
191
+ isOpen,
192
+ prefersReducedMotion
193
+ }),
194
+ ...props,
195
+ onAnimationComplete: anim => {
196
+ if (!isOpen) {
197
+ // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes
198
+ setIsAnimating(false);
199
+ }
200
+ props.onAnimationComplete?.(anim);
201
+ },
202
+ onAnimationStart: anim => {
203
+ setIsAnimating(true);
204
+ props.onAnimationStart?.(anim);
205
+ }
206
+ })
207
+ });
208
+ };
109
209
  export const LayoutGridAntiAliased = /*#__PURE__*/_styled(LayoutGrid, {
110
210
  target: "e1xddtpe3",
111
211
  label: "LayoutGridAntiAliased"
@@ -114,7 +214,7 @@ export const LayoutGridAntiAliased = /*#__PURE__*/_styled(LayoutGrid, {
114
214
  styles: "-webkit-font-smoothing:antialiased"
115
215
  } : {
116
216
  name: "1y5tre4",
117
- styles: "-webkit-font-smoothing:antialiased/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFtSXVEIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */",
217
+ styles: "-webkit-font-smoothing:antialiased/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AAqPuD","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */",
118
218
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
119
219
  });
120
220
 
@@ -127,13 +227,13 @@ export const DescriptionSectionContainer = /*#__PURE__*/_styled(FlexBox, {
127
227
  color: 'text',
128
228
  outline: '1px solid currentColor !important'
129
229
  }
130
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF3STJDIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
230
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AA0P2C","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
131
231
  export const StyledAppBar = /*#__PURE__*/_styled(AppBar, {
132
232
  target: "e1xddtpe1",
133
233
  label: "StyledAppBar"
134
234
  })(css({
135
235
  boxShadow: `none`
136
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFpSjRCIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
236
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AAmQ4B","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
137
237
  export const StyledNavBar = /*#__PURE__*/_styled("ul", {
138
238
  target: "e1xddtpe0",
139
239
  label: "StyledNavBar"
@@ -144,7 +244,7 @@ export const StyledNavBar = /*#__PURE__*/_styled("ul", {
144
244
  listStyle: `none`,
145
245
  margin: 0,
146
246
  width: `100%`
147
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUF1SjRCIiwiZmlsZSI6Ii4uLy4uLy4uL3NyYy9BcHBIZWFkZXIvc2hhcmVkL2VsZW1lbnRzLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFuY2hvcixcbiAgRmxleEJveCxcbiAgTGF5b3V0R3JpZCxcbiAgVGV4dCxcbiAgV2l0aENoaWxkcmVuUHJvcCxcbn0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQnO1xuaW1wb3J0IHsgQXJyb3dDaGV2cm9uRG93bkZpbGxlZEljb24gfSBmcm9tICdAY29kZWNhZGVteS9nYW11dC1pY29ucyc7XG5pbXBvcnQge1xuICBDb2xvcnMsXG4gIGNzcyxcbiAgcHhSZW0sXG4gIHN0YXRlcyxcbiAgdXNlQ3VycmVudE1vZGUsXG59IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgeyBTdHlsZVByb3BzIH0gZnJvbSAnQGNvZGVjYWRlbXkvdmFyaWFuY2UnO1xuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnO1xuaW1wb3J0IHsgQW5pbWF0ZVByZXNlbmNlLCBtb3Rpb24sIFZhcmlhbnRzIH0gZnJvbSAnZnJhbWVyLW1vdGlvbic7XG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCc7XG5cbmltcG9ydCB7IEFwcEJhciB9IGZyb20gJy4uLy4uL0FwcEJhcic7XG5cbmV4cG9ydCBjb25zdCBhcHBIZWFkZXJNb2JpbGVCcmVha3BvaW50ID0gJ2xnJyBhcyBjb25zdDtcblxuZXhwb3J0IGludGVyZmFjZSBBbmltYXRlZEhlYWRlclpvbmVQcm9wcyBleHRlbmRzIFdpdGhDaGlsZHJlblByb3Age1xuICB2aXNpYmxlPzogYm9vbGVhbjtcbn1cblxuY29uc3QgYW5pbWF0ZWRQb3BvdmVyVmFyaWFudHM6IFZhcmlhbnRzID0ge1xuICBlbnRlcjogeyBvcGFjaXR5OiAxLCB0cmFuc2l0aW9uOiB7IGR1cmF0aW9uOiAwLjIgfSB9LFxuICBleGl0OiB7IG9wYWNpdHk6IDAsIHRyYW5zaXRpb246IHsgZHVyYXRpb246IDAuMiB9IH0sXG59O1xuXG5jb25zdCBib3JkZXJDb2xvciA9ICdib3JkZXItcHJpbWFyeSc7XG5cbmV4cG9ydCBjb25zdCB1c2VNZWdhTWVudUhlYWRlclJlc3BvbnNpdmVTdHlsZXMgPSAoKToge1xuICBiZzogQ29sb3JzO1xuICBib3JkZXJDb2xvcjogQ29sb3JzO1xuICBjb2xvcjogQ29sb3JzO1xufSA9PiB7XG4gIGNvbnN0IG1vZGUgPSB1c2VDdXJyZW50TW9kZSgpO1xuXG4gIGNvbnN0IGJnID0gbW9kZSA9PT0gJ2RhcmsnID8gJ3doaXRlJyA6ICduYXZ5LTgwMCc7XG4gIGNvbnN0IGNvbG9yID0gbW9kZSA9PT0gJ2RhcmsnID8gJ25hdnktOTAwJyA6ICdibHVlLTAnO1xuXG4gIHJldHVybiB7IGJnLCBib3JkZXJDb2xvcjogY29sb3IsIGNvbG9yIH07XG59O1xuXG5leHBvcnQgY29uc3QgQW5pbWF0ZWRIZWFkZXJab25lOiBSZWFjdC5GQzxBbmltYXRlZEhlYWRlclpvbmVQcm9wcz4gPSAoe1xuICBjaGlsZHJlbixcbiAgdmlzaWJsZSxcbn0pID0+IHtcbiAgcmV0dXJuIHZpc2libGUgPyAoXG4gICAgPEFuaW1hdGVQcmVzZW5jZT5cbiAgICAgIDxtb3Rpb24uZGl2XG4gICAgICAgIGFuaW1hdGU9XCJlbnRlclwiXG4gICAgICAgIGV4aXQ9XCJleGl0XCJcbiAgICAgICAgaW5pdGlhbD1cImV4aXRcIlxuICAgICAgICB2YXJpYW50cz17YW5pbWF0ZWRQb3BvdmVyVmFyaWFudHN9XG4gICAgICA+XG4gICAgICAgIHtjaGlsZHJlbn1cbiAgICAgIDwvbW90aW9uLmRpdj5cbiAgICA8L0FuaW1hdGVQcmVzZW5jZT5cbiAgKSA6IG51bGw7XG59O1xuXG5leHBvcnQgY29uc3QgRHJvcGRvd25BbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiBgY2VudGVyYCxcbiAgICBkaXNwbGF5OiBgZmxleGAsXG4gICAgcGFkZGluZzogYDAuNXJlbSAwYCxcbiAgICB0ZXh0QWxpZ246IGBjZW50ZXJgLFxuICAgIHdoaXRlU3BhY2U6IGBub3dyYXBgLFxuXG4gICAgJyY6Zm9jdXM6OmJlZm9yZSc6IHtcbiAgICAgIG9wYWNpdHk6IDEsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duU3RhdGVzID0gc3RhdGVzKHtcbiAgb3Blbjoge1xuICAgIHRyYW5zZm9ybTogYHJvdGF0ZSgtMTgwZGVnKWAsXG4gIH0sXG59KTtcblxuZXhwb3J0IGNvbnN0IERyb3Bkb3duSWNvbiA9IHN0eWxlZChBcnJvd0NoZXZyb25Eb3duRmlsbGVkSWNvbik8XG4gIFN0eWxlUHJvcHM8dHlwZW9mIGRyb3Bkb3duU3RhdGVzPlxuPihcbiAgY3NzKHtcbiAgICBtYXJnaW5MZWZ0OiBweFJlbSg1KSxcbiAgICB0cmFuc2l0aW9uOiBgdHJhbnNmb3JtIDAuMzVzIGVhc2Utb3V0YCxcbiAgICB0cmFuc2Zvcm1PcmlnaW46IGBjZW50ZXIgJHtweFJlbSg1KX1gLFxuICB9KSxcbiAgZHJvcGRvd25TdGF0ZXNcbik7XG5cbmV4cG9ydCBjb25zdCBTdHlsZWRUZXh0ID0gc3R5bGVkKFRleHQpKFxuICBjc3Moe1xuICAgICcmOjphZnRlcic6IHtcbiAgICAgIGRpc3BsYXk6IGBibG9ja2AsXG4gICAgICBjb250ZW50OiBgYXR0cih0aXRsZSlgLFxuICAgICAgZm9udFdlaWdodDogYGJvbGRgLFxuICAgICAgaGVpZ2h0OiBgMXB4YCxcbiAgICAgIGNvbG9yOiBgdHJhbnNwYXJlbnRgLFxuICAgICAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICAgICAgdmlzaWJpbGl0eTogYGhpZGRlbmAsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IGRyb3Bkb3duQW5pbWF0aW9ucyA9IHtcbiAgb3ZlcmZsb3c6IGBoaWRkZW5gLFxuICBwb3NpdGlvbjogYGFic29sdXRlYCxcbn0gYXMgY29uc3Q7XG5cbmV4cG9ydCBjb25zdCBBbmltYXRlZE1lZ2FNZW51RHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIGJnOiBgYmFja2dyb3VuZGAsXG4gICAgYm9yZGVyQ29sb3IsXG4gICAgYm9yZGVyU3R5bGU6IGBzb2xpZGAsXG4gICAgLi4uZHJvcGRvd25BbmltYXRpb25zLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IEFuaW1hdGVkU2ltcGxlRHJvcGRvd24gPSBzdHlsZWQobW90aW9uLmNyZWF0ZSgnZGl2JykpKFxuICBjc3Moe1xuICAgIC4uLmRyb3Bkb3duQW5pbWF0aW9ucyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBMYXlvdXRHcmlkQW50aUFsaWFzZWQgPSBzdHlsZWQoTGF5b3V0R3JpZClgXG4gIC13ZWJraXQtZm9udC1zbW9vdGhpbmc6IGFudGlhbGlhc2VkO1xuYDtcblxuLyogZm9yIFJlc291cmNlcyAmIENhdGFsb2cgbWVudXMgKi9cbmV4cG9ydCBjb25zdCBEZXNjcmlwdGlvblNlY3Rpb25Db250YWluZXIgPSBzdHlsZWQoRmxleEJveCkoXG4gIGNzcyh7XG4gICAgJyY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICBvdXRsaW5lOiAnMXB4IHNvbGlkIGN1cnJlbnRDb2xvciAhaW1wb3J0YW50JyxcbiAgICB9LFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZEFwcEJhciA9IHN0eWxlZChBcHBCYXIpKFxuICBjc3Moe1xuICAgIGJveFNoYWRvdzogYG5vbmVgLFxuICB9KVxuKTtcblxuZXhwb3J0IGNvbnN0IFN0eWxlZE5hdkJhciA9IHN0eWxlZC51bChcbiAgY3NzKHtcbiAgICBhbGlnbkl0ZW1zOiAnc3RyZXRjaCcsXG4gICAgZGlzcGxheTogYGZsZXhgLFxuICAgIHBhZGRpbmc6IDAsXG4gICAgbGlzdFN0eWxlOiBgbm9uZWAsXG4gICAgbWFyZ2luOiAwLFxuICAgIHdpZHRoOiBgMTAwJWAsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgYXBwSGVhZGVyU3BhY2luZyA9IHtcbiAgc3RhbmRhcmQ6IDgsXG5cbiAgZW50ZXJwcmlzZTogMTIsXG59IGFzIGNvbnN0O1xuIl19 */");
247
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/AppHeader/shared/elements.tsx"],"names":[],"mappings":"AAyQ4B","file":"../../../src/AppHeader/shared/elements.tsx","sourcesContent":["import {\n  Anchor,\n  Box,\n  FlexBox,\n  LayoutGrid,\n  Text,\n  WithChildrenProp,\n} from '@codecademy/gamut';\nimport { ArrowChevronDownFilledIcon } from '@codecademy/gamut-icons';\nimport {\n  Colors,\n  css,\n  pxRem,\n  states,\n  useCurrentMode,\n} from '@codecademy/gamut-styles';\nimport { StyleProps } from '@codecademy/variance';\nimport styled from '@emotion/styled';\nimport {\n  AnimatePresence,\n  motion,\n  useReducedMotion,\n  Variants,\n} from 'framer-motion';\nimport * as React from 'react';\n\nimport { AppBar } from '../../AppBar';\nimport { GradientBackground } from '../AppHeaderElements/AppHeaderSection/elements';\n\nexport const appHeaderMobileBreakpoint = 'lg' as const;\n\nexport interface AnimatedHeaderZoneProps extends WithChildrenProp {\n  visible?: boolean;\n}\n\nconst animatedPopoverVariants: Variants = {\n  enter: { opacity: 1, transition: { duration: 0.2 } },\n  exit: { opacity: 0, transition: { duration: 0.2 } },\n};\n\nexport const useMegaMenuHeaderResponsiveStyles = (): {\n  bg: Colors;\n  borderColor: Colors;\n  color: Colors;\n} => {\n  const mode = useCurrentMode();\n\n  const bg = mode === 'dark' ? 'white' : 'navy-800';\n  const color = mode === 'dark' ? 'navy-900' : 'blue-0';\n\n  return { bg, borderColor: color, color };\n};\n\nexport const AnimatedHeaderZone: React.FC<AnimatedHeaderZoneProps> = ({\n  children,\n  visible,\n}) => {\n  return visible ? (\n    <AnimatePresence>\n      <motion.div\n        animate=\"enter\"\n        exit=\"exit\"\n        initial=\"exit\"\n        variants={animatedPopoverVariants}\n      >\n        {children}\n      </motion.div>\n    </AnimatePresence>\n  ) : null;\n};\n\nexport const DropdownAnchor = styled(Anchor)(\n  css({\n    alignItems: `center`,\n    display: `flex`,\n    padding: `0.5rem 0`,\n    textAlign: `center`,\n    whiteSpace: `nowrap`,\n\n    '&:focus::before': {\n      opacity: 1,\n    },\n  })\n);\n\nconst dropdownStates = states({\n  open: {\n    transform: `rotate(-180deg)`,\n  },\n});\n\nexport const DropdownIcon = styled(ArrowChevronDownFilledIcon)<\n  StyleProps<typeof dropdownStates>\n>(\n  css({\n    marginLeft: pxRem(5),\n    transition: `transform 0.35s ease-out`,\n    transformOrigin: `center ${pxRem(5)}`,\n  }),\n  dropdownStates\n);\n\nexport const StyledText = styled(Text)(\n  css({\n    '&::after': {\n      display: `block`,\n      content: `attr(title)`,\n      fontWeight: `bold`,\n      height: `1px`,\n      color: `transparent`,\n      overflow: `hidden`,\n      visibility: `hidden`,\n    },\n  })\n);\n\nconst getDropdownAnimations = ({\n  isOpen,\n  prefersReducedMotion,\n}: {\n  isOpen: boolean;\n  prefersReducedMotion: boolean | null;\n}) => {\n  const openState = {\n    transform: 'translateY(0)',\n    opacity: 1,\n  };\n  const closedState = {\n    transform: prefersReducedMotion ? 'none' : 'translateY(-8px)',\n    opacity: 0,\n  };\n\n  return {\n    animate: isOpen ? openState : closedState,\n    initial: closedState,\n    transition: {\n      duration: 0.15,\n      ease: 'easeOut',\n    },\n  };\n};\n\nconst BaseDropdown = styled(motion.create('div'))(() =>\n  css({ position: `absolute` })\n);\n\nconst MegaMenuBackdrop = styled(motion.create('div'))(\n  css({\n    position: 'relative',\n    zIndex: -1,\n  })\n);\n\nexport const AnimatedMegaMenuDropdown = React.forwardRef(\n  (\n    {\n      isOpen,\n      ...props\n    }: React.ComponentProps<typeof BaseDropdown> & {\n      isOpen: boolean;\n    },\n    ref: React.ForwardedRef<HTMLDivElement>\n  ) => {\n    const [isAnimating, setIsAnimating] = React.useState(false);\n    const prefersReducedMotion = useReducedMotion();\n\n    return (\n      <>\n        <AnimatePresence>\n          {isOpen && (\n            <MegaMenuBackdrop\n              initial={{ opacity: 0 }}\n              animate={{ opacity: 1 }}\n              exit={{ opacity: 0 }}\n              transition={{ duration: 0.15, ease: 'easeOut' }}\n            >\n              <GradientBackground\n                key=\"dropdown-backdrop\"\n                position=\"fixed\"\n                top=\"0\"\n                left=\"0\"\n                right=\"0\"\n                bottom=\"0\"\n              />\n            </MegaMenuBackdrop>\n          )}\n        </AnimatePresence>\n        {/* the preferred way to animate this would be via AnimatePresence, but it is unclear if\n            keeping closed dropdown content in the DOM has a positive effect on SEO */}\n        <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n          <BaseDropdown\n            key=\"dropdown-content\"\n            ref={ref}\n            {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n            {...props}\n            onAnimationComplete={(anim) => {\n              if (!isOpen) {\n                // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n                setIsAnimating(false);\n              }\n              props.onAnimationComplete?.(anim);\n            }}\n            onAnimationStart={(anim) => {\n              setIsAnimating(true);\n              props.onAnimationStart?.(anim);\n            }}\n          />\n        </Box>\n      </>\n    );\n  }\n);\n\nexport const AnimatedSimpleDropdown = ({\n  isOpen,\n  ...props\n}: React.ComponentProps<typeof BaseDropdown> & {\n  isOpen: boolean;\n}) => {\n  const [isAnimating, setIsAnimating] = React.useState(false);\n  const prefersReducedMotion = useReducedMotion();\n\n  // the preferred way to animate this would be via AnimatePresence, but it is unclear if\n  // keeping closed dropdown content in the DOM has a positive effect on SEO\n  return (\n    <Box display={isOpen || isAnimating ? 'block' : 'none'}>\n      <BaseDropdown\n        {...getDropdownAnimations({ isOpen, prefersReducedMotion })}\n        {...props}\n        onAnimationComplete={(anim) => {\n          if (!isOpen) {\n            // only consider animation complete when dropdown is closed to prevent flickering from rapid state changes\n            setIsAnimating(false);\n          }\n          props.onAnimationComplete?.(anim);\n        }}\n        onAnimationStart={(anim) => {\n          setIsAnimating(true);\n          props.onAnimationStart?.(anim);\n        }}\n      />\n    </Box>\n  );\n};\n\nexport const LayoutGridAntiAliased = styled(LayoutGrid)`\n  -webkit-font-smoothing: antialiased;\n`;\n\n/* for Resources & Catalog menus */\nexport const DescriptionSectionContainer = styled(FlexBox)(\n  css({\n    '&:focus-visible': {\n      color: 'text',\n      outline: '1px solid currentColor !important',\n    },\n  })\n);\n\nexport const StyledAppBar = styled(AppBar)(\n  css({\n    boxShadow: `none`,\n  })\n);\n\nexport const StyledNavBar = styled.ul(\n  css({\n    alignItems: 'stretch',\n    display: `flex`,\n    padding: 0,\n    listStyle: `none`,\n    margin: 0,\n    width: `100%`,\n  })\n);\n\nexport const appHeaderSpacing = {\n  standard: 8,\n\n  enterprise: 12,\n} as const;\n"]} */");
148
248
  export const appHeaderSpacing = {
149
249
  standard: 8,
150
250
  enterprise: 12
@@ -103,5 +103,5 @@ export type FormattedAppHeaderItems = {
103
103
  export type FormattedMobileAppHeaderItems = FormattedAppHeaderItems & {
104
104
  mainMenu: AppHeaderItem[];
105
105
  };
106
- export type AnchorRefItemType = RefObject<HTMLAnchorElement>['current'];
106
+ export type AnchorRefItemType = RefObject<HTMLAnchorElement>['current'] | RefObject<HTMLButtonElement>['current'] | null;
107
107
  export {};
@@ -7,9 +7,8 @@ type AppHeaderItemToElementType = {
7
7
  isStandalone?: boolean;
8
8
  isTeams?: boolean;
9
9
  redirectParam?: string;
10
- onKeyDown?: (event: React.KeyboardEvent) => void;
11
10
  mobile?: boolean;
12
11
  spacing?: ComponentProps<typeof Menu>['spacing'];
13
12
  };
14
- export declare const mapAppHeaderItemToElement: ({ action, item, isStandalone, isTeams, redirectParam, onKeyDown, mobile, spacing, }: AppHeaderItemToElementType) => ReactNode;
13
+ export declare const mapAppHeaderItemToElement: ({ action, item, isStandalone, isTeams, redirectParam, mobile, spacing, }: AppHeaderItemToElementType) => ReactNode;
15
14
  export {};
@@ -12,7 +12,6 @@ export const mapAppHeaderItemToElement = ({
12
12
  isStandalone,
13
13
  isTeams,
14
14
  redirectParam,
15
- onKeyDown,
16
15
  mobile = false,
17
16
  spacing = 'normal'
18
17
  }) => {
@@ -34,7 +33,6 @@ export const mapAppHeaderItemToElement = ({
34
33
  case 'resources-simple-dropdown':
35
34
  case 'profile-dropdown':
36
35
  return /*#__PURE__*/_jsx(AppHeaderDropdown, {
37
- onKeyDown: onKeyDown,
38
36
  action: action,
39
37
  item: item,
40
38
  standalone: isStandalone,