@codecademy/brand 4.2.0 → 4.2.1-alpha.21bfd035d.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.
@@ -10,6 +10,11 @@ export type InstructorCardProps = {
10
10
  instructorImageUrl?: string;
11
11
  /** Formatted start date for the next cohort (e.g. "Feb 10") */
12
12
  nextCohortStartDate?: string;
13
+ /**
14
+ * Promotional label from the latest cohort (e.g. "Last chance - 20% off").
15
+ * Rendered centered on the top edge of the card, overlapping the border.
16
+ */
17
+ promotionBadgeLabel?: string;
13
18
  /** Link destination */
14
19
  href: string;
15
20
  /** Click handler */
@@ -1,13 +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, Box, Card, FlexBox, Text } from '@codecademy/gamut';
3
+ import { Anchor, Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';
4
4
  import { CalendarIcon, CertificateIcon } from '@codecademy/gamut-icons';
5
5
  import { CheckerDense } from '@codecademy/gamut-patterns';
6
- import { css } from '@codecademy/gamut-styles';
6
+ import { css, theme } from '@codecademy/gamut-styles';
7
7
  import { forwardRef } from 'react';
8
8
  import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
9
9
  const CardAnchor = /*#__PURE__*/_styled(Anchor, {
10
- target: "e171y2mq2",
10
+ target: "e171y2mq4",
11
11
  label: "CardAnchor"
12
12
  })(css({
13
13
  display: 'flex',
@@ -17,18 +17,26 @@ const CardAnchor = /*#__PURE__*/_styled(Anchor, {
17
17
  color: 'text',
18
18
  textDecoration: 'none'
19
19
  }
20
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9JbnN0cnVjdG9yQ2FyZC9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBT21CIiwiZmlsZSI6Ii4uLy4uL3NyYy9JbnN0cnVjdG9yQ2FyZC9pbmRleC50c3giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBbmNob3IsIEJveCwgQ2FyZCwgRmxleEJveCwgVGV4dCB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0JztcbmltcG9ydCB7IENhbGVuZGFySWNvbiwgQ2VydGlmaWNhdGVJY29uIH0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQtaWNvbnMnO1xuaW1wb3J0IHsgQ2hlY2tlckRlbnNlIH0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQtcGF0dGVybnMnO1xuaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGNvZGVjYWRlbXkvZ2FtdXQtc3R5bGVzJztcbmltcG9ydCBzdHlsZWQgZnJvbSAnQGVtb3Rpb24vc3R5bGVkJztcbmltcG9ydCB7IGZvcndhcmRSZWYsIFByb3BzV2l0aFJlZiB9IGZyb20gJ3JlYWN0JztcblxuY29uc3QgQ2FyZEFuY2hvciA9IHN0eWxlZChBbmNob3IpKFxuICBjc3Moe1xuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBmbGV4RGlyZWN0aW9uOiAnY29sdW1uJyxcbiAgICBoZWlnaHQ6ICcxMDAlJyxcbiAgICAnJjpob3ZlciwgJjpmb2N1cywgJjpmb2N1cy12aXNpYmxlJzoge1xuICAgICAgY29sb3I6ICd0ZXh0JyxcbiAgICAgIHRleHREZWNvcmF0aW9uOiAnbm9uZScsXG4gICAgfSxcbiAgfSlcbik7XG5cbmNvbnN0IFN0eWxlZEltZyA9IHN0eWxlZC5pbWdgXG4gIHdpZHRoOiAxMDAlO1xuICBoZWlnaHQ6IDEwMCU7XG4gIG9iamVjdC1maXQ6IGNvdmVyO1xuICBkaXNwbGF5OiBibG9jaztcbmA7XG5cbmNvbnN0IERvdHRlZExpbmUgPSBzdHlsZWQoQ2hlY2tlckRlbnNlKShcbiAgY3NzKHtcbiAgICBoZWlnaHQ6ICcxcHgnLFxuICAgIGRpc3BsYXk6ICdmbGV4JyxcbiAgICBwb3NpdGlvbjogJ2Fic29sdXRlJyxcbiAgICBtYXJnaW5Ub3A6ICctMXB4JyxcbiAgfSlcbik7XG5cbmV4cG9ydCBjb25zdCBEaXZpZGVyOiBSZWFjdC5GQyA9ICgpID0+IHtcbiAgcmV0dXJuIChcbiAgICAvLyB0aGUgaGVpZ2h0IGFuZCBwb3NpdGlvbiBwcm9wZXJ0aWVzIGFyZSBuZWNlc3NhcnkgZm9yIGlPUyBkZXZpY2VzXG4gICAgPEJveCBweT17NH0gaGVpZ2h0PXswfSBwb3NpdGlvbj1cInJlbGF0aXZlXCIgYXJpYS1oaWRkZW4+XG4gICAgICA8RG90dGVkTGluZSAvPlxuICAgIDwvQm94PlxuICApO1xufTtcblxudHlwZSBJbmZvUm93UHJvcHMgPSB7XG4gIGljb246IFJlYWN0LlJlYWN0Tm9kZTtcbiAgY2hpbGRyZW46IFJlYWN0LlJlYWN0Tm9kZTtcbn07XG5cbmNvbnN0IEluZm9Sb3c6IFJlYWN0LkZDPEluZm9Sb3dQcm9wcz4gPSAoeyBpY29uLCBjaGlsZHJlbiB9KSA9PiAoXG4gIDxGbGV4Qm94IGFsaWduSXRlbXM9XCJjZW50ZXJcIiBnYXA9ezh9PlxuICAgIHtpY29ufVxuICAgIDxUZXh0IGZvbnRTaXplPXsxNH0gbGluZUhlaWdodD1cImJhc2VcIiBjb2xvcj1cInRleHRcIj5cbiAgICAgIHtjaGlsZHJlbn1cbiAgICA8L1RleHQ+XG4gIDwvRmxleEJveD5cbik7XG5cbmV4cG9ydCB0eXBlIEluc3RydWN0b3JDYXJkUHJvcHMgPSB7XG4gIC8qKiBUaGUgYm9vdGNhbXAgdGl0bGUgKi9cbiAgdGl0bGU6IHN0cmluZztcbiAgLyoqIEluc3RydWN0b3IncyBmdWxsIG5hbWUgKi9cbiAgaW5zdHJ1Y3Rvck5hbWU6IHN0cmluZztcbiAgLyoqIEluc3RydWN0b3IncyBwcm9mZXNzaW9uYWwgdGl0bGUgKi9cbiAgaW5zdHJ1Y3RvclRpdGxlOiBzdHJpbmc7XG4gIC8qKiBVUkwgZm9yIHRoZSBpbnN0cnVjdG9yJ3MgcHJvZmlsZSBwaWMgKGluY2x1ZGVzIGJhY2tncm91bmQgZ3JhZGllbnQpICovXG4gIGluc3RydWN0b3JJbWFnZVVybD86IHN0cmluZztcbiAgLyoqIEZvcm1hdHRlZCBzdGFydCBkYXRlIGZvciB0aGUgbmV4dCBjb2hvcnQgKGUuZy4gXCJGZWIgMTBcIikgKi9cbiAgbmV4dENvaG9ydFN0YXJ0RGF0ZT86IHN0cmluZztcbiAgLyoqIExpbmsgZGVzdGluYXRpb24gKi9cbiAgaHJlZjogc3RyaW5nO1xuICAvKiogQ2xpY2sgaGFuZGxlciAqL1xuICBvbkNsaWNrOiAoKSA9PiB2b2lkO1xufTtcblxuZXhwb3J0IGNvbnN0IEluc3RydWN0b3JDYXJkID0gZm9yd2FyZFJlZjxcbiAgSFRNTEFuY2hvckVsZW1lbnQsXG4gIFByb3BzV2l0aFJlZjxJbnN0cnVjdG9yQ2FyZFByb3BzPlxuPihcbiAgKFxuICAgIHtcbiAgICAgIHRpdGxlLFxuICAgICAgaW5zdHJ1Y3Rvck5hbWUsXG4gICAgICBpbnN0cnVjdG9yVGl0bGUsXG4gICAgICBpbnN0cnVjdG9ySW1hZ2VVcmwsXG4gICAgICBuZXh0Q29ob3J0U3RhcnREYXRlLFxuICAgICAgaHJlZixcbiAgICAgIG9uQ2xpY2ssXG4gICAgfSxcbiAgICByZWZcbiAgKSA9PiB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxDYXJkXG4gICAgICAgIGJvcmRlclJhZGl1cz1cIm1kXCJcbiAgICAgICAgYm9yZGVyQ29sb3I9XCJib3JkZXItcHJpbWFyeVwiXG4gICAgICAgIG92ZXJmbG93PVwiaGlkZGVuXCJcbiAgICAgICAgZGlzcGxheT1cImZsZXhcIlxuICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgaXNJbnRlcmFjdGl2ZVxuICAgICAgICBwPXswfVxuICAgICAgICBoZWlnaHQ9XCIxMDAlXCJcbiAgICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgID5cbiAgICAgICAgPENhcmRBbmNob3IgaHJlZj17aHJlZn0gb25DbGljaz17b25DbGlja30gdmFyaWFudD1cImludGVyZmFjZVwiIHJlZj17cmVmfT5cbiAgICAgICAgICA8Qm94XG4gICAgICAgICAgICB3aWR0aD1cIjEwMCVcIlxuICAgICAgICAgICAgaGVpZ2h0PXsxNjB9XG4gICAgICAgICAgICBwb3NpdGlvbj1cInJlbGF0aXZlXCJcbiAgICAgICAgICAgIG92ZXJmbG93PVwiaGlkZGVuXCJcbiAgICAgICAgICAgIGJnPVwibmF2eS04MDBcIlxuICAgICAgICAgID5cbiAgICAgICAgICAgIHtpbnN0cnVjdG9ySW1hZ2VVcmwgJiYgKFxuICAgICAgICAgICAgICA8U3R5bGVkSW1nIHNyYz17aW5zdHJ1Y3RvckltYWdlVXJsfSBhbHQ9e2luc3RydWN0b3JOYW1lfSAvPlxuICAgICAgICAgICAgKX1cbiAgICAgICAgICA8L0JveD5cblxuICAgICAgICAgIDxGbGV4Qm94XG4gICAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgICAgIGZsZXg9ezF9XG4gICAgICAgICAgICBwdD17MTJ9XG4gICAgICAgICAgICBwYj17NH1cbiAgICAgICAgICAgIHB4PXsxMn1cbiAgICAgICAgICAgIG92ZXJmbG93PVwiaGlkZGVuXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICA8RmxleEJveFxuICAgICAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgICAgICAgYWxpZ25TZWxmPVwic3RyZXRjaFwiXG4gICAgICAgICAgICAgIGZsZXg9ezF9XG4gICAgICAgICAgICAgIGdhcD17NH1cbiAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICBhcz1cImgzXCJcbiAgICAgICAgICAgICAgICBmb250U2l6ZT17MjB9XG4gICAgICAgICAgICAgICAgZm9udFdlaWdodD1cImJvbGRcIlxuICAgICAgICAgICAgICAgIGxpbmVIZWlnaHQ9XCJ0aXRsZVwiXG4gICAgICAgICAgICAgICAgY29sb3I9XCJ0ZXh0XCJcbiAgICAgICAgICAgICAgICBtPXswfVxuICAgICAgICAgICAgICAgIHRydW5jYXRlPVwiZWxsaXBzaXNcIlxuICAgICAgICAgICAgICAgIHRydW5jYXRlTGluZXM9ezN9XG4gICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICB7dGl0bGV9XG4gICAgICAgICAgICAgIDwvVGV4dD5cblxuICAgICAgICAgICAgICA8RmxleEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZ2FwPXs0fSBtdD1cImF1dG9cIiBtaW5XaWR0aD17MH0+XG4gICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgIGZvbnRTaXplPXsxNH1cbiAgICAgICAgICAgICAgICAgIGZvbnRXZWlnaHQ9XCJib2xkXCJcbiAgICAgICAgICAgICAgICAgIGxpbmVIZWlnaHQ9XCJiYXNlXCJcbiAgICAgICAgICAgICAgICAgIGNvbG9yPVwidGV4dFwiXG4gICAgICAgICAgICAgICAgPlxuICAgICAgICAgICAgICAgICAge2luc3RydWN0b3JOYW1lfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgICA8VGV4dFxuICAgICAgICAgICAgICAgICAgZm9udFNpemU9ezE0fVxuICAgICAgICAgICAgICAgICAgbGluZUhlaWdodD1cImJhc2VcIlxuICAgICAgICAgICAgICAgICAgY29sb3I9XCJ0ZXh0LXNlY29uZGFyeVwiXG4gICAgICAgICAgICAgICAgICB0cnVuY2F0ZT1cImVsbGlwc2lzXCJcbiAgICAgICAgICAgICAgICAgIHRydW5jYXRlTGluZXM9ezF9XG4gICAgICAgICAgICAgICAgICB3aWR0aD1cIjEwMCVcIlxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIHtpbnN0cnVjdG9yVGl0bGV9XG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L0ZsZXhCb3g+XG4gICAgICAgICAgICA8L0ZsZXhCb3g+XG5cbiAgICAgICAgICAgIDxGbGV4Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBhbGlnblNlbGY9XCJzdHJldGNoXCIgbXQ9ezR9PlxuICAgICAgICAgICAgICB7bmV4dENvaG9ydFN0YXJ0RGF0ZSAmJiAoXG4gICAgICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgICAgIDxEaXZpZGVyIC8+XG4gICAgICAgICAgICAgICAgICA8SW5mb1JvdyBpY29uPXs8Q2FsZW5kYXJJY29uIHNpemU9ezE2fSBhcmlhLWhpZGRlbiAvPn0+XG4gICAgICAgICAgICAgICAgICAgIE5leHQgY29ob3J0OiBTdGFydHN7JyAnfVxuICAgICAgICAgICAgICAgICAgICA8VGV4dCBhcz1cInNwYW5cIiBmb250V2VpZ2h0PVwiYm9sZFwiPlxuICAgICAgICAgICAgICAgICAgICAgIHtuZXh0Q29ob3J0U3RhcnREYXRlfVxuICAgICAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgICA8L0luZm9Sb3c+XG4gICAgICAgICAgICAgICAgPC8+XG4gICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgIDxEaXZpZGVyIC8+XG4gICAgICAgICAgICAgIDxJbmZvUm93IGljb249ezxDZXJ0aWZpY2F0ZUljb24gc2l6ZT17MTZ9IGFyaWEtaGlkZGVuIC8+fT5cbiAgICAgICAgICAgICAgICBXaXRoeycgJ31cbiAgICAgICAgICAgICAgICA8VGV4dCBhcz1cInNwYW5cIiBmb250V2VpZ2h0PVwiYm9sZFwiPlxuICAgICAgICAgICAgICAgICAgQ2VydGlmaWNhdGVcbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgIDwvSW5mb1Jvdz5cbiAgICAgICAgICAgIDwvRmxleEJveD5cbiAgICAgICAgICA8L0ZsZXhCb3g+XG4gICAgICAgIDwvQ2FyZEFuY2hvcj5cbiAgICAgIDwvQ2FyZD5cbiAgICApO1xuICB9XG4pO1xuIl19 */");
20
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/InstructorCard/index.tsx"],"names":[],"mappings":"AAOmB","file":"../../src/InstructorCard/index.tsx","sourcesContent":["import { Anchor, Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CalendarIcon, CertificateIcon } from '@codecademy/gamut-icons';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, PropsWithRef } from 'react';\n\nconst CardAnchor = styled(Anchor)(\n  css({\n    display: 'flex',\n    flexDirection: 'column',\n    height: '100%',\n    '&:hover, &:focus, &:focus-visible': {\n      color: 'text',\n      textDecoration: 'none',\n    },\n  })\n);\n\nconst StyledImg = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  display: block;\n`;\n\nconst LastChanceBadge = styled(Badge)`\n  color: ${theme.colors.red};\n  border-color: ${theme.colors.red};\n  background-color: ${theme.colors['red-0']};\n  border: 1px solid ${theme.colors['red-500']};\n`;\n\nconst PromotionBadge = styled(Badge)`\n  color: ${theme.colors.black};\n  border-color: ${theme.colors.yellow};\n  background-color: ${theme.colors['yellow-0']};\n  border: 1px solid ${theme.colors['yellow-500']};\n`;\n\nconst DottedLine = styled(CheckerDense)(\n  css({\n    height: '1px',\n    display: 'flex',\n    position: 'absolute',\n    marginTop: '-1px',\n  })\n);\n\nexport const Divider: React.FC = () => {\n  return (\n    // the height and position properties are necessary for iOS devices\n    <Box py={4} height={0} position=\"relative\" aria-hidden>\n      <DottedLine />\n    </Box>\n  );\n};\n\ntype InfoRowProps = {\n  icon: React.ReactNode;\n  children: React.ReactNode;\n};\n\nconst InfoRow: React.FC<InfoRowProps> = ({ icon, children }) => (\n  <FlexBox alignItems=\"center\" gap={8}>\n    {icon}\n    <Text fontSize={14} lineHeight=\"base\" color=\"text\">\n      {children}\n    </Text>\n  </FlexBox>\n);\n\nexport type InstructorCardProps = {\n  /** The bootcamp title */\n  title: string;\n  /** Instructor's full name */\n  instructorName: string;\n  /** Instructor's professional title */\n  instructorTitle: string;\n  /** URL for the instructor's profile pic (includes background gradient) */\n  instructorImageUrl?: string;\n  /** Formatted start date for the next cohort (e.g. \"Feb 10\") */\n  nextCohortStartDate?: string;\n  /**\n   * Promotional label from the latest cohort (e.g. \"Last chance - 20% off\").\n   * Rendered centered on the top edge of the card, overlapping the border.\n   */\n  promotionBadgeLabel?: string;\n  /** Link destination */\n  href: string;\n  /** Click handler */\n  onClick: () => void;\n};\n\nexport const InstructorCard = forwardRef<\n  HTMLAnchorElement,\n  PropsWithRef<InstructorCardProps>\n>(\n  (\n    {\n      title,\n      instructorName,\n      instructorTitle,\n      instructorImageUrl,\n      nextCohortStartDate,\n      promotionBadgeLabel,\n      href,\n      onClick,\n    },\n    ref\n  ) => {\n    const isLastChance = promotionBadgeLabel\n      ?.toLowerCase()\n      .includes('last chance');\n\n    const PromotionBadgeComponent = isLastChance\n      ? LastChanceBadge\n      : PromotionBadge;\n\n    return (\n      <Box position=\"relative\" width=\"100%\" height=\"100%\">\n        {promotionBadgeLabel ? (\n          <FlexBox\n            justifyContent=\"center\"\n            position=\"absolute\"\n            left={0}\n            right={0}\n            top={0}\n            zIndex={2}\n            style={{\n              transform: 'translateY(-50%)',\n              pointerEvents: 'none',\n            }}\n          >\n            <PromotionBadgeComponent variant=\"tertiary\" size=\"base\">\n              {promotionBadgeLabel}\n            </PromotionBadgeComponent>\n          </FlexBox>\n        ) : null}\n        <Card\n          borderRadius=\"md\"\n          borderColor=\"border-primary\"\n          overflow=\"hidden\"\n          display=\"flex\"\n          flexDirection=\"column\"\n          isInteractive\n          p={0}\n          height=\"100%\"\n          width=\"100%\"\n        >\n          <CardAnchor\n            href={href}\n            onClick={onClick}\n            variant=\"interface\"\n            ref={ref}\n          >\n            <Box\n              width=\"100%\"\n              height={160}\n              position=\"relative\"\n              overflow=\"hidden\"\n              bg=\"navy-800\"\n            >\n              {instructorImageUrl && (\n                <StyledImg src={instructorImageUrl} alt={instructorName} />\n              )}\n            </Box>\n\n            <FlexBox\n              flexDirection=\"column\"\n              flex={1}\n              pt={12}\n              pb={4}\n              px={12}\n              overflow=\"hidden\"\n            >\n              <FlexBox\n                flexDirection=\"column\"\n                alignSelf=\"stretch\"\n                flex={1}\n                gap={4}\n              >\n                <Text\n                  as=\"h3\"\n                  fontSize={20}\n                  fontWeight=\"bold\"\n                  lineHeight=\"title\"\n                  color=\"text\"\n                  m={0}\n                  truncate=\"ellipsis\"\n                  truncateLines={3}\n                >\n                  {title}\n                </Text>\n\n                <FlexBox flexDirection=\"column\" gap={4} mt=\"auto\" minWidth={0}>\n                  <Text\n                    fontSize={14}\n                    fontWeight=\"bold\"\n                    lineHeight=\"base\"\n                    color=\"text\"\n                  >\n                    {instructorName}\n                  </Text>\n                  <Text\n                    fontSize={14}\n                    lineHeight=\"base\"\n                    color=\"text-secondary\"\n                    truncate=\"ellipsis\"\n                    truncateLines={1}\n                    width=\"100%\"\n                  >\n                    {instructorTitle}\n                  </Text>\n                </FlexBox>\n              </FlexBox>\n\n              <FlexBox flexDirection=\"column\" alignSelf=\"stretch\" mt={4}>\n                {nextCohortStartDate && (\n                  <>\n                    <Divider />\n                    <InfoRow icon={<CalendarIcon size={16} aria-hidden />}>\n                      Next cohort: Starts{' '}\n                      <Text as=\"span\" fontWeight=\"bold\">\n                        {nextCohortStartDate}\n                      </Text>\n                    </InfoRow>\n                  </>\n                )}\n                <Divider />\n                <InfoRow icon={<CertificateIcon size={16} aria-hidden />}>\n                  With{' '}\n                  <Text as=\"span\" fontWeight=\"bold\">\n                    Certificate\n                  </Text>\n                </InfoRow>\n              </FlexBox>\n            </FlexBox>\n          </CardAnchor>\n        </Card>\n      </Box>\n    );\n  }\n);\n"]} */");
21
21
  const StyledImg = /*#__PURE__*/_styled("img", {
22
- target: "e171y2mq1",
22
+ target: "e171y2mq3",
23
23
  label: "StyledImg"
24
24
  })(process.env.NODE_ENV === "production" ? {
25
25
  name: "q5j908",
26
26
  styles: "width:100%;height:100%;object-fit:cover;display:block"
27
27
  } : {
28
28
  name: "q5j908",
29
- styles: "width:100%;height:100%;object-fit:cover;display:block/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9JbnN0cnVjdG9yQ2FyZC9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBbUI0QiIsImZpbGUiOiIuLi8uLi9zcmMvSW5zdHJ1Y3RvckNhcmQvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQW5jaG9yLCBCb3gsIENhcmQsIEZsZXhCb3gsIFRleHQgfSBmcm9tICdAY29kZWNhZGVteS9nYW11dCc7XG5pbXBvcnQgeyBDYWxlbmRhckljb24sIENlcnRpZmljYXRlSWNvbiB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LWljb25zJztcbmltcG9ydCB7IENoZWNrZXJEZW5zZSB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXBhdHRlcm5zJztcbmltcG9ydCB7IGNzcyB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQgeyBmb3J3YXJkUmVmLCBQcm9wc1dpdGhSZWYgfSBmcm9tICdyZWFjdCc7XG5cbmNvbnN0IENhcmRBbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgZmxleERpcmVjdGlvbjogJ2NvbHVtbicsXG4gICAgaGVpZ2h0OiAnMTAwJScsXG4gICAgJyY6aG92ZXIsICY6Zm9jdXMsICY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICB0ZXh0RGVjb3JhdGlvbjogJ25vbmUnLFxuICAgIH0sXG4gIH0pXG4pO1xuXG5jb25zdCBTdHlsZWRJbWcgPSBzdHlsZWQuaW1nYFxuICB3aWR0aDogMTAwJTtcbiAgaGVpZ2h0OiAxMDAlO1xuICBvYmplY3QtZml0OiBjb3ZlcjtcbiAgZGlzcGxheTogYmxvY2s7XG5gO1xuXG5jb25zdCBEb3R0ZWRMaW5lID0gc3R5bGVkKENoZWNrZXJEZW5zZSkoXG4gIGNzcyh7XG4gICAgaGVpZ2h0OiAnMXB4JyxcbiAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgbWFyZ2luVG9wOiAnLTFweCcsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgRGl2aWRlcjogUmVhY3QuRkMgPSAoKSA9PiB7XG4gIHJldHVybiAoXG4gICAgLy8gdGhlIGhlaWdodCBhbmQgcG9zaXRpb24gcHJvcGVydGllcyBhcmUgbmVjZXNzYXJ5IGZvciBpT1MgZGV2aWNlc1xuICAgIDxCb3ggcHk9ezR9IGhlaWdodD17MH0gcG9zaXRpb249XCJyZWxhdGl2ZVwiIGFyaWEtaGlkZGVuPlxuICAgICAgPERvdHRlZExpbmUgLz5cbiAgICA8L0JveD5cbiAgKTtcbn07XG5cbnR5cGUgSW5mb1Jvd1Byb3BzID0ge1xuICBpY29uOiBSZWFjdC5SZWFjdE5vZGU7XG4gIGNoaWxkcmVuOiBSZWFjdC5SZWFjdE5vZGU7XG59O1xuXG5jb25zdCBJbmZvUm93OiBSZWFjdC5GQzxJbmZvUm93UHJvcHM+ID0gKHsgaWNvbiwgY2hpbGRyZW4gfSkgPT4gKFxuICA8RmxleEJveCBhbGlnbkl0ZW1zPVwiY2VudGVyXCIgZ2FwPXs4fT5cbiAgICB7aWNvbn1cbiAgICA8VGV4dCBmb250U2l6ZT17MTR9IGxpbmVIZWlnaHQ9XCJiYXNlXCIgY29sb3I9XCJ0ZXh0XCI+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9UZXh0PlxuICA8L0ZsZXhCb3g+XG4pO1xuXG5leHBvcnQgdHlwZSBJbnN0cnVjdG9yQ2FyZFByb3BzID0ge1xuICAvKiogVGhlIGJvb3RjYW1wIHRpdGxlICovXG4gIHRpdGxlOiBzdHJpbmc7XG4gIC8qKiBJbnN0cnVjdG9yJ3MgZnVsbCBuYW1lICovXG4gIGluc3RydWN0b3JOYW1lOiBzdHJpbmc7XG4gIC8qKiBJbnN0cnVjdG9yJ3MgcHJvZmVzc2lvbmFsIHRpdGxlICovXG4gIGluc3RydWN0b3JUaXRsZTogc3RyaW5nO1xuICAvKiogVVJMIGZvciB0aGUgaW5zdHJ1Y3RvcidzIHByb2ZpbGUgcGljIChpbmNsdWRlcyBiYWNrZ3JvdW5kIGdyYWRpZW50KSAqL1xuICBpbnN0cnVjdG9ySW1hZ2VVcmw/OiBzdHJpbmc7XG4gIC8qKiBGb3JtYXR0ZWQgc3RhcnQgZGF0ZSBmb3IgdGhlIG5leHQgY29ob3J0IChlLmcuIFwiRmViIDEwXCIpICovXG4gIG5leHRDb2hvcnRTdGFydERhdGU/OiBzdHJpbmc7XG4gIC8qKiBMaW5rIGRlc3RpbmF0aW9uICovXG4gIGhyZWY6IHN0cmluZztcbiAgLyoqIENsaWNrIGhhbmRsZXIgKi9cbiAgb25DbGljazogKCkgPT4gdm9pZDtcbn07XG5cbmV4cG9ydCBjb25zdCBJbnN0cnVjdG9yQ2FyZCA9IGZvcndhcmRSZWY8XG4gIEhUTUxBbmNob3JFbGVtZW50LFxuICBQcm9wc1dpdGhSZWY8SW5zdHJ1Y3RvckNhcmRQcm9wcz5cbj4oXG4gIChcbiAgICB7XG4gICAgICB0aXRsZSxcbiAgICAgIGluc3RydWN0b3JOYW1lLFxuICAgICAgaW5zdHJ1Y3RvclRpdGxlLFxuICAgICAgaW5zdHJ1Y3RvckltYWdlVXJsLFxuICAgICAgbmV4dENvaG9ydFN0YXJ0RGF0ZSxcbiAgICAgIGhyZWYsXG4gICAgICBvbkNsaWNrLFxuICAgIH0sXG4gICAgcmVmXG4gICkgPT4ge1xuICAgIHJldHVybiAoXG4gICAgICA8Q2FyZFxuICAgICAgICBib3JkZXJSYWRpdXM9XCJtZFwiXG4gICAgICAgIGJvcmRlckNvbG9yPVwiYm9yZGVyLXByaW1hcnlcIlxuICAgICAgICBvdmVyZmxvdz1cImhpZGRlblwiXG4gICAgICAgIGRpc3BsYXk9XCJmbGV4XCJcbiAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgIGlzSW50ZXJhY3RpdmVcbiAgICAgICAgcD17MH1cbiAgICAgICAgaGVpZ2h0PVwiMTAwJVwiXG4gICAgICAgIHdpZHRoPVwiMTAwJVwiXG4gICAgICA+XG4gICAgICAgIDxDYXJkQW5jaG9yIGhyZWY9e2hyZWZ9IG9uQ2xpY2s9e29uQ2xpY2t9IHZhcmlhbnQ9XCJpbnRlcmZhY2VcIiByZWY9e3JlZn0+XG4gICAgICAgICAgPEJveFxuICAgICAgICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgICAgICAgIGhlaWdodD17MTYwfVxuICAgICAgICAgICAgcG9zaXRpb249XCJyZWxhdGl2ZVwiXG4gICAgICAgICAgICBvdmVyZmxvdz1cImhpZGRlblwiXG4gICAgICAgICAgICBiZz1cIm5hdnktODAwXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7aW5zdHJ1Y3RvckltYWdlVXJsICYmIChcbiAgICAgICAgICAgICAgPFN0eWxlZEltZyBzcmM9e2luc3RydWN0b3JJbWFnZVVybH0gYWx0PXtpbnN0cnVjdG9yTmFtZX0gLz5cbiAgICAgICAgICAgICl9XG4gICAgICAgICAgPC9Cb3g+XG5cbiAgICAgICAgICA8RmxleEJveFxuICAgICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgICBmbGV4PXsxfVxuICAgICAgICAgICAgcHQ9ezEyfVxuICAgICAgICAgICAgcGI9ezR9XG4gICAgICAgICAgICBweD17MTJ9XG4gICAgICAgICAgICBvdmVyZmxvdz1cImhpZGRlblwiXG4gICAgICAgICAgPlxuICAgICAgICAgICAgPEZsZXhCb3hcbiAgICAgICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgICAgIGFsaWduU2VsZj1cInN0cmV0Y2hcIlxuICAgICAgICAgICAgICBmbGV4PXsxfVxuICAgICAgICAgICAgICBnYXA9ezR9XG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgYXM9XCJoM1wiXG4gICAgICAgICAgICAgICAgZm9udFNpemU9ezIwfVxuICAgICAgICAgICAgICAgIGZvbnRXZWlnaHQ9XCJib2xkXCJcbiAgICAgICAgICAgICAgICBsaW5lSGVpZ2h0PVwidGl0bGVcIlxuICAgICAgICAgICAgICAgIGNvbG9yPVwidGV4dFwiXG4gICAgICAgICAgICAgICAgbT17MH1cbiAgICAgICAgICAgICAgICB0cnVuY2F0ZT1cImVsbGlwc2lzXCJcbiAgICAgICAgICAgICAgICB0cnVuY2F0ZUxpbmVzPXszfVxuICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAge3RpdGxlfVxuICAgICAgICAgICAgICA8L1RleHQ+XG5cbiAgICAgICAgICAgICAgPEZsZXhCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGdhcD17NH0gbXQ9XCJhdXRvXCIgbWluV2lkdGg9ezB9PlxuICAgICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgICBmb250U2l6ZT17MTR9XG4gICAgICAgICAgICAgICAgICBmb250V2VpZ2h0PVwiYm9sZFwiXG4gICAgICAgICAgICAgICAgICBsaW5lSGVpZ2h0PVwiYmFzZVwiXG4gICAgICAgICAgICAgICAgICBjb2xvcj1cInRleHRcIlxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIHtpbnN0cnVjdG9yTmFtZX1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgIGZvbnRTaXplPXsxNH1cbiAgICAgICAgICAgICAgICAgIGxpbmVIZWlnaHQ9XCJiYXNlXCJcbiAgICAgICAgICAgICAgICAgIGNvbG9yPVwidGV4dC1zZWNvbmRhcnlcIlxuICAgICAgICAgICAgICAgICAgdHJ1bmNhdGU9XCJlbGxpcHNpc1wiXG4gICAgICAgICAgICAgICAgICB0cnVuY2F0ZUxpbmVzPXsxfVxuICAgICAgICAgICAgICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICB7aW5zdHJ1Y3RvclRpdGxlfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPC9GbGV4Qm94PlxuICAgICAgICAgICAgPC9GbGV4Qm94PlxuXG4gICAgICAgICAgICA8RmxleEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgYWxpZ25TZWxmPVwic3RyZXRjaFwiIG10PXs0fT5cbiAgICAgICAgICAgICAge25leHRDb2hvcnRTdGFydERhdGUgJiYgKFxuICAgICAgICAgICAgICAgIDw+XG4gICAgICAgICAgICAgICAgICA8RGl2aWRlciAvPlxuICAgICAgICAgICAgICAgICAgPEluZm9Sb3cgaWNvbj17PENhbGVuZGFySWNvbiBzaXplPXsxNn0gYXJpYS1oaWRkZW4gLz59PlxuICAgICAgICAgICAgICAgICAgICBOZXh0IGNvaG9ydDogU3RhcnRzeycgJ31cbiAgICAgICAgICAgICAgICAgICAgPFRleHQgYXM9XCJzcGFuXCIgZm9udFdlaWdodD1cImJvbGRcIj5cbiAgICAgICAgICAgICAgICAgICAgICB7bmV4dENvaG9ydFN0YXJ0RGF0ZX1cbiAgICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgICAgPC9JbmZvUm93PlxuICAgICAgICAgICAgICAgIDwvPlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICA8RGl2aWRlciAvPlxuICAgICAgICAgICAgICA8SW5mb1JvdyBpY29uPXs8Q2VydGlmaWNhdGVJY29uIHNpemU9ezE2fSBhcmlhLWhpZGRlbiAvPn0+XG4gICAgICAgICAgICAgICAgV2l0aHsnICd9XG4gICAgICAgICAgICAgICAgPFRleHQgYXM9XCJzcGFuXCIgZm9udFdlaWdodD1cImJvbGRcIj5cbiAgICAgICAgICAgICAgICAgIENlcnRpZmljYXRlXG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L0luZm9Sb3c+XG4gICAgICAgICAgICA8L0ZsZXhCb3g+XG4gICAgICAgICAgPC9GbGV4Qm94PlxuICAgICAgICA8L0NhcmRBbmNob3I+XG4gICAgICA8L0NhcmQ+XG4gICAgKTtcbiAgfVxuKTtcbiJdfQ== */",
29
+ styles: "width:100%;height:100%;object-fit:cover;display:block/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/InstructorCard/index.tsx"],"names":[],"mappings":"AAmB4B","file":"../../src/InstructorCard/index.tsx","sourcesContent":["import { Anchor, Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CalendarIcon, CertificateIcon } from '@codecademy/gamut-icons';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, PropsWithRef } from 'react';\n\nconst CardAnchor = styled(Anchor)(\n  css({\n    display: 'flex',\n    flexDirection: 'column',\n    height: '100%',\n    '&:hover, &:focus, &:focus-visible': {\n      color: 'text',\n      textDecoration: 'none',\n    },\n  })\n);\n\nconst StyledImg = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  display: block;\n`;\n\nconst LastChanceBadge = styled(Badge)`\n  color: ${theme.colors.red};\n  border-color: ${theme.colors.red};\n  background-color: ${theme.colors['red-0']};\n  border: 1px solid ${theme.colors['red-500']};\n`;\n\nconst PromotionBadge = styled(Badge)`\n  color: ${theme.colors.black};\n  border-color: ${theme.colors.yellow};\n  background-color: ${theme.colors['yellow-0']};\n  border: 1px solid ${theme.colors['yellow-500']};\n`;\n\nconst DottedLine = styled(CheckerDense)(\n  css({\n    height: '1px',\n    display: 'flex',\n    position: 'absolute',\n    marginTop: '-1px',\n  })\n);\n\nexport const Divider: React.FC = () => {\n  return (\n    // the height and position properties are necessary for iOS devices\n    <Box py={4} height={0} position=\"relative\" aria-hidden>\n      <DottedLine />\n    </Box>\n  );\n};\n\ntype InfoRowProps = {\n  icon: React.ReactNode;\n  children: React.ReactNode;\n};\n\nconst InfoRow: React.FC<InfoRowProps> = ({ icon, children }) => (\n  <FlexBox alignItems=\"center\" gap={8}>\n    {icon}\n    <Text fontSize={14} lineHeight=\"base\" color=\"text\">\n      {children}\n    </Text>\n  </FlexBox>\n);\n\nexport type InstructorCardProps = {\n  /** The bootcamp title */\n  title: string;\n  /** Instructor's full name */\n  instructorName: string;\n  /** Instructor's professional title */\n  instructorTitle: string;\n  /** URL for the instructor's profile pic (includes background gradient) */\n  instructorImageUrl?: string;\n  /** Formatted start date for the next cohort (e.g. \"Feb 10\") */\n  nextCohortStartDate?: string;\n  /**\n   * Promotional label from the latest cohort (e.g. \"Last chance - 20% off\").\n   * Rendered centered on the top edge of the card, overlapping the border.\n   */\n  promotionBadgeLabel?: string;\n  /** Link destination */\n  href: string;\n  /** Click handler */\n  onClick: () => void;\n};\n\nexport const InstructorCard = forwardRef<\n  HTMLAnchorElement,\n  PropsWithRef<InstructorCardProps>\n>(\n  (\n    {\n      title,\n      instructorName,\n      instructorTitle,\n      instructorImageUrl,\n      nextCohortStartDate,\n      promotionBadgeLabel,\n      href,\n      onClick,\n    },\n    ref\n  ) => {\n    const isLastChance = promotionBadgeLabel\n      ?.toLowerCase()\n      .includes('last chance');\n\n    const PromotionBadgeComponent = isLastChance\n      ? LastChanceBadge\n      : PromotionBadge;\n\n    return (\n      <Box position=\"relative\" width=\"100%\" height=\"100%\">\n        {promotionBadgeLabel ? (\n          <FlexBox\n            justifyContent=\"center\"\n            position=\"absolute\"\n            left={0}\n            right={0}\n            top={0}\n            zIndex={2}\n            style={{\n              transform: 'translateY(-50%)',\n              pointerEvents: 'none',\n            }}\n          >\n            <PromotionBadgeComponent variant=\"tertiary\" size=\"base\">\n              {promotionBadgeLabel}\n            </PromotionBadgeComponent>\n          </FlexBox>\n        ) : null}\n        <Card\n          borderRadius=\"md\"\n          borderColor=\"border-primary\"\n          overflow=\"hidden\"\n          display=\"flex\"\n          flexDirection=\"column\"\n          isInteractive\n          p={0}\n          height=\"100%\"\n          width=\"100%\"\n        >\n          <CardAnchor\n            href={href}\n            onClick={onClick}\n            variant=\"interface\"\n            ref={ref}\n          >\n            <Box\n              width=\"100%\"\n              height={160}\n              position=\"relative\"\n              overflow=\"hidden\"\n              bg=\"navy-800\"\n            >\n              {instructorImageUrl && (\n                <StyledImg src={instructorImageUrl} alt={instructorName} />\n              )}\n            </Box>\n\n            <FlexBox\n              flexDirection=\"column\"\n              flex={1}\n              pt={12}\n              pb={4}\n              px={12}\n              overflow=\"hidden\"\n            >\n              <FlexBox\n                flexDirection=\"column\"\n                alignSelf=\"stretch\"\n                flex={1}\n                gap={4}\n              >\n                <Text\n                  as=\"h3\"\n                  fontSize={20}\n                  fontWeight=\"bold\"\n                  lineHeight=\"title\"\n                  color=\"text\"\n                  m={0}\n                  truncate=\"ellipsis\"\n                  truncateLines={3}\n                >\n                  {title}\n                </Text>\n\n                <FlexBox flexDirection=\"column\" gap={4} mt=\"auto\" minWidth={0}>\n                  <Text\n                    fontSize={14}\n                    fontWeight=\"bold\"\n                    lineHeight=\"base\"\n                    color=\"text\"\n                  >\n                    {instructorName}\n                  </Text>\n                  <Text\n                    fontSize={14}\n                    lineHeight=\"base\"\n                    color=\"text-secondary\"\n                    truncate=\"ellipsis\"\n                    truncateLines={1}\n                    width=\"100%\"\n                  >\n                    {instructorTitle}\n                  </Text>\n                </FlexBox>\n              </FlexBox>\n\n              <FlexBox flexDirection=\"column\" alignSelf=\"stretch\" mt={4}>\n                {nextCohortStartDate && (\n                  <>\n                    <Divider />\n                    <InfoRow icon={<CalendarIcon size={16} aria-hidden />}>\n                      Next cohort: Starts{' '}\n                      <Text as=\"span\" fontWeight=\"bold\">\n                        {nextCohortStartDate}\n                      </Text>\n                    </InfoRow>\n                  </>\n                )}\n                <Divider />\n                <InfoRow icon={<CertificateIcon size={16} aria-hidden />}>\n                  With{' '}\n                  <Text as=\"span\" fontWeight=\"bold\">\n                    Certificate\n                  </Text>\n                </InfoRow>\n              </FlexBox>\n            </FlexBox>\n          </CardAnchor>\n        </Card>\n      </Box>\n    );\n  }\n);\n"]} */",
30
30
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
31
31
  });
32
+ const LastChanceBadge = /*#__PURE__*/_styled(Badge, {
33
+ target: "e171y2mq2",
34
+ label: "LastChanceBadge"
35
+ })("color:", theme.colors.red, ";border-color:", theme.colors.red, ";background-color:", theme.colors['red-0'], ";border:1px solid ", theme.colors['red-500'], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/InstructorCard/index.tsx"],"names":[],"mappings":"AA0BqC","file":"../../src/InstructorCard/index.tsx","sourcesContent":["import { Anchor, Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CalendarIcon, CertificateIcon } from '@codecademy/gamut-icons';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, PropsWithRef } from 'react';\n\nconst CardAnchor = styled(Anchor)(\n  css({\n    display: 'flex',\n    flexDirection: 'column',\n    height: '100%',\n    '&:hover, &:focus, &:focus-visible': {\n      color: 'text',\n      textDecoration: 'none',\n    },\n  })\n);\n\nconst StyledImg = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  display: block;\n`;\n\nconst LastChanceBadge = styled(Badge)`\n  color: ${theme.colors.red};\n  border-color: ${theme.colors.red};\n  background-color: ${theme.colors['red-0']};\n  border: 1px solid ${theme.colors['red-500']};\n`;\n\nconst PromotionBadge = styled(Badge)`\n  color: ${theme.colors.black};\n  border-color: ${theme.colors.yellow};\n  background-color: ${theme.colors['yellow-0']};\n  border: 1px solid ${theme.colors['yellow-500']};\n`;\n\nconst DottedLine = styled(CheckerDense)(\n  css({\n    height: '1px',\n    display: 'flex',\n    position: 'absolute',\n    marginTop: '-1px',\n  })\n);\n\nexport const Divider: React.FC = () => {\n  return (\n    // the height and position properties are necessary for iOS devices\n    <Box py={4} height={0} position=\"relative\" aria-hidden>\n      <DottedLine />\n    </Box>\n  );\n};\n\ntype InfoRowProps = {\n  icon: React.ReactNode;\n  children: React.ReactNode;\n};\n\nconst InfoRow: React.FC<InfoRowProps> = ({ icon, children }) => (\n  <FlexBox alignItems=\"center\" gap={8}>\n    {icon}\n    <Text fontSize={14} lineHeight=\"base\" color=\"text\">\n      {children}\n    </Text>\n  </FlexBox>\n);\n\nexport type InstructorCardProps = {\n  /** The bootcamp title */\n  title: string;\n  /** Instructor's full name */\n  instructorName: string;\n  /** Instructor's professional title */\n  instructorTitle: string;\n  /** URL for the instructor's profile pic (includes background gradient) */\n  instructorImageUrl?: string;\n  /** Formatted start date for the next cohort (e.g. \"Feb 10\") */\n  nextCohortStartDate?: string;\n  /**\n   * Promotional label from the latest cohort (e.g. \"Last chance - 20% off\").\n   * Rendered centered on the top edge of the card, overlapping the border.\n   */\n  promotionBadgeLabel?: string;\n  /** Link destination */\n  href: string;\n  /** Click handler */\n  onClick: () => void;\n};\n\nexport const InstructorCard = forwardRef<\n  HTMLAnchorElement,\n  PropsWithRef<InstructorCardProps>\n>(\n  (\n    {\n      title,\n      instructorName,\n      instructorTitle,\n      instructorImageUrl,\n      nextCohortStartDate,\n      promotionBadgeLabel,\n      href,\n      onClick,\n    },\n    ref\n  ) => {\n    const isLastChance = promotionBadgeLabel\n      ?.toLowerCase()\n      .includes('last chance');\n\n    const PromotionBadgeComponent = isLastChance\n      ? LastChanceBadge\n      : PromotionBadge;\n\n    return (\n      <Box position=\"relative\" width=\"100%\" height=\"100%\">\n        {promotionBadgeLabel ? (\n          <FlexBox\n            justifyContent=\"center\"\n            position=\"absolute\"\n            left={0}\n            right={0}\n            top={0}\n            zIndex={2}\n            style={{\n              transform: 'translateY(-50%)',\n              pointerEvents: 'none',\n            }}\n          >\n            <PromotionBadgeComponent variant=\"tertiary\" size=\"base\">\n              {promotionBadgeLabel}\n            </PromotionBadgeComponent>\n          </FlexBox>\n        ) : null}\n        <Card\n          borderRadius=\"md\"\n          borderColor=\"border-primary\"\n          overflow=\"hidden\"\n          display=\"flex\"\n          flexDirection=\"column\"\n          isInteractive\n          p={0}\n          height=\"100%\"\n          width=\"100%\"\n        >\n          <CardAnchor\n            href={href}\n            onClick={onClick}\n            variant=\"interface\"\n            ref={ref}\n          >\n            <Box\n              width=\"100%\"\n              height={160}\n              position=\"relative\"\n              overflow=\"hidden\"\n              bg=\"navy-800\"\n            >\n              {instructorImageUrl && (\n                <StyledImg src={instructorImageUrl} alt={instructorName} />\n              )}\n            </Box>\n\n            <FlexBox\n              flexDirection=\"column\"\n              flex={1}\n              pt={12}\n              pb={4}\n              px={12}\n              overflow=\"hidden\"\n            >\n              <FlexBox\n                flexDirection=\"column\"\n                alignSelf=\"stretch\"\n                flex={1}\n                gap={4}\n              >\n                <Text\n                  as=\"h3\"\n                  fontSize={20}\n                  fontWeight=\"bold\"\n                  lineHeight=\"title\"\n                  color=\"text\"\n                  m={0}\n                  truncate=\"ellipsis\"\n                  truncateLines={3}\n                >\n                  {title}\n                </Text>\n\n                <FlexBox flexDirection=\"column\" gap={4} mt=\"auto\" minWidth={0}>\n                  <Text\n                    fontSize={14}\n                    fontWeight=\"bold\"\n                    lineHeight=\"base\"\n                    color=\"text\"\n                  >\n                    {instructorName}\n                  </Text>\n                  <Text\n                    fontSize={14}\n                    lineHeight=\"base\"\n                    color=\"text-secondary\"\n                    truncate=\"ellipsis\"\n                    truncateLines={1}\n                    width=\"100%\"\n                  >\n                    {instructorTitle}\n                  </Text>\n                </FlexBox>\n              </FlexBox>\n\n              <FlexBox flexDirection=\"column\" alignSelf=\"stretch\" mt={4}>\n                {nextCohortStartDate && (\n                  <>\n                    <Divider />\n                    <InfoRow icon={<CalendarIcon size={16} aria-hidden />}>\n                      Next cohort: Starts{' '}\n                      <Text as=\"span\" fontWeight=\"bold\">\n                        {nextCohortStartDate}\n                      </Text>\n                    </InfoRow>\n                  </>\n                )}\n                <Divider />\n                <InfoRow icon={<CertificateIcon size={16} aria-hidden />}>\n                  With{' '}\n                  <Text as=\"span\" fontWeight=\"bold\">\n                    Certificate\n                  </Text>\n                </InfoRow>\n              </FlexBox>\n            </FlexBox>\n          </CardAnchor>\n        </Card>\n      </Box>\n    );\n  }\n);\n"]} */"));
36
+ const PromotionBadge = /*#__PURE__*/_styled(Badge, {
37
+ target: "e171y2mq1",
38
+ label: "PromotionBadge"
39
+ })("color:", theme.colors.black, ";border-color:", theme.colors.yellow, ";background-color:", theme.colors['yellow-0'], ";border:1px solid ", theme.colors['yellow-500'], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/InstructorCard/index.tsx"],"names":[],"mappings":"AAiCoC","file":"../../src/InstructorCard/index.tsx","sourcesContent":["import { Anchor, Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CalendarIcon, CertificateIcon } from '@codecademy/gamut-icons';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, PropsWithRef } from 'react';\n\nconst CardAnchor = styled(Anchor)(\n  css({\n    display: 'flex',\n    flexDirection: 'column',\n    height: '100%',\n    '&:hover, &:focus, &:focus-visible': {\n      color: 'text',\n      textDecoration: 'none',\n    },\n  })\n);\n\nconst StyledImg = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  display: block;\n`;\n\nconst LastChanceBadge = styled(Badge)`\n  color: ${theme.colors.red};\n  border-color: ${theme.colors.red};\n  background-color: ${theme.colors['red-0']};\n  border: 1px solid ${theme.colors['red-500']};\n`;\n\nconst PromotionBadge = styled(Badge)`\n  color: ${theme.colors.black};\n  border-color: ${theme.colors.yellow};\n  background-color: ${theme.colors['yellow-0']};\n  border: 1px solid ${theme.colors['yellow-500']};\n`;\n\nconst DottedLine = styled(CheckerDense)(\n  css({\n    height: '1px',\n    display: 'flex',\n    position: 'absolute',\n    marginTop: '-1px',\n  })\n);\n\nexport const Divider: React.FC = () => {\n  return (\n    // the height and position properties are necessary for iOS devices\n    <Box py={4} height={0} position=\"relative\" aria-hidden>\n      <DottedLine />\n    </Box>\n  );\n};\n\ntype InfoRowProps = {\n  icon: React.ReactNode;\n  children: React.ReactNode;\n};\n\nconst InfoRow: React.FC<InfoRowProps> = ({ icon, children }) => (\n  <FlexBox alignItems=\"center\" gap={8}>\n    {icon}\n    <Text fontSize={14} lineHeight=\"base\" color=\"text\">\n      {children}\n    </Text>\n  </FlexBox>\n);\n\nexport type InstructorCardProps = {\n  /** The bootcamp title */\n  title: string;\n  /** Instructor's full name */\n  instructorName: string;\n  /** Instructor's professional title */\n  instructorTitle: string;\n  /** URL for the instructor's profile pic (includes background gradient) */\n  instructorImageUrl?: string;\n  /** Formatted start date for the next cohort (e.g. \"Feb 10\") */\n  nextCohortStartDate?: string;\n  /**\n   * Promotional label from the latest cohort (e.g. \"Last chance - 20% off\").\n   * Rendered centered on the top edge of the card, overlapping the border.\n   */\n  promotionBadgeLabel?: string;\n  /** Link destination */\n  href: string;\n  /** Click handler */\n  onClick: () => void;\n};\n\nexport const InstructorCard = forwardRef<\n  HTMLAnchorElement,\n  PropsWithRef<InstructorCardProps>\n>(\n  (\n    {\n      title,\n      instructorName,\n      instructorTitle,\n      instructorImageUrl,\n      nextCohortStartDate,\n      promotionBadgeLabel,\n      href,\n      onClick,\n    },\n    ref\n  ) => {\n    const isLastChance = promotionBadgeLabel\n      ?.toLowerCase()\n      .includes('last chance');\n\n    const PromotionBadgeComponent = isLastChance\n      ? LastChanceBadge\n      : PromotionBadge;\n\n    return (\n      <Box position=\"relative\" width=\"100%\" height=\"100%\">\n        {promotionBadgeLabel ? (\n          <FlexBox\n            justifyContent=\"center\"\n            position=\"absolute\"\n            left={0}\n            right={0}\n            top={0}\n            zIndex={2}\n            style={{\n              transform: 'translateY(-50%)',\n              pointerEvents: 'none',\n            }}\n          >\n            <PromotionBadgeComponent variant=\"tertiary\" size=\"base\">\n              {promotionBadgeLabel}\n            </PromotionBadgeComponent>\n          </FlexBox>\n        ) : null}\n        <Card\n          borderRadius=\"md\"\n          borderColor=\"border-primary\"\n          overflow=\"hidden\"\n          display=\"flex\"\n          flexDirection=\"column\"\n          isInteractive\n          p={0}\n          height=\"100%\"\n          width=\"100%\"\n        >\n          <CardAnchor\n            href={href}\n            onClick={onClick}\n            variant=\"interface\"\n            ref={ref}\n          >\n            <Box\n              width=\"100%\"\n              height={160}\n              position=\"relative\"\n              overflow=\"hidden\"\n              bg=\"navy-800\"\n            >\n              {instructorImageUrl && (\n                <StyledImg src={instructorImageUrl} alt={instructorName} />\n              )}\n            </Box>\n\n            <FlexBox\n              flexDirection=\"column\"\n              flex={1}\n              pt={12}\n              pb={4}\n              px={12}\n              overflow=\"hidden\"\n            >\n              <FlexBox\n                flexDirection=\"column\"\n                alignSelf=\"stretch\"\n                flex={1}\n                gap={4}\n              >\n                <Text\n                  as=\"h3\"\n                  fontSize={20}\n                  fontWeight=\"bold\"\n                  lineHeight=\"title\"\n                  color=\"text\"\n                  m={0}\n                  truncate=\"ellipsis\"\n                  truncateLines={3}\n                >\n                  {title}\n                </Text>\n\n                <FlexBox flexDirection=\"column\" gap={4} mt=\"auto\" minWidth={0}>\n                  <Text\n                    fontSize={14}\n                    fontWeight=\"bold\"\n                    lineHeight=\"base\"\n                    color=\"text\"\n                  >\n                    {instructorName}\n                  </Text>\n                  <Text\n                    fontSize={14}\n                    lineHeight=\"base\"\n                    color=\"text-secondary\"\n                    truncate=\"ellipsis\"\n                    truncateLines={1}\n                    width=\"100%\"\n                  >\n                    {instructorTitle}\n                  </Text>\n                </FlexBox>\n              </FlexBox>\n\n              <FlexBox flexDirection=\"column\" alignSelf=\"stretch\" mt={4}>\n                {nextCohortStartDate && (\n                  <>\n                    <Divider />\n                    <InfoRow icon={<CalendarIcon size={16} aria-hidden />}>\n                      Next cohort: Starts{' '}\n                      <Text as=\"span\" fontWeight=\"bold\">\n                        {nextCohortStartDate}\n                      </Text>\n                    </InfoRow>\n                  </>\n                )}\n                <Divider />\n                <InfoRow icon={<CertificateIcon size={16} aria-hidden />}>\n                  With{' '}\n                  <Text as=\"span\" fontWeight=\"bold\">\n                    Certificate\n                  </Text>\n                </InfoRow>\n              </FlexBox>\n            </FlexBox>\n          </CardAnchor>\n        </Card>\n      </Box>\n    );\n  }\n);\n"]} */"));
32
40
  const DottedLine = /*#__PURE__*/_styled(CheckerDense, {
33
41
  target: "e171y2mq0",
34
42
  label: "DottedLine"
@@ -37,7 +45,7 @@ const DottedLine = /*#__PURE__*/_styled(CheckerDense, {
37
45
  display: 'flex',
38
46
  position: 'absolute',
39
47
  marginTop: '-1px'
40
- }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9JbnN0cnVjdG9yQ2FyZC9pbmRleC50c3giXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBMEJtQiIsImZpbGUiOiIuLi8uLi9zcmMvSW5zdHJ1Y3RvckNhcmQvaW5kZXgudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQW5jaG9yLCBCb3gsIENhcmQsIEZsZXhCb3gsIFRleHQgfSBmcm9tICdAY29kZWNhZGVteS9nYW11dCc7XG5pbXBvcnQgeyBDYWxlbmRhckljb24sIENlcnRpZmljYXRlSWNvbiB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LWljb25zJztcbmltcG9ydCB7IENoZWNrZXJEZW5zZSB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXBhdHRlcm5zJztcbmltcG9ydCB7IGNzcyB9IGZyb20gJ0Bjb2RlY2FkZW15L2dhbXV0LXN0eWxlcyc7XG5pbXBvcnQgc3R5bGVkIGZyb20gJ0BlbW90aW9uL3N0eWxlZCc7XG5pbXBvcnQgeyBmb3J3YXJkUmVmLCBQcm9wc1dpdGhSZWYgfSBmcm9tICdyZWFjdCc7XG5cbmNvbnN0IENhcmRBbmNob3IgPSBzdHlsZWQoQW5jaG9yKShcbiAgY3NzKHtcbiAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgZmxleERpcmVjdGlvbjogJ2NvbHVtbicsXG4gICAgaGVpZ2h0OiAnMTAwJScsXG4gICAgJyY6aG92ZXIsICY6Zm9jdXMsICY6Zm9jdXMtdmlzaWJsZSc6IHtcbiAgICAgIGNvbG9yOiAndGV4dCcsXG4gICAgICB0ZXh0RGVjb3JhdGlvbjogJ25vbmUnLFxuICAgIH0sXG4gIH0pXG4pO1xuXG5jb25zdCBTdHlsZWRJbWcgPSBzdHlsZWQuaW1nYFxuICB3aWR0aDogMTAwJTtcbiAgaGVpZ2h0OiAxMDAlO1xuICBvYmplY3QtZml0OiBjb3ZlcjtcbiAgZGlzcGxheTogYmxvY2s7XG5gO1xuXG5jb25zdCBEb3R0ZWRMaW5lID0gc3R5bGVkKENoZWNrZXJEZW5zZSkoXG4gIGNzcyh7XG4gICAgaGVpZ2h0OiAnMXB4JyxcbiAgICBkaXNwbGF5OiAnZmxleCcsXG4gICAgcG9zaXRpb246ICdhYnNvbHV0ZScsXG4gICAgbWFyZ2luVG9wOiAnLTFweCcsXG4gIH0pXG4pO1xuXG5leHBvcnQgY29uc3QgRGl2aWRlcjogUmVhY3QuRkMgPSAoKSA9PiB7XG4gIHJldHVybiAoXG4gICAgLy8gdGhlIGhlaWdodCBhbmQgcG9zaXRpb24gcHJvcGVydGllcyBhcmUgbmVjZXNzYXJ5IGZvciBpT1MgZGV2aWNlc1xuICAgIDxCb3ggcHk9ezR9IGhlaWdodD17MH0gcG9zaXRpb249XCJyZWxhdGl2ZVwiIGFyaWEtaGlkZGVuPlxuICAgICAgPERvdHRlZExpbmUgLz5cbiAgICA8L0JveD5cbiAgKTtcbn07XG5cbnR5cGUgSW5mb1Jvd1Byb3BzID0ge1xuICBpY29uOiBSZWFjdC5SZWFjdE5vZGU7XG4gIGNoaWxkcmVuOiBSZWFjdC5SZWFjdE5vZGU7XG59O1xuXG5jb25zdCBJbmZvUm93OiBSZWFjdC5GQzxJbmZvUm93UHJvcHM+ID0gKHsgaWNvbiwgY2hpbGRyZW4gfSkgPT4gKFxuICA8RmxleEJveCBhbGlnbkl0ZW1zPVwiY2VudGVyXCIgZ2FwPXs4fT5cbiAgICB7aWNvbn1cbiAgICA8VGV4dCBmb250U2l6ZT17MTR9IGxpbmVIZWlnaHQ9XCJiYXNlXCIgY29sb3I9XCJ0ZXh0XCI+XG4gICAgICB7Y2hpbGRyZW59XG4gICAgPC9UZXh0PlxuICA8L0ZsZXhCb3g+XG4pO1xuXG5leHBvcnQgdHlwZSBJbnN0cnVjdG9yQ2FyZFByb3BzID0ge1xuICAvKiogVGhlIGJvb3RjYW1wIHRpdGxlICovXG4gIHRpdGxlOiBzdHJpbmc7XG4gIC8qKiBJbnN0cnVjdG9yJ3MgZnVsbCBuYW1lICovXG4gIGluc3RydWN0b3JOYW1lOiBzdHJpbmc7XG4gIC8qKiBJbnN0cnVjdG9yJ3MgcHJvZmVzc2lvbmFsIHRpdGxlICovXG4gIGluc3RydWN0b3JUaXRsZTogc3RyaW5nO1xuICAvKiogVVJMIGZvciB0aGUgaW5zdHJ1Y3RvcidzIHByb2ZpbGUgcGljIChpbmNsdWRlcyBiYWNrZ3JvdW5kIGdyYWRpZW50KSAqL1xuICBpbnN0cnVjdG9ySW1hZ2VVcmw/OiBzdHJpbmc7XG4gIC8qKiBGb3JtYXR0ZWQgc3RhcnQgZGF0ZSBmb3IgdGhlIG5leHQgY29ob3J0IChlLmcuIFwiRmViIDEwXCIpICovXG4gIG5leHRDb2hvcnRTdGFydERhdGU/OiBzdHJpbmc7XG4gIC8qKiBMaW5rIGRlc3RpbmF0aW9uICovXG4gIGhyZWY6IHN0cmluZztcbiAgLyoqIENsaWNrIGhhbmRsZXIgKi9cbiAgb25DbGljazogKCkgPT4gdm9pZDtcbn07XG5cbmV4cG9ydCBjb25zdCBJbnN0cnVjdG9yQ2FyZCA9IGZvcndhcmRSZWY8XG4gIEhUTUxBbmNob3JFbGVtZW50LFxuICBQcm9wc1dpdGhSZWY8SW5zdHJ1Y3RvckNhcmRQcm9wcz5cbj4oXG4gIChcbiAgICB7XG4gICAgICB0aXRsZSxcbiAgICAgIGluc3RydWN0b3JOYW1lLFxuICAgICAgaW5zdHJ1Y3RvclRpdGxlLFxuICAgICAgaW5zdHJ1Y3RvckltYWdlVXJsLFxuICAgICAgbmV4dENvaG9ydFN0YXJ0RGF0ZSxcbiAgICAgIGhyZWYsXG4gICAgICBvbkNsaWNrLFxuICAgIH0sXG4gICAgcmVmXG4gICkgPT4ge1xuICAgIHJldHVybiAoXG4gICAgICA8Q2FyZFxuICAgICAgICBib3JkZXJSYWRpdXM9XCJtZFwiXG4gICAgICAgIGJvcmRlckNvbG9yPVwiYm9yZGVyLXByaW1hcnlcIlxuICAgICAgICBvdmVyZmxvdz1cImhpZGRlblwiXG4gICAgICAgIGRpc3BsYXk9XCJmbGV4XCJcbiAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgIGlzSW50ZXJhY3RpdmVcbiAgICAgICAgcD17MH1cbiAgICAgICAgaGVpZ2h0PVwiMTAwJVwiXG4gICAgICAgIHdpZHRoPVwiMTAwJVwiXG4gICAgICA+XG4gICAgICAgIDxDYXJkQW5jaG9yIGhyZWY9e2hyZWZ9IG9uQ2xpY2s9e29uQ2xpY2t9IHZhcmlhbnQ9XCJpbnRlcmZhY2VcIiByZWY9e3JlZn0+XG4gICAgICAgICAgPEJveFxuICAgICAgICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgICAgICAgIGhlaWdodD17MTYwfVxuICAgICAgICAgICAgcG9zaXRpb249XCJyZWxhdGl2ZVwiXG4gICAgICAgICAgICBvdmVyZmxvdz1cImhpZGRlblwiXG4gICAgICAgICAgICBiZz1cIm5hdnktODAwXCJcbiAgICAgICAgICA+XG4gICAgICAgICAgICB7aW5zdHJ1Y3RvckltYWdlVXJsICYmIChcbiAgICAgICAgICAgICAgPFN0eWxlZEltZyBzcmM9e2luc3RydWN0b3JJbWFnZVVybH0gYWx0PXtpbnN0cnVjdG9yTmFtZX0gLz5cbiAgICAgICAgICAgICl9XG4gICAgICAgICAgPC9Cb3g+XG5cbiAgICAgICAgICA8RmxleEJveFxuICAgICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgICBmbGV4PXsxfVxuICAgICAgICAgICAgcHQ9ezEyfVxuICAgICAgICAgICAgcGI9ezR9XG4gICAgICAgICAgICBweD17MTJ9XG4gICAgICAgICAgICBvdmVyZmxvdz1cImhpZGRlblwiXG4gICAgICAgICAgPlxuICAgICAgICAgICAgPEZsZXhCb3hcbiAgICAgICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgICAgIGFsaWduU2VsZj1cInN0cmV0Y2hcIlxuICAgICAgICAgICAgICBmbGV4PXsxfVxuICAgICAgICAgICAgICBnYXA9ezR9XG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgYXM9XCJoM1wiXG4gICAgICAgICAgICAgICAgZm9udFNpemU9ezIwfVxuICAgICAgICAgICAgICAgIGZvbnRXZWlnaHQ9XCJib2xkXCJcbiAgICAgICAgICAgICAgICBsaW5lSGVpZ2h0PVwidGl0bGVcIlxuICAgICAgICAgICAgICAgIGNvbG9yPVwidGV4dFwiXG4gICAgICAgICAgICAgICAgbT17MH1cbiAgICAgICAgICAgICAgICB0cnVuY2F0ZT1cImVsbGlwc2lzXCJcbiAgICAgICAgICAgICAgICB0cnVuY2F0ZUxpbmVzPXszfVxuICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAge3RpdGxlfVxuICAgICAgICAgICAgICA8L1RleHQ+XG5cbiAgICAgICAgICAgICAgPEZsZXhCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGdhcD17NH0gbXQ9XCJhdXRvXCIgbWluV2lkdGg9ezB9PlxuICAgICAgICAgICAgICAgIDxUZXh0XG4gICAgICAgICAgICAgICAgICBmb250U2l6ZT17MTR9XG4gICAgICAgICAgICAgICAgICBmb250V2VpZ2h0PVwiYm9sZFwiXG4gICAgICAgICAgICAgICAgICBsaW5lSGVpZ2h0PVwiYmFzZVwiXG4gICAgICAgICAgICAgICAgICBjb2xvcj1cInRleHRcIlxuICAgICAgICAgICAgICAgID5cbiAgICAgICAgICAgICAgICAgIHtpbnN0cnVjdG9yTmFtZX1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPFRleHRcbiAgICAgICAgICAgICAgICAgIGZvbnRTaXplPXsxNH1cbiAgICAgICAgICAgICAgICAgIGxpbmVIZWlnaHQ9XCJiYXNlXCJcbiAgICAgICAgICAgICAgICAgIGNvbG9yPVwidGV4dC1zZWNvbmRhcnlcIlxuICAgICAgICAgICAgICAgICAgdHJ1bmNhdGU9XCJlbGxpcHNpc1wiXG4gICAgICAgICAgICAgICAgICB0cnVuY2F0ZUxpbmVzPXsxfVxuICAgICAgICAgICAgICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgICAgICAgICAgICA+XG4gICAgICAgICAgICAgICAgICB7aW5zdHJ1Y3RvclRpdGxlfVxuICAgICAgICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgICAgICAgPC9GbGV4Qm94PlxuICAgICAgICAgICAgPC9GbGV4Qm94PlxuXG4gICAgICAgICAgICA8RmxleEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgYWxpZ25TZWxmPVwic3RyZXRjaFwiIG10PXs0fT5cbiAgICAgICAgICAgICAge25leHRDb2hvcnRTdGFydERhdGUgJiYgKFxuICAgICAgICAgICAgICAgIDw+XG4gICAgICAgICAgICAgICAgICA8RGl2aWRlciAvPlxuICAgICAgICAgICAgICAgICAgPEluZm9Sb3cgaWNvbj17PENhbGVuZGFySWNvbiBzaXplPXsxNn0gYXJpYS1oaWRkZW4gLz59PlxuICAgICAgICAgICAgICAgICAgICBOZXh0IGNvaG9ydDogU3RhcnRzeycgJ31cbiAgICAgICAgICAgICAgICAgICAgPFRleHQgYXM9XCJzcGFuXCIgZm9udFdlaWdodD1cImJvbGRcIj5cbiAgICAgICAgICAgICAgICAgICAgICB7bmV4dENvaG9ydFN0YXJ0RGF0ZX1cbiAgICAgICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICAgICAgPC9JbmZvUm93PlxuICAgICAgICAgICAgICAgIDwvPlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICA8RGl2aWRlciAvPlxuICAgICAgICAgICAgICA8SW5mb1JvdyBpY29uPXs8Q2VydGlmaWNhdGVJY29uIHNpemU9ezE2fSBhcmlhLWhpZGRlbiAvPn0+XG4gICAgICAgICAgICAgICAgV2l0aHsnICd9XG4gICAgICAgICAgICAgICAgPFRleHQgYXM9XCJzcGFuXCIgZm9udFdlaWdodD1cImJvbGRcIj5cbiAgICAgICAgICAgICAgICAgIENlcnRpZmljYXRlXG4gICAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgICA8L0luZm9Sb3c+XG4gICAgICAgICAgICA8L0ZsZXhCb3g+XG4gICAgICAgICAgPC9GbGV4Qm94PlxuICAgICAgICA8L0NhcmRBbmNob3I+XG4gICAgICA8L0NhcmQ+XG4gICAgKTtcbiAgfVxuKTtcbiJdfQ== */");
48
+ }), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/InstructorCard/index.tsx"],"names":[],"mappings":"AAwCmB","file":"../../src/InstructorCard/index.tsx","sourcesContent":["import { Anchor, Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CalendarIcon, CertificateIcon } from '@codecademy/gamut-icons';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, theme } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\nimport { forwardRef, PropsWithRef } from 'react';\n\nconst CardAnchor = styled(Anchor)(\n  css({\n    display: 'flex',\n    flexDirection: 'column',\n    height: '100%',\n    '&:hover, &:focus, &:focus-visible': {\n      color: 'text',\n      textDecoration: 'none',\n    },\n  })\n);\n\nconst StyledImg = styled.img`\n  width: 100%;\n  height: 100%;\n  object-fit: cover;\n  display: block;\n`;\n\nconst LastChanceBadge = styled(Badge)`\n  color: ${theme.colors.red};\n  border-color: ${theme.colors.red};\n  background-color: ${theme.colors['red-0']};\n  border: 1px solid ${theme.colors['red-500']};\n`;\n\nconst PromotionBadge = styled(Badge)`\n  color: ${theme.colors.black};\n  border-color: ${theme.colors.yellow};\n  background-color: ${theme.colors['yellow-0']};\n  border: 1px solid ${theme.colors['yellow-500']};\n`;\n\nconst DottedLine = styled(CheckerDense)(\n  css({\n    height: '1px',\n    display: 'flex',\n    position: 'absolute',\n    marginTop: '-1px',\n  })\n);\n\nexport const Divider: React.FC = () => {\n  return (\n    // the height and position properties are necessary for iOS devices\n    <Box py={4} height={0} position=\"relative\" aria-hidden>\n      <DottedLine />\n    </Box>\n  );\n};\n\ntype InfoRowProps = {\n  icon: React.ReactNode;\n  children: React.ReactNode;\n};\n\nconst InfoRow: React.FC<InfoRowProps> = ({ icon, children }) => (\n  <FlexBox alignItems=\"center\" gap={8}>\n    {icon}\n    <Text fontSize={14} lineHeight=\"base\" color=\"text\">\n      {children}\n    </Text>\n  </FlexBox>\n);\n\nexport type InstructorCardProps = {\n  /** The bootcamp title */\n  title: string;\n  /** Instructor's full name */\n  instructorName: string;\n  /** Instructor's professional title */\n  instructorTitle: string;\n  /** URL for the instructor's profile pic (includes background gradient) */\n  instructorImageUrl?: string;\n  /** Formatted start date for the next cohort (e.g. \"Feb 10\") */\n  nextCohortStartDate?: string;\n  /**\n   * Promotional label from the latest cohort (e.g. \"Last chance - 20% off\").\n   * Rendered centered on the top edge of the card, overlapping the border.\n   */\n  promotionBadgeLabel?: string;\n  /** Link destination */\n  href: string;\n  /** Click handler */\n  onClick: () => void;\n};\n\nexport const InstructorCard = forwardRef<\n  HTMLAnchorElement,\n  PropsWithRef<InstructorCardProps>\n>(\n  (\n    {\n      title,\n      instructorName,\n      instructorTitle,\n      instructorImageUrl,\n      nextCohortStartDate,\n      promotionBadgeLabel,\n      href,\n      onClick,\n    },\n    ref\n  ) => {\n    const isLastChance = promotionBadgeLabel\n      ?.toLowerCase()\n      .includes('last chance');\n\n    const PromotionBadgeComponent = isLastChance\n      ? LastChanceBadge\n      : PromotionBadge;\n\n    return (\n      <Box position=\"relative\" width=\"100%\" height=\"100%\">\n        {promotionBadgeLabel ? (\n          <FlexBox\n            justifyContent=\"center\"\n            position=\"absolute\"\n            left={0}\n            right={0}\n            top={0}\n            zIndex={2}\n            style={{\n              transform: 'translateY(-50%)',\n              pointerEvents: 'none',\n            }}\n          >\n            <PromotionBadgeComponent variant=\"tertiary\" size=\"base\">\n              {promotionBadgeLabel}\n            </PromotionBadgeComponent>\n          </FlexBox>\n        ) : null}\n        <Card\n          borderRadius=\"md\"\n          borderColor=\"border-primary\"\n          overflow=\"hidden\"\n          display=\"flex\"\n          flexDirection=\"column\"\n          isInteractive\n          p={0}\n          height=\"100%\"\n          width=\"100%\"\n        >\n          <CardAnchor\n            href={href}\n            onClick={onClick}\n            variant=\"interface\"\n            ref={ref}\n          >\n            <Box\n              width=\"100%\"\n              height={160}\n              position=\"relative\"\n              overflow=\"hidden\"\n              bg=\"navy-800\"\n            >\n              {instructorImageUrl && (\n                <StyledImg src={instructorImageUrl} alt={instructorName} />\n              )}\n            </Box>\n\n            <FlexBox\n              flexDirection=\"column\"\n              flex={1}\n              pt={12}\n              pb={4}\n              px={12}\n              overflow=\"hidden\"\n            >\n              <FlexBox\n                flexDirection=\"column\"\n                alignSelf=\"stretch\"\n                flex={1}\n                gap={4}\n              >\n                <Text\n                  as=\"h3\"\n                  fontSize={20}\n                  fontWeight=\"bold\"\n                  lineHeight=\"title\"\n                  color=\"text\"\n                  m={0}\n                  truncate=\"ellipsis\"\n                  truncateLines={3}\n                >\n                  {title}\n                </Text>\n\n                <FlexBox flexDirection=\"column\" gap={4} mt=\"auto\" minWidth={0}>\n                  <Text\n                    fontSize={14}\n                    fontWeight=\"bold\"\n                    lineHeight=\"base\"\n                    color=\"text\"\n                  >\n                    {instructorName}\n                  </Text>\n                  <Text\n                    fontSize={14}\n                    lineHeight=\"base\"\n                    color=\"text-secondary\"\n                    truncate=\"ellipsis\"\n                    truncateLines={1}\n                    width=\"100%\"\n                  >\n                    {instructorTitle}\n                  </Text>\n                </FlexBox>\n              </FlexBox>\n\n              <FlexBox flexDirection=\"column\" alignSelf=\"stretch\" mt={4}>\n                {nextCohortStartDate && (\n                  <>\n                    <Divider />\n                    <InfoRow icon={<CalendarIcon size={16} aria-hidden />}>\n                      Next cohort: Starts{' '}\n                      <Text as=\"span\" fontWeight=\"bold\">\n                        {nextCohortStartDate}\n                      </Text>\n                    </InfoRow>\n                  </>\n                )}\n                <Divider />\n                <InfoRow icon={<CertificateIcon size={16} aria-hidden />}>\n                  With{' '}\n                  <Text as=\"span\" fontWeight=\"bold\">\n                    Certificate\n                  </Text>\n                </InfoRow>\n              </FlexBox>\n            </FlexBox>\n          </CardAnchor>\n        </Card>\n      </Box>\n    );\n  }\n);\n"]} */");
41
49
  export const Divider = () => {
42
50
  return (
43
51
  /*#__PURE__*/
@@ -70,106 +78,130 @@ export const InstructorCard = /*#__PURE__*/forwardRef(({
70
78
  instructorTitle,
71
79
  instructorImageUrl,
72
80
  nextCohortStartDate,
81
+ promotionBadgeLabel,
73
82
  href,
74
83
  onClick
75
84
  }, ref) => {
76
- return /*#__PURE__*/_jsx(Card, {
77
- borderRadius: "md",
78
- borderColor: "border-primary",
79
- overflow: "hidden",
80
- display: "flex",
81
- flexDirection: "column",
82
- isInteractive: true,
83
- p: 0,
84
- height: "100%",
85
+ const isLastChance = promotionBadgeLabel?.toLowerCase().includes('last chance');
86
+ const PromotionBadgeComponent = isLastChance ? LastChanceBadge : PromotionBadge;
87
+ return /*#__PURE__*/_jsxs(Box, {
88
+ position: "relative",
85
89
  width: "100%",
86
- children: /*#__PURE__*/_jsxs(CardAnchor, {
87
- href: href,
88
- onClick: onClick,
89
- variant: "interface",
90
- ref: ref,
91
- children: [/*#__PURE__*/_jsx(Box, {
92
- width: "100%",
93
- height: 160,
94
- position: "relative",
95
- overflow: "hidden",
96
- bg: "navy-800",
97
- children: instructorImageUrl && /*#__PURE__*/_jsx(StyledImg, {
98
- src: instructorImageUrl,
99
- alt: instructorName
100
- })
101
- }), /*#__PURE__*/_jsxs(FlexBox, {
102
- flexDirection: "column",
103
- flex: 1,
104
- pt: 12,
105
- pb: 4,
106
- px: 12,
107
- overflow: "hidden",
108
- children: [/*#__PURE__*/_jsxs(FlexBox, {
90
+ height: "100%",
91
+ children: [promotionBadgeLabel ? /*#__PURE__*/_jsx(FlexBox, {
92
+ justifyContent: "center",
93
+ position: "absolute",
94
+ left: 0,
95
+ right: 0,
96
+ top: 0,
97
+ zIndex: 2,
98
+ style: {
99
+ transform: 'translateY(-50%)',
100
+ pointerEvents: 'none'
101
+ },
102
+ children: /*#__PURE__*/_jsx(PromotionBadgeComponent, {
103
+ variant: "tertiary",
104
+ size: "base",
105
+ children: promotionBadgeLabel
106
+ })
107
+ }) : null, /*#__PURE__*/_jsx(Card, {
108
+ borderRadius: "md",
109
+ borderColor: "border-primary",
110
+ overflow: "hidden",
111
+ display: "flex",
112
+ flexDirection: "column",
113
+ isInteractive: true,
114
+ p: 0,
115
+ height: "100%",
116
+ width: "100%",
117
+ children: /*#__PURE__*/_jsxs(CardAnchor, {
118
+ href: href,
119
+ onClick: onClick,
120
+ variant: "interface",
121
+ ref: ref,
122
+ children: [/*#__PURE__*/_jsx(Box, {
123
+ width: "100%",
124
+ height: 160,
125
+ position: "relative",
126
+ overflow: "hidden",
127
+ bg: "navy-800",
128
+ children: instructorImageUrl && /*#__PURE__*/_jsx(StyledImg, {
129
+ src: instructorImageUrl,
130
+ alt: instructorName
131
+ })
132
+ }), /*#__PURE__*/_jsxs(FlexBox, {
109
133
  flexDirection: "column",
110
- alignSelf: "stretch",
111
134
  flex: 1,
112
- gap: 4,
113
- children: [/*#__PURE__*/_jsx(Text, {
114
- as: "h3",
115
- fontSize: 20,
116
- fontWeight: "bold",
117
- lineHeight: "title",
118
- color: "text",
119
- m: 0,
120
- truncate: "ellipsis",
121
- truncateLines: 3,
122
- children: title
123
- }), /*#__PURE__*/_jsxs(FlexBox, {
135
+ pt: 12,
136
+ pb: 4,
137
+ px: 12,
138
+ overflow: "hidden",
139
+ children: [/*#__PURE__*/_jsxs(FlexBox, {
124
140
  flexDirection: "column",
141
+ alignSelf: "stretch",
142
+ flex: 1,
125
143
  gap: 4,
126
- mt: "auto",
127
- minWidth: 0,
128
144
  children: [/*#__PURE__*/_jsx(Text, {
129
- fontSize: 14,
145
+ as: "h3",
146
+ fontSize: 20,
130
147
  fontWeight: "bold",
131
- lineHeight: "base",
148
+ lineHeight: "title",
132
149
  color: "text",
133
- children: instructorName
134
- }), /*#__PURE__*/_jsx(Text, {
135
- fontSize: 14,
136
- lineHeight: "base",
137
- color: "text-secondary",
150
+ m: 0,
138
151
  truncate: "ellipsis",
139
- truncateLines: 1,
140
- width: "100%",
141
- children: instructorTitle
152
+ truncateLines: 3,
153
+ children: title
154
+ }), /*#__PURE__*/_jsxs(FlexBox, {
155
+ flexDirection: "column",
156
+ gap: 4,
157
+ mt: "auto",
158
+ minWidth: 0,
159
+ children: [/*#__PURE__*/_jsx(Text, {
160
+ fontSize: 14,
161
+ fontWeight: "bold",
162
+ lineHeight: "base",
163
+ color: "text",
164
+ children: instructorName
165
+ }), /*#__PURE__*/_jsx(Text, {
166
+ fontSize: 14,
167
+ lineHeight: "base",
168
+ color: "text-secondary",
169
+ truncate: "ellipsis",
170
+ truncateLines: 1,
171
+ width: "100%",
172
+ children: instructorTitle
173
+ })]
142
174
  })]
143
- })]
144
- }), /*#__PURE__*/_jsxs(FlexBox, {
145
- flexDirection: "column",
146
- alignSelf: "stretch",
147
- mt: 4,
148
- children: [nextCohortStartDate && /*#__PURE__*/_jsxs(_Fragment, {
149
- children: [/*#__PURE__*/_jsx(Divider, {}), /*#__PURE__*/_jsxs(InfoRow, {
150
- icon: /*#__PURE__*/_jsx(CalendarIcon, {
175
+ }), /*#__PURE__*/_jsxs(FlexBox, {
176
+ flexDirection: "column",
177
+ alignSelf: "stretch",
178
+ mt: 4,
179
+ children: [nextCohortStartDate && /*#__PURE__*/_jsxs(_Fragment, {
180
+ children: [/*#__PURE__*/_jsx(Divider, {}), /*#__PURE__*/_jsxs(InfoRow, {
181
+ icon: /*#__PURE__*/_jsx(CalendarIcon, {
182
+ size: 16,
183
+ "aria-hidden": true
184
+ }),
185
+ children: ["Next cohort: Starts", ' ', /*#__PURE__*/_jsx(Text, {
186
+ as: "span",
187
+ fontWeight: "bold",
188
+ children: nextCohortStartDate
189
+ })]
190
+ })]
191
+ }), /*#__PURE__*/_jsx(Divider, {}), /*#__PURE__*/_jsxs(InfoRow, {
192
+ icon: /*#__PURE__*/_jsx(CertificateIcon, {
151
193
  size: 16,
152
194
  "aria-hidden": true
153
195
  }),
154
- children: ["Next cohort: Starts", ' ', /*#__PURE__*/_jsx(Text, {
196
+ children: ["With", ' ', /*#__PURE__*/_jsx(Text, {
155
197
  as: "span",
156
198
  fontWeight: "bold",
157
- children: nextCohortStartDate
199
+ children: "Certificate"
158
200
  })]
159
201
  })]
160
- }), /*#__PURE__*/_jsx(Divider, {}), /*#__PURE__*/_jsxs(InfoRow, {
161
- icon: /*#__PURE__*/_jsx(CertificateIcon, {
162
- size: 16,
163
- "aria-hidden": true
164
- }),
165
- children: ["With", ' ', /*#__PURE__*/_jsx(Text, {
166
- as: "span",
167
- fontWeight: "bold",
168
- children: "Certificate"
169
- })]
170
202
  })]
171
203
  })]
172
- })]
173
- })
204
+ })
205
+ })]
174
206
  });
175
207
  });
@@ -63,6 +63,17 @@ export const InstructorCardWithoutNextCohort = {
63
63
  }),
64
64
  name: 'InstructorCard without Next Cohort'
65
65
  };
66
+ export const InstructorCardWithPromotionBadge = {
67
+ render: args => /*#__PURE__*/_jsx(Box, {
68
+ pb: 8,
69
+ pt: 16,
70
+ children: /*#__PURE__*/_jsx(InstructorCard, {
71
+ ...args,
72
+ promotionBadgeLabel: "Last chance - 20% off"
73
+ })
74
+ }),
75
+ name: 'InstructorCard with promotion badge'
76
+ };
66
77
  export const InstructorCardGrid = {
67
78
  render: args => /*#__PURE__*/_jsxs(LayoutGrid, {
68
79
  gap: 24,
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@codecademy/brand",
3
- "version": "4.2.0",
3
+ "version": "4.2.1-alpha.21bfd035d.0",
4
4
  "description": "Brand component library for Codecademy",
5
5
  "license": "MIT",
6
6
  "repository": "git@github.com:codecademy-engineering/mono.git",