@codecademy/brand 3.6.0-alpha.eb8166002f.0 → 3.6.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/AppHeader/AppHeaderElements/AppHeaderCatalogDropdown/NavPanels.js +31 -103
- package/dist/AppHeaderMobile/AppHeaderMainMenuMobile/index.js +1 -1
- package/dist/GlobalHeader/GlobalHeaderVariants.js +1 -1
- package/dist/LearningOutcomeFlyout/index.js +52 -11
- package/dist/LearningOutcomeFlyout/types.d.ts +2 -0
- package/dist/PricingSection/ConditionalColorMode.d.ts +4 -0
- package/dist/PricingSection/ConditionalColorMode.js +13 -0
- package/dist/PricingSection/PricingCard/PricingAmount.d.ts +7 -3
- package/dist/PricingSection/PricingCard/PricingAmount.js +92 -61
- package/dist/PricingSection/PricingCard/consts.js +14 -0
- package/dist/PricingSection/PricingCard/index.d.ts +6 -6
- package/dist/PricingSection/PricingCard/index.js +112 -153
- package/dist/PricingSection/PricingCard/types.d.ts +40 -0
- package/dist/PricingSection/PricingCard/types.js +63 -0
- package/dist/PricingSection/ProductCTA.d.ts +7 -5
- package/dist/PricingSection/ProductCTA.js +96 -31
- package/dist/PricingSection/Products.d.ts +4 -4
- package/dist/PricingSection/Products.js +28 -13
- package/dist/PricingSection/config.d.ts +1 -1
- package/dist/PricingSection/config.js +4 -2
- package/dist/PricingSection/index.d.ts +4 -5
- package/dist/PricingSection/index.js +27 -46
- package/dist/PricingSection/types.d.ts +3 -2
- package/dist/PricingSection/types.js +1 -0
- package/package.json +1 -1
|
@@ -1,64 +1,35 @@
|
|
|
1
1
|
import _styled from "@emotion/styled/base";
|
|
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
2
|
import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';
|
|
4
3
|
import { CheckerDense } from '@codecademy/gamut-patterns';
|
|
5
|
-
import {
|
|
4
|
+
import { css, pxRem, theme, variant } from '@codecademy/gamut-styles';
|
|
5
|
+
import { ConditionalColorMode } from '../ConditionalColorMode';
|
|
6
6
|
import { getProductDescription } from '../config';
|
|
7
|
-
import { PlanFeature } from '../PlanFeature';
|
|
8
7
|
import { ProductCTA } from '../ProductCTA';
|
|
9
8
|
import { Product } from '../types';
|
|
10
|
-
import { planDetails } from './consts';
|
|
11
9
|
import { PricingAmount } from './PricingAmount';
|
|
12
10
|
import { RecommendedBadge } from './RecommendedBadge';
|
|
13
11
|
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
14
12
|
export const smSize = 728;
|
|
15
|
-
const mdSize = 304;
|
|
16
|
-
const StyledList = /*#__PURE__*/_styled(FlexBox, {
|
|
17
|
-
target: "e3n4z5j4",
|
|
18
|
-
label: "StyledList"
|
|
19
|
-
})(process.env.NODE_ENV === "production" ? {
|
|
20
|
-
name: "1nd3o22",
|
|
21
|
-
styles: "padding-left:0;margin-bottom:0"
|
|
22
|
-
} : {
|
|
23
|
-
name: "1nd3o22",
|
|
24
|
-
styles: "padding-left:0;margin-bottom:0/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAgBkC","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { pxRem, system, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { getProductDescription, PlansByType } from '../config';\nimport { PlanFeature } from '../PlanFeature';\nimport { ProductCTA } from '../ProductCTA';\nimport { CopyConfig, CtaConfig, Product, ProductDetails } from '../types';\nimport { planDetails } from './consts';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\n\nexport const smSize = 728;\nconst mdSize = 304;\n\nconst StyledList = styled(FlexBox)`\n  padding-left: 0;\n  margin-bottom: 0;\n`.withComponent('ul');\n\nconst StyledCard = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledCardWithBorder = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 2,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledListRow = styled('li')<{ showRecommendedBadge?: boolean }>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${({ showRecommendedBadge }) => `\n    margin-top: ${showRecommendedBadge ? '1.5rem' : '0'};\n  `}\n  ${theme.breakpoints.md} {\n    margin-top: 0;\n    min-width: ${pxRem(mdSize)};\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {},\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst renderIncludesLabel = (product: Product, plansByType: PlansByType) => {\n  let includesLabel = '';\n\n  if (product === Product.Basic) {\n    includesLabel = 'Included in Basic';\n  } else if (\n    // only include the plus plan label if there is also a basic plan card visible\n    plansByType[Product.Basic] &&\n    (product === Product.Pro || product === Product.Silver)\n  ) {\n    includesLabel = 'Everything in Basic and more';\n  } else if (\n    // only include the pro plan label if there is also a plus plan card visible\n    (plansByType[Product.Pro] || plansByType[Product.Silver]) &&\n    product === Product.Gold\n  ) {\n    includesLabel = 'Everything in Plus and more';\n  }\n\n  return includesLabel ? (\n    <Text mt={16} fontWeight={700} width=\"fit-content\" m=\"auto\">\n      {includesLabel}\n    </Text>\n  ) : null;\n};\n\nexport const PricingCard: React.FC<{\n  product: Product;\n  productDetails: ProductDetails;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  isCurrentPlan?: boolean;\n  ctaConfig?: CtaConfig[keyof CtaConfig];\n  copyConfig?: CopyConfig[keyof CopyConfig];\n}> = ({\n  product,\n  productDetails,\n  plansByType,\n  isUserInIndia,\n  isCurrentPlan,\n  ctaConfig,\n  copyConfig,\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isSilverProduct = product === Product.Silver;\n  const isLegacyProduct = product === Product.Pro;\n\n  const planDetail = planDetails[product];\n\n  const CardComponent = !isGoldProduct ? StyledCard : StyledCardWithBorder;\n  const isMultiple = true;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <CardComponent position=\"relative\" display=\"flex\">\n        {product === Product.Gold && <RecommendedBadge />}\n        <CardHeader\n          variant={product}\n          py={8}\n          px={24}\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n        >\n          <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n            {\n              plansByType[product as keyof typeof plansByType].monthly\n                .displayName\n            }\n          </Text>\n          {isGoldProduct && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Build a career\n            </Badge>\n          )}\n          {(isSilverProduct || isLegacyProduct) && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Learn a skill\n            </Badge>\n          )}\n        </CardHeader>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            <PricingAmount\n              monthlyPrice={\n                plansByType[product as keyof typeof plansByType].monthly.price\n              }\n              price={productDetails.price}\n              product={product}\n              isUserInIndia={isUserInIndia}\n            />\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <CheckerDense mt={16} height=\"0.1rem\" />\n            <FlexBox\n              flexDirection=\"column\"\n              alignItems=\"center\"\n              my={12}\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n                {copyConfig?.description || getProductDescription(product)}\n              </Text>\n              {renderIncludesLabel(product, plansByType)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox\n          justifyContent=\"flex-end\"\n          alignContent={isMultiple ? 'flex-start' : 'unset'}\n          alignItems=\"center\"\n          position=\"relative\"\n          px={isMultiple ? 24 : { _: 40, sm: 0 }}\n          py={24}\n          m=\"auto\"\n          flexDirection=\"column\"\n          flexWrap=\"wrap\"\n          flexGrow={isMultiple ? 1 : { _: 1, sm: 'unset' }}\n          width={isMultiple ? 'auto' : { _: '100%', sm: '65%' }}\n        >\n          <StyledList\n            flexDirection={{ _: 'column' }}\n            flexWrap={isMultiple ? 'unset' : { _: 'nowrap', xs: 'wrap' }}\n            height={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: 188,\n                  }\n            }\n            width={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: '100%',\n                  }\n            }\n          >\n            {planDetail.features.map((feature) => (\n              <PlanFeature\n                key={feature.label}\n                feature={feature}\n                isMultiple={isMultiple}\n              />\n            ))}\n          </StyledList>\n        </FlexBox>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            product={product}\n            ctaConfig={ctaConfig}\n            isCurrentPlan={isCurrentPlan}\n          />\n        </FlexBox>\n      </CardComponent>\n    </StyledListRow>\n  );\n};\n"]} */",
|
|
25
|
-
toString: _EMOTION_STRINGIFIED_CSS_ERROR__
|
|
26
|
-
}).withComponent('ul', {
|
|
27
|
-
target: "e3n4z5j5",
|
|
28
|
-
label: "StyledList"
|
|
29
|
-
});
|
|
30
13
|
const StyledCard = /*#__PURE__*/_styled(Card, {
|
|
31
14
|
target: "e3n4z5j3",
|
|
32
15
|
label: "StyledCard"
|
|
33
|
-
})(
|
|
16
|
+
})(css({
|
|
34
17
|
height: 1,
|
|
35
18
|
borderWidth: 0,
|
|
36
19
|
padding: 0,
|
|
37
20
|
width: '100%',
|
|
38
|
-
flexDirection: 'column'
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAqBmB","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { pxRem, system, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { getProductDescription, PlansByType } from '../config';\nimport { PlanFeature } from '../PlanFeature';\nimport { ProductCTA } from '../ProductCTA';\nimport { CopyConfig, CtaConfig, Product, ProductDetails } from '../types';\nimport { planDetails } from './consts';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\n\nexport const smSize = 728;\nconst mdSize = 304;\n\nconst StyledList = styled(FlexBox)`\n  padding-left: 0;\n  margin-bottom: 0;\n`.withComponent('ul');\n\nconst StyledCard = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledCardWithBorder = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 2,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledListRow = styled('li')<{ showRecommendedBadge?: boolean }>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${({ showRecommendedBadge }) => `\n    margin-top: ${showRecommendedBadge ? '1.5rem' : '0'};\n  `}\n  ${theme.breakpoints.md} {\n    margin-top: 0;\n    min-width: ${pxRem(mdSize)};\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {},\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst renderIncludesLabel = (product: Product, plansByType: PlansByType) => {\n  let includesLabel = '';\n\n  if (product === Product.Basic) {\n    includesLabel = 'Included in Basic';\n  } else if (\n    // only include the plus plan label if there is also a basic plan card visible\n    plansByType[Product.Basic] &&\n    (product === Product.Pro || product === Product.Silver)\n  ) {\n    includesLabel = 'Everything in Basic and more';\n  } else if (\n    // only include the pro plan label if there is also a plus plan card visible\n    (plansByType[Product.Pro] || plansByType[Product.Silver]) &&\n    product === Product.Gold\n  ) {\n    includesLabel = 'Everything in Plus and more';\n  }\n\n  return includesLabel ? (\n    <Text mt={16} fontWeight={700} width=\"fit-content\" m=\"auto\">\n      {includesLabel}\n    </Text>\n  ) : null;\n};\n\nexport const PricingCard: React.FC<{\n  product: Product;\n  productDetails: ProductDetails;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  isCurrentPlan?: boolean;\n  ctaConfig?: CtaConfig[keyof CtaConfig];\n  copyConfig?: CopyConfig[keyof CopyConfig];\n}> = ({\n  product,\n  productDetails,\n  plansByType,\n  isUserInIndia,\n  isCurrentPlan,\n  ctaConfig,\n  copyConfig,\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isSilverProduct = product === Product.Silver;\n  const isLegacyProduct = product === Product.Pro;\n\n  const planDetail = planDetails[product];\n\n  const CardComponent = !isGoldProduct ? StyledCard : StyledCardWithBorder;\n  const isMultiple = true;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <CardComponent position=\"relative\" display=\"flex\">\n        {product === Product.Gold && <RecommendedBadge />}\n        <CardHeader\n          variant={product}\n          py={8}\n          px={24}\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n        >\n          <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n            {\n              plansByType[product as keyof typeof plansByType].monthly\n                .displayName\n            }\n          </Text>\n          {isGoldProduct && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Build a career\n            </Badge>\n          )}\n          {(isSilverProduct || isLegacyProduct) && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Learn a skill\n            </Badge>\n          )}\n        </CardHeader>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            <PricingAmount\n              monthlyPrice={\n                plansByType[product as keyof typeof plansByType].monthly.price\n              }\n              price={productDetails.price}\n              product={product}\n              isUserInIndia={isUserInIndia}\n            />\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <CheckerDense mt={16} height=\"0.1rem\" />\n            <FlexBox\n              flexDirection=\"column\"\n              alignItems=\"center\"\n              my={12}\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n                {copyConfig?.description || getProductDescription(product)}\n              </Text>\n              {renderIncludesLabel(product, plansByType)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox\n          justifyContent=\"flex-end\"\n          alignContent={isMultiple ? 'flex-start' : 'unset'}\n          alignItems=\"center\"\n          position=\"relative\"\n          px={isMultiple ? 24 : { _: 40, sm: 0 }}\n          py={24}\n          m=\"auto\"\n          flexDirection=\"column\"\n          flexWrap=\"wrap\"\n          flexGrow={isMultiple ? 1 : { _: 1, sm: 'unset' }}\n          width={isMultiple ? 'auto' : { _: '100%', sm: '65%' }}\n        >\n          <StyledList\n            flexDirection={{ _: 'column' }}\n            flexWrap={isMultiple ? 'unset' : { _: 'nowrap', xs: 'wrap' }}\n            height={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: 188,\n                  }\n            }\n            width={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: '100%',\n                  }\n            }\n          >\n            {planDetail.features.map((feature) => (\n              <PlanFeature\n                key={feature.label}\n                feature={feature}\n                isMultiple={isMultiple}\n              />\n            ))}\n          </StyledList>\n        </FlexBox>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            product={product}\n            ctaConfig={ctaConfig}\n            isCurrentPlan={isCurrentPlan}\n          />\n        </FlexBox>\n      </CardComponent>\n    </StyledListRow>\n  );\n};\n"]} */");
|
|
42
|
-
const StyledCardWithBorder = /*#__PURE__*/_styled(Card, {
|
|
21
|
+
flexDirection: 'column'
|
|
22
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAemB","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, pxRem, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { ConditionalColorMode } from '../ConditionalColorMode';\nimport { getProductDescription, PlansByType } from '../config';\nimport { ProductCTA } from '../ProductCTA';\nimport { Product, ProductDetails, User } from '../types';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\nimport { AllCurrency } from './types';\n\nexport const smSize = 728;\n\nconst StyledCard = styled(Card)(\n  css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n  })\n);\n\nconst Divider = styled(CheckerDense)(\n  css({\n    height: '0.1rem',\n  })\n);\n\ntype StyledListRowProps = {\n  showRecommendedBadge: boolean;\n};\n\nconst StyledListRow = styled('li')<StyledListRowProps>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${theme.breakpoints.lg} {\n    min-width: 36rem;\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Enterprise]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {\n      backgroundColor: theme.colors.background,\n    },\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst getBadgeText = (product: Product): string | null => {\n  switch (product) {\n    case Product.Gold:\n      return 'Advance your career';\n    case Product.Silver:\n    case Product.Pro:\n      return 'Learn a skill';\n    case Product.Teams:\n      return 'Recommended for 2+';\n    case Product.Enterprise:\n      return 'Recommended for 25+';\n    default:\n      return null;\n  }\n};\n\nconst renderProductDescription = (product: Product) => {\n  const productFeatures = (\n    <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n      {getProductDescription(product)}\n    </Text>\n  );\n\n  return (\n    <FlexBox mt={8} alignItems=\"center\" flexDirection=\"column\">\n      {productFeatures}\n    </FlexBox>\n  );\n};\n\nexport const PricingCard = ({\n  product,\n  productDetails,\n  currency,\n  plansByType,\n  isUserInIndia,\n  user,\n}: {\n  product: Product;\n  productDetails: ProductDetails;\n  currency: AllCurrency;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  user: User;\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isTeamsProduct = product === Product.Teams;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <StyledCard display=\"flex\" shadow=\"patternLeft\">\n        {isGoldProduct && <RecommendedBadge />}\n        <ConditionalColorMode condition={isTeamsProduct}>\n          <CardHeader\n            variant={product}\n            py={8}\n            px={24}\n            alignItems=\"center\"\n            justifyContent=\"space-between\"\n          >\n            <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n              {plansByType[product].monthly?.displayName ||\n                plansByType[product].annual?.displayName}\n            </Text>\n            {isGoldProduct && <Text screenreader>Recommended</Text>}\n            {getBadgeText(product) && (\n              <Badge variant=\"tertiary\" ml={12}>\n                {getBadgeText(product)}\n              </Badge>\n            )}\n          </CardHeader>\n        </ConditionalColorMode>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            {product === Product.Enterprise ? (\n              <FlexBox\n                column\n                alignItems=\"center\"\n                justifyContent=\"center\"\n                minHeight={134}\n              >\n                <Text color=\"text-secondary\" mb={8} textAlign=\"center\">\n                  Looking to train a larger team?\n                </Text>\n                <Text variant=\"title-lg\" textAlign=\"center\">\n                  Contact us for a custom quote\n                </Text>\n              </FlexBox>\n            ) : (\n              <PricingAmount\n                monthlyPrice={plansByType[product].monthly?.price}\n                price={productDetails.price}\n                product={product}\n                currency={currency}\n                isUserInIndia={isUserInIndia}\n              />\n            )}\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <Divider mt={16} />\n            <FlexBox\n              flexDirection=\"column\"\n              my={12}\n              /**\n               * Previously we were arbitrarily sizing cards to a width that was\n               * \"too small\" for the content they contained. The actual affect this was\n               * having was shrinking this section down to 256px, so rather than size\n               * the card, I'm opting to size this particular container w/i.\n               * This allows the cards themselves to be more flexible and simply styled\n               * while still maintaining the \"proper size\" on smaller screens.\n               */\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              {renderProductDescription(product)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            user={user}\n            productDetails={productDetails}\n            product={product}\n          />\n        </FlexBox>\n      </StyledCard>\n    </StyledListRow>\n  );\n};\n"]} */");
|
|
23
|
+
const Divider = /*#__PURE__*/_styled(CheckerDense, {
|
|
43
24
|
target: "e3n4z5j2",
|
|
44
|
-
label: "
|
|
45
|
-
})(
|
|
46
|
-
height:
|
|
47
|
-
|
|
48
|
-
padding: 0,
|
|
49
|
-
width: '100%',
|
|
50
|
-
flexDirection: 'column',
|
|
51
|
-
display: 'flex',
|
|
52
|
-
zIndex: 1
|
|
53
|
-
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAiC6B","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { pxRem, system, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { getProductDescription, PlansByType } from '../config';\nimport { PlanFeature } from '../PlanFeature';\nimport { ProductCTA } from '../ProductCTA';\nimport { CopyConfig, CtaConfig, Product, ProductDetails } from '../types';\nimport { planDetails } from './consts';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\n\nexport const smSize = 728;\nconst mdSize = 304;\n\nconst StyledList = styled(FlexBox)`\n  padding-left: 0;\n  margin-bottom: 0;\n`.withComponent('ul');\n\nconst StyledCard = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledCardWithBorder = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 2,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledListRow = styled('li')<{ showRecommendedBadge?: boolean }>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${({ showRecommendedBadge }) => `\n    margin-top: ${showRecommendedBadge ? '1.5rem' : '0'};\n  `}\n  ${theme.breakpoints.md} {\n    margin-top: 0;\n    min-width: ${pxRem(mdSize)};\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {},\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst renderIncludesLabel = (product: Product, plansByType: PlansByType) => {\n  let includesLabel = '';\n\n  if (product === Product.Basic) {\n    includesLabel = 'Included in Basic';\n  } else if (\n    // only include the plus plan label if there is also a basic plan card visible\n    plansByType[Product.Basic] &&\n    (product === Product.Pro || product === Product.Silver)\n  ) {\n    includesLabel = 'Everything in Basic and more';\n  } else if (\n    // only include the pro plan label if there is also a plus plan card visible\n    (plansByType[Product.Pro] || plansByType[Product.Silver]) &&\n    product === Product.Gold\n  ) {\n    includesLabel = 'Everything in Plus and more';\n  }\n\n  return includesLabel ? (\n    <Text mt={16} fontWeight={700} width=\"fit-content\" m=\"auto\">\n      {includesLabel}\n    </Text>\n  ) : null;\n};\n\nexport const PricingCard: React.FC<{\n  product: Product;\n  productDetails: ProductDetails;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  isCurrentPlan?: boolean;\n  ctaConfig?: CtaConfig[keyof CtaConfig];\n  copyConfig?: CopyConfig[keyof CopyConfig];\n}> = ({\n  product,\n  productDetails,\n  plansByType,\n  isUserInIndia,\n  isCurrentPlan,\n  ctaConfig,\n  copyConfig,\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isSilverProduct = product === Product.Silver;\n  const isLegacyProduct = product === Product.Pro;\n\n  const planDetail = planDetails[product];\n\n  const CardComponent = !isGoldProduct ? StyledCard : StyledCardWithBorder;\n  const isMultiple = true;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <CardComponent position=\"relative\" display=\"flex\">\n        {product === Product.Gold && <RecommendedBadge />}\n        <CardHeader\n          variant={product}\n          py={8}\n          px={24}\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n        >\n          <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n            {\n              plansByType[product as keyof typeof plansByType].monthly\n                .displayName\n            }\n          </Text>\n          {isGoldProduct && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Build a career\n            </Badge>\n          )}\n          {(isSilverProduct || isLegacyProduct) && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Learn a skill\n            </Badge>\n          )}\n        </CardHeader>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            <PricingAmount\n              monthlyPrice={\n                plansByType[product as keyof typeof plansByType].monthly.price\n              }\n              price={productDetails.price}\n              product={product}\n              isUserInIndia={isUserInIndia}\n            />\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <CheckerDense mt={16} height=\"0.1rem\" />\n            <FlexBox\n              flexDirection=\"column\"\n              alignItems=\"center\"\n              my={12}\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n                {copyConfig?.description || getProductDescription(product)}\n              </Text>\n              {renderIncludesLabel(product, plansByType)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox\n          justifyContent=\"flex-end\"\n          alignContent={isMultiple ? 'flex-start' : 'unset'}\n          alignItems=\"center\"\n          position=\"relative\"\n          px={isMultiple ? 24 : { _: 40, sm: 0 }}\n          py={24}\n          m=\"auto\"\n          flexDirection=\"column\"\n          flexWrap=\"wrap\"\n          flexGrow={isMultiple ? 1 : { _: 1, sm: 'unset' }}\n          width={isMultiple ? 'auto' : { _: '100%', sm: '65%' }}\n        >\n          <StyledList\n            flexDirection={{ _: 'column' }}\n            flexWrap={isMultiple ? 'unset' : { _: 'nowrap', xs: 'wrap' }}\n            height={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: 188,\n                  }\n            }\n            width={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: '100%',\n                  }\n            }\n          >\n            {planDetail.features.map((feature) => (\n              <PlanFeature\n                key={feature.label}\n                feature={feature}\n                isMultiple={isMultiple}\n              />\n            ))}\n          </StyledList>\n        </FlexBox>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            product={product}\n            ctaConfig={ctaConfig}\n            isCurrentPlan={isCurrentPlan}\n          />\n        </FlexBox>\n      </CardComponent>\n    </StyledListRow>\n  );\n};\n"]} */");
|
|
25
|
+
label: "Divider"
|
|
26
|
+
})(css({
|
|
27
|
+
height: '0.1rem'
|
|
28
|
+
}), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAyBgB","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, pxRem, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { ConditionalColorMode } from '../ConditionalColorMode';\nimport { getProductDescription, PlansByType } from '../config';\nimport { ProductCTA } from '../ProductCTA';\nimport { Product, ProductDetails, User } from '../types';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\nimport { AllCurrency } from './types';\n\nexport const smSize = 728;\n\nconst StyledCard = styled(Card)(\n  css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n  })\n);\n\nconst Divider = styled(CheckerDense)(\n  css({\n    height: '0.1rem',\n  })\n);\n\ntype StyledListRowProps = {\n  showRecommendedBadge: boolean;\n};\n\nconst StyledListRow = styled('li')<StyledListRowProps>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${theme.breakpoints.lg} {\n    min-width: 36rem;\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Enterprise]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {\n      backgroundColor: theme.colors.background,\n    },\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst getBadgeText = (product: Product): string | null => {\n  switch (product) {\n    case Product.Gold:\n      return 'Advance your career';\n    case Product.Silver:\n    case Product.Pro:\n      return 'Learn a skill';\n    case Product.Teams:\n      return 'Recommended for 2+';\n    case Product.Enterprise:\n      return 'Recommended for 25+';\n    default:\n      return null;\n  }\n};\n\nconst renderProductDescription = (product: Product) => {\n  const productFeatures = (\n    <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n      {getProductDescription(product)}\n    </Text>\n  );\n\n  return (\n    <FlexBox mt={8} alignItems=\"center\" flexDirection=\"column\">\n      {productFeatures}\n    </FlexBox>\n  );\n};\n\nexport const PricingCard = ({\n  product,\n  productDetails,\n  currency,\n  plansByType,\n  isUserInIndia,\n  user,\n}: {\n  product: Product;\n  productDetails: ProductDetails;\n  currency: AllCurrency;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  user: User;\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isTeamsProduct = product === Product.Teams;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <StyledCard display=\"flex\" shadow=\"patternLeft\">\n        {isGoldProduct && <RecommendedBadge />}\n        <ConditionalColorMode condition={isTeamsProduct}>\n          <CardHeader\n            variant={product}\n            py={8}\n            px={24}\n            alignItems=\"center\"\n            justifyContent=\"space-between\"\n          >\n            <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n              {plansByType[product].monthly?.displayName ||\n                plansByType[product].annual?.displayName}\n            </Text>\n            {isGoldProduct && <Text screenreader>Recommended</Text>}\n            {getBadgeText(product) && (\n              <Badge variant=\"tertiary\" ml={12}>\n                {getBadgeText(product)}\n              </Badge>\n            )}\n          </CardHeader>\n        </ConditionalColorMode>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            {product === Product.Enterprise ? (\n              <FlexBox\n                column\n                alignItems=\"center\"\n                justifyContent=\"center\"\n                minHeight={134}\n              >\n                <Text color=\"text-secondary\" mb={8} textAlign=\"center\">\n                  Looking to train a larger team?\n                </Text>\n                <Text variant=\"title-lg\" textAlign=\"center\">\n                  Contact us for a custom quote\n                </Text>\n              </FlexBox>\n            ) : (\n              <PricingAmount\n                monthlyPrice={plansByType[product].monthly?.price}\n                price={productDetails.price}\n                product={product}\n                currency={currency}\n                isUserInIndia={isUserInIndia}\n              />\n            )}\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <Divider mt={16} />\n            <FlexBox\n              flexDirection=\"column\"\n              my={12}\n              /**\n               * Previously we were arbitrarily sizing cards to a width that was\n               * \"too small\" for the content they contained. The actual affect this was\n               * having was shrinking this section down to 256px, so rather than size\n               * the card, I'm opting to size this particular container w/i.\n               * This allows the cards themselves to be more flexible and simply styled\n               * while still maintaining the \"proper size\" on smaller screens.\n               */\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              {renderProductDescription(product)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            user={user}\n            productDetails={productDetails}\n            product={product}\n          />\n        </FlexBox>\n      </StyledCard>\n    </StyledListRow>\n  );\n};\n"]} */");
|
|
54
29
|
const StyledListRow = /*#__PURE__*/_styled('li', {
|
|
55
30
|
target: "e3n4z5j1",
|
|
56
31
|
label: "StyledListRow"
|
|
57
|
-
})("width:100%;max-width:", pxRem(smSize), ";height:100%;display:grid;", (
|
|
58
|
-
showRecommendedBadge
|
|
59
|
-
}) => `
|
|
60
|
-
margin-top: ${showRecommendedBadge ? '1.5rem' : '0'};
|
|
61
|
-
`, " ", theme.breakpoints.md, "{margin-top:0;min-width:", pxRem(mdSize), ";}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AA6CsE","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { pxRem, system, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { getProductDescription, PlansByType } from '../config';\nimport { PlanFeature } from '../PlanFeature';\nimport { ProductCTA } from '../ProductCTA';\nimport { CopyConfig, CtaConfig, Product, ProductDetails } from '../types';\nimport { planDetails } from './consts';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\n\nexport const smSize = 728;\nconst mdSize = 304;\n\nconst StyledList = styled(FlexBox)`\n  padding-left: 0;\n  margin-bottom: 0;\n`.withComponent('ul');\n\nconst StyledCard = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledCardWithBorder = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 2,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledListRow = styled('li')<{ showRecommendedBadge?: boolean }>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${({ showRecommendedBadge }) => `\n    margin-top: ${showRecommendedBadge ? '1.5rem' : '0'};\n  `}\n  ${theme.breakpoints.md} {\n    margin-top: 0;\n    min-width: ${pxRem(mdSize)};\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {},\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst renderIncludesLabel = (product: Product, plansByType: PlansByType) => {\n  let includesLabel = '';\n\n  if (product === Product.Basic) {\n    includesLabel = 'Included in Basic';\n  } else if (\n    // only include the plus plan label if there is also a basic plan card visible\n    plansByType[Product.Basic] &&\n    (product === Product.Pro || product === Product.Silver)\n  ) {\n    includesLabel = 'Everything in Basic and more';\n  } else if (\n    // only include the pro plan label if there is also a plus plan card visible\n    (plansByType[Product.Pro] || plansByType[Product.Silver]) &&\n    product === Product.Gold\n  ) {\n    includesLabel = 'Everything in Plus and more';\n  }\n\n  return includesLabel ? (\n    <Text mt={16} fontWeight={700} width=\"fit-content\" m=\"auto\">\n      {includesLabel}\n    </Text>\n  ) : null;\n};\n\nexport const PricingCard: React.FC<{\n  product: Product;\n  productDetails: ProductDetails;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  isCurrentPlan?: boolean;\n  ctaConfig?: CtaConfig[keyof CtaConfig];\n  copyConfig?: CopyConfig[keyof CopyConfig];\n}> = ({\n  product,\n  productDetails,\n  plansByType,\n  isUserInIndia,\n  isCurrentPlan,\n  ctaConfig,\n  copyConfig,\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isSilverProduct = product === Product.Silver;\n  const isLegacyProduct = product === Product.Pro;\n\n  const planDetail = planDetails[product];\n\n  const CardComponent = !isGoldProduct ? StyledCard : StyledCardWithBorder;\n  const isMultiple = true;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <CardComponent position=\"relative\" display=\"flex\">\n        {product === Product.Gold && <RecommendedBadge />}\n        <CardHeader\n          variant={product}\n          py={8}\n          px={24}\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n        >\n          <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n            {\n              plansByType[product as keyof typeof plansByType].monthly\n                .displayName\n            }\n          </Text>\n          {isGoldProduct && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Build a career\n            </Badge>\n          )}\n          {(isSilverProduct || isLegacyProduct) && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Learn a skill\n            </Badge>\n          )}\n        </CardHeader>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            <PricingAmount\n              monthlyPrice={\n                plansByType[product as keyof typeof plansByType].monthly.price\n              }\n              price={productDetails.price}\n              product={product}\n              isUserInIndia={isUserInIndia}\n            />\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <CheckerDense mt={16} height=\"0.1rem\" />\n            <FlexBox\n              flexDirection=\"column\"\n              alignItems=\"center\"\n              my={12}\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n                {copyConfig?.description || getProductDescription(product)}\n              </Text>\n              {renderIncludesLabel(product, plansByType)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox\n          justifyContent=\"flex-end\"\n          alignContent={isMultiple ? 'flex-start' : 'unset'}\n          alignItems=\"center\"\n          position=\"relative\"\n          px={isMultiple ? 24 : { _: 40, sm: 0 }}\n          py={24}\n          m=\"auto\"\n          flexDirection=\"column\"\n          flexWrap=\"wrap\"\n          flexGrow={isMultiple ? 1 : { _: 1, sm: 'unset' }}\n          width={isMultiple ? 'auto' : { _: '100%', sm: '65%' }}\n        >\n          <StyledList\n            flexDirection={{ _: 'column' }}\n            flexWrap={isMultiple ? 'unset' : { _: 'nowrap', xs: 'wrap' }}\n            height={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: 188,\n                  }\n            }\n            width={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: '100%',\n                  }\n            }\n          >\n            {planDetail.features.map((feature) => (\n              <PlanFeature\n                key={feature.label}\n                feature={feature}\n                isMultiple={isMultiple}\n              />\n            ))}\n          </StyledList>\n        </FlexBox>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            product={product}\n            ctaConfig={ctaConfig}\n            isCurrentPlan={isCurrentPlan}\n          />\n        </FlexBox>\n      </CardComponent>\n    </StyledListRow>\n  );\n};\n"]} */"));
|
|
32
|
+
})("width:100%;max-width:", pxRem(smSize), ";height:100%;display:grid;", theme.breakpoints.lg, "{min-width:36rem;}" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAmCsD","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, pxRem, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { ConditionalColorMode } from '../ConditionalColorMode';\nimport { getProductDescription, PlansByType } from '../config';\nimport { ProductCTA } from '../ProductCTA';\nimport { Product, ProductDetails, User } from '../types';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\nimport { AllCurrency } from './types';\n\nexport const smSize = 728;\n\nconst StyledCard = styled(Card)(\n  css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n  })\n);\n\nconst Divider = styled(CheckerDense)(\n  css({\n    height: '0.1rem',\n  })\n);\n\ntype StyledListRowProps = {\n  showRecommendedBadge: boolean;\n};\n\nconst StyledListRow = styled('li')<StyledListRowProps>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${theme.breakpoints.lg} {\n    min-width: 36rem;\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Enterprise]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {\n      backgroundColor: theme.colors.background,\n    },\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst getBadgeText = (product: Product): string | null => {\n  switch (product) {\n    case Product.Gold:\n      return 'Advance your career';\n    case Product.Silver:\n    case Product.Pro:\n      return 'Learn a skill';\n    case Product.Teams:\n      return 'Recommended for 2+';\n    case Product.Enterprise:\n      return 'Recommended for 25+';\n    default:\n      return null;\n  }\n};\n\nconst renderProductDescription = (product: Product) => {\n  const productFeatures = (\n    <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n      {getProductDescription(product)}\n    </Text>\n  );\n\n  return (\n    <FlexBox mt={8} alignItems=\"center\" flexDirection=\"column\">\n      {productFeatures}\n    </FlexBox>\n  );\n};\n\nexport const PricingCard = ({\n  product,\n  productDetails,\n  currency,\n  plansByType,\n  isUserInIndia,\n  user,\n}: {\n  product: Product;\n  productDetails: ProductDetails;\n  currency: AllCurrency;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  user: User;\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isTeamsProduct = product === Product.Teams;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <StyledCard display=\"flex\" shadow=\"patternLeft\">\n        {isGoldProduct && <RecommendedBadge />}\n        <ConditionalColorMode condition={isTeamsProduct}>\n          <CardHeader\n            variant={product}\n            py={8}\n            px={24}\n            alignItems=\"center\"\n            justifyContent=\"space-between\"\n          >\n            <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n              {plansByType[product].monthly?.displayName ||\n                plansByType[product].annual?.displayName}\n            </Text>\n            {isGoldProduct && <Text screenreader>Recommended</Text>}\n            {getBadgeText(product) && (\n              <Badge variant=\"tertiary\" ml={12}>\n                {getBadgeText(product)}\n              </Badge>\n            )}\n          </CardHeader>\n        </ConditionalColorMode>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            {product === Product.Enterprise ? (\n              <FlexBox\n                column\n                alignItems=\"center\"\n                justifyContent=\"center\"\n                minHeight={134}\n              >\n                <Text color=\"text-secondary\" mb={8} textAlign=\"center\">\n                  Looking to train a larger team?\n                </Text>\n                <Text variant=\"title-lg\" textAlign=\"center\">\n                  Contact us for a custom quote\n                </Text>\n              </FlexBox>\n            ) : (\n              <PricingAmount\n                monthlyPrice={plansByType[product].monthly?.price}\n                price={productDetails.price}\n                product={product}\n                currency={currency}\n                isUserInIndia={isUserInIndia}\n              />\n            )}\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <Divider mt={16} />\n            <FlexBox\n              flexDirection=\"column\"\n              my={12}\n              /**\n               * Previously we were arbitrarily sizing cards to a width that was\n               * \"too small\" for the content they contained. The actual affect this was\n               * having was shrinking this section down to 256px, so rather than size\n               * the card, I'm opting to size this particular container w/i.\n               * This allows the cards themselves to be more flexible and simply styled\n               * while still maintaining the \"proper size\" on smaller screens.\n               */\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              {renderProductDescription(product)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            user={user}\n            productDetails={productDetails}\n            product={product}\n          />\n        </FlexBox>\n      </StyledCard>\n    </StyledListRow>\n  );\n};\n"]} */"));
|
|
62
33
|
const cardHeaderVariants = variant({
|
|
63
34
|
base: {
|
|
64
35
|
backgroundColor: 'transparent'
|
|
@@ -70,6 +41,12 @@ const cardHeaderVariants = variant({
|
|
|
70
41
|
borderBottomColor: theme.colors['navy-200'],
|
|
71
42
|
color: 'navy-500'
|
|
72
43
|
},
|
|
44
|
+
[Product.Enterprise]: {
|
|
45
|
+
borderBottomWidth: 1,
|
|
46
|
+
borderBottomStyle: 'solid',
|
|
47
|
+
borderBottomColor: theme.colors['navy-200'],
|
|
48
|
+
color: 'navy-500'
|
|
49
|
+
},
|
|
73
50
|
[Product.Silver]: {
|
|
74
51
|
backgroundColor: theme.colors.paleYellow
|
|
75
52
|
},
|
|
@@ -79,161 +56,143 @@ const cardHeaderVariants = variant({
|
|
|
79
56
|
[Product.Pro]: {
|
|
80
57
|
backgroundColor: theme.colors.yellow
|
|
81
58
|
},
|
|
82
|
-
[Product.Teams]: {
|
|
59
|
+
[Product.Teams]: {
|
|
60
|
+
backgroundColor: theme.colors.background
|
|
61
|
+
}
|
|
83
62
|
}
|
|
84
63
|
});
|
|
85
64
|
const CardHeader = /*#__PURE__*/_styled(FlexBox, {
|
|
86
65
|
target: "e3n4z5j0",
|
|
87
66
|
label: "CardHeader"
|
|
88
|
-
})(cardHeaderVariants, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AAmFmB","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { pxRem, system, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { getProductDescription, PlansByType } from '../config';\nimport { PlanFeature } from '../PlanFeature';\nimport { ProductCTA } from '../ProductCTA';\nimport { CopyConfig, CtaConfig, Product, ProductDetails } from '../types';\nimport { planDetails } from './consts';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\n\nexport const smSize = 728;\nconst mdSize = 304;\n\nconst StyledList = styled(FlexBox)`\n  padding-left: 0;\n  margin-bottom: 0;\n`.withComponent('ul');\n\nconst StyledCard = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledCardWithBorder = styled(Card)(\n  system.css({\n    height: 1,\n    borderWidth: 2,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n    display: 'flex',\n    zIndex: 1,\n  })\n);\n\nconst StyledListRow = styled('li')<{ showRecommendedBadge?: boolean }>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${({ showRecommendedBadge }) => `\n    margin-top: ${showRecommendedBadge ? '1.5rem' : '0'};\n  `}\n  ${theme.breakpoints.md} {\n    margin-top: 0;\n    min-width: ${pxRem(mdSize)};\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {},\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst renderIncludesLabel = (product: Product, plansByType: PlansByType) => {\n  let includesLabel = '';\n\n  if (product === Product.Basic) {\n    includesLabel = 'Included in Basic';\n  } else if (\n    // only include the plus plan label if there is also a basic plan card visible\n    plansByType[Product.Basic] &&\n    (product === Product.Pro || product === Product.Silver)\n  ) {\n    includesLabel = 'Everything in Basic and more';\n  } else if (\n    // only include the pro plan label if there is also a plus plan card visible\n    (plansByType[Product.Pro] || plansByType[Product.Silver]) &&\n    product === Product.Gold\n  ) {\n    includesLabel = 'Everything in Plus and more';\n  }\n\n  return includesLabel ? (\n    <Text mt={16} fontWeight={700} width=\"fit-content\" m=\"auto\">\n      {includesLabel}\n    </Text>\n  ) : null;\n};\n\nexport const PricingCard: React.FC<{\n  product: Product;\n  productDetails: ProductDetails;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  isCurrentPlan?: boolean;\n  ctaConfig?: CtaConfig[keyof CtaConfig];\n  copyConfig?: CopyConfig[keyof CopyConfig];\n}> = ({\n  product,\n  productDetails,\n  plansByType,\n  isUserInIndia,\n  isCurrentPlan,\n  ctaConfig,\n  copyConfig,\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isSilverProduct = product === Product.Silver;\n  const isLegacyProduct = product === Product.Pro;\n\n  const planDetail = planDetails[product];\n\n  const CardComponent = !isGoldProduct ? StyledCard : StyledCardWithBorder;\n  const isMultiple = true;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <CardComponent position=\"relative\" display=\"flex\">\n        {product === Product.Gold && <RecommendedBadge />}\n        <CardHeader\n          variant={product}\n          py={8}\n          px={24}\n          alignItems=\"center\"\n          justifyContent=\"space-between\"\n        >\n          <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n            {\n              plansByType[product as keyof typeof plansByType].monthly\n                .displayName\n            }\n          </Text>\n          {isGoldProduct && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Build a career\n            </Badge>\n          )}\n          {(isSilverProduct || isLegacyProduct) && (\n            <Badge variant=\"tertiary\" ml={12}>\n              Learn a skill\n            </Badge>\n          )}\n        </CardHeader>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            <PricingAmount\n              monthlyPrice={\n                plansByType[product as keyof typeof plansByType].monthly.price\n              }\n              price={productDetails.price}\n              product={product}\n              isUserInIndia={isUserInIndia}\n            />\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <CheckerDense mt={16} height=\"0.1rem\" />\n            <FlexBox\n              flexDirection=\"column\"\n              alignItems=\"center\"\n              my={12}\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n                {copyConfig?.description || getProductDescription(product)}\n              </Text>\n              {renderIncludesLabel(product, plansByType)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox\n          justifyContent=\"flex-end\"\n          alignContent={isMultiple ? 'flex-start' : 'unset'}\n          alignItems=\"center\"\n          position=\"relative\"\n          px={isMultiple ? 24 : { _: 40, sm: 0 }}\n          py={24}\n          m=\"auto\"\n          flexDirection=\"column\"\n          flexWrap=\"wrap\"\n          flexGrow={isMultiple ? 1 : { _: 1, sm: 'unset' }}\n          width={isMultiple ? 'auto' : { _: '100%', sm: '65%' }}\n        >\n          <StyledList\n            flexDirection={{ _: 'column' }}\n            flexWrap={isMultiple ? 'unset' : { _: 'nowrap', xs: 'wrap' }}\n            height={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: 188,\n                  }\n            }\n            width={\n              isMultiple\n                ? 'unset'\n                : {\n                    _: 'unset',\n                    xs: '100%',\n                  }\n            }\n          >\n            {planDetail.features.map((feature) => (\n              <PlanFeature\n                key={feature.label}\n                feature={feature}\n                isMultiple={isMultiple}\n              />\n            ))}\n          </StyledList>\n        </FlexBox>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            product={product}\n            ctaConfig={ctaConfig}\n            isCurrentPlan={isCurrentPlan}\n          />\n        </FlexBox>\n      </CardComponent>\n    </StyledListRow>\n  );\n};\n"]} */");
|
|
89
|
-
const
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
67
|
+
})(cardHeaderVariants, process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../src/PricingSection/PricingCard/index.tsx"],"names":[],"mappings":"AA6EmB","file":"../../../src/PricingSection/PricingCard/index.tsx","sourcesContent":["import { Badge, Box, Card, FlexBox, Text } from '@codecademy/gamut';\nimport { CheckerDense } from '@codecademy/gamut-patterns';\nimport { css, pxRem, theme, variant } from '@codecademy/gamut-styles';\nimport styled from '@emotion/styled';\n\nimport { ConditionalColorMode } from '../ConditionalColorMode';\nimport { getProductDescription, PlansByType } from '../config';\nimport { ProductCTA } from '../ProductCTA';\nimport { Product, ProductDetails, User } from '../types';\nimport { PricingAmount } from './PricingAmount';\nimport { RecommendedBadge } from './RecommendedBadge';\nimport { AllCurrency } from './types';\n\nexport const smSize = 728;\n\nconst StyledCard = styled(Card)(\n  css({\n    height: 1,\n    borderWidth: 0,\n    padding: 0,\n    width: '100%',\n    flexDirection: 'column',\n  })\n);\n\nconst Divider = styled(CheckerDense)(\n  css({\n    height: '0.1rem',\n  })\n);\n\ntype StyledListRowProps = {\n  showRecommendedBadge: boolean;\n};\n\nconst StyledListRow = styled('li')<StyledListRowProps>`\n  width: 100%;\n  max-width: ${pxRem(smSize)};\n  height: 100%;\n  display: grid;\n  ${theme.breakpoints.lg} {\n    min-width: 36rem;\n  }\n`;\n\nconst cardHeaderVariants = variant({\n  base: {\n    backgroundColor: 'transparent',\n  },\n  variants: {\n    [Product.Basic]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Enterprise]: {\n      borderBottomWidth: 1,\n      borderBottomStyle: 'solid',\n      borderBottomColor: theme.colors['navy-200'],\n      color: 'navy-500',\n    },\n    [Product.Silver]: {\n      backgroundColor: theme.colors.paleYellow,\n    },\n    [Product.Gold]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Pro]: {\n      backgroundColor: theme.colors.yellow,\n    },\n    [Product.Teams]: {\n      backgroundColor: theme.colors.background,\n    },\n  },\n});\n\nconst CardHeader = styled(FlexBox)(cardHeaderVariants);\n\nconst getBadgeText = (product: Product): string | null => {\n  switch (product) {\n    case Product.Gold:\n      return 'Advance your career';\n    case Product.Silver:\n    case Product.Pro:\n      return 'Learn a skill';\n    case Product.Teams:\n      return 'Recommended for 2+';\n    case Product.Enterprise:\n      return 'Recommended for 25+';\n    default:\n      return null;\n  }\n};\n\nconst renderProductDescription = (product: Product) => {\n  const productFeatures = (\n    <Text maxWidth={{ md: '235px' }} textAlign=\"center\">\n      {getProductDescription(product)}\n    </Text>\n  );\n\n  return (\n    <FlexBox mt={8} alignItems=\"center\" flexDirection=\"column\">\n      {productFeatures}\n    </FlexBox>\n  );\n};\n\nexport const PricingCard = ({\n  product,\n  productDetails,\n  currency,\n  plansByType,\n  isUserInIndia,\n  user,\n}: {\n  product: Product;\n  productDetails: ProductDetails;\n  currency: AllCurrency;\n  plansByType: PlansByType;\n  isUserInIndia: boolean;\n  user: User;\n}) => {\n  const isGoldProduct = product === Product.Gold;\n  const isTeamsProduct = product === Product.Teams;\n\n  return (\n    <StyledListRow showRecommendedBadge={product === Product.Gold}>\n      <StyledCard display=\"flex\" shadow=\"patternLeft\">\n        {isGoldProduct && <RecommendedBadge />}\n        <ConditionalColorMode condition={isTeamsProduct}>\n          <CardHeader\n            variant={product}\n            py={8}\n            px={24}\n            alignItems=\"center\"\n            justifyContent=\"space-between\"\n          >\n            <Text as=\"h2\" fontSize={{ _: 22, xl: 26 }}>\n              {plansByType[product].monthly?.displayName ||\n                plansByType[product].annual?.displayName}\n            </Text>\n            {isGoldProduct && <Text screenreader>Recommended</Text>}\n            {getBadgeText(product) && (\n              <Badge variant=\"tertiary\" ml={12}>\n                {getBadgeText(product)}\n              </Badge>\n            )}\n          </CardHeader>\n        </ConditionalColorMode>\n\n        <Box px={24} pt={24}>\n          <FlexBox center alignItems=\"flex-start\">\n            {product === Product.Enterprise ? (\n              <FlexBox\n                column\n                alignItems=\"center\"\n                justifyContent=\"center\"\n                minHeight={134}\n              >\n                <Text color=\"text-secondary\" mb={8} textAlign=\"center\">\n                  Looking to train a larger team?\n                </Text>\n                <Text variant=\"title-lg\" textAlign=\"center\">\n                  Contact us for a custom quote\n                </Text>\n              </FlexBox>\n            ) : (\n              <PricingAmount\n                monthlyPrice={plansByType[product].monthly?.price}\n                price={productDetails.price}\n                product={product}\n                currency={currency}\n                isUserInIndia={isUserInIndia}\n              />\n            )}\n          </FlexBox>\n\n          <FlexBox column alignItems=\"center\">\n            <Divider mt={16} />\n            <FlexBox\n              flexDirection=\"column\"\n              my={12}\n              /**\n               * Previously we were arbitrarily sizing cards to a width that was\n               * \"too small\" for the content they contained. The actual affect this was\n               * having was shrinking this section down to 256px, so rather than size\n               * the card, I'm opting to size this particular container w/i.\n               * This allows the cards themselves to be more flexible and simply styled\n               * while still maintaining the \"proper size\" on smaller screens.\n               */\n              width={{ _: 'unset', md: pxRem(256) }}\n            >\n              {renderProductDescription(product)}\n            </FlexBox>\n          </FlexBox>\n        </Box>\n        <FlexBox mb={24} mt=\"auto\" column>\n          <ProductCTA\n            user={user}\n            productDetails={productDetails}\n            product={product}\n          />\n        </FlexBox>\n      </StyledCard>\n    </StyledListRow>\n  );\n};\n"]} */");
|
|
68
|
+
const getBadgeText = product => {
|
|
69
|
+
switch (product) {
|
|
70
|
+
case Product.Gold:
|
|
71
|
+
return 'Advance your career';
|
|
72
|
+
case Product.Silver:
|
|
73
|
+
case Product.Pro:
|
|
74
|
+
return 'Learn a skill';
|
|
75
|
+
case Product.Teams:
|
|
76
|
+
return 'Recommended for 2+';
|
|
77
|
+
case Product.Enterprise:
|
|
78
|
+
return 'Recommended for 25+';
|
|
79
|
+
default:
|
|
80
|
+
return null;
|
|
101
81
|
}
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
82
|
+
};
|
|
83
|
+
const renderProductDescription = product => {
|
|
84
|
+
const productFeatures = /*#__PURE__*/_jsx(Text, {
|
|
85
|
+
maxWidth: {
|
|
86
|
+
md: '235px'
|
|
87
|
+
},
|
|
88
|
+
textAlign: "center",
|
|
89
|
+
children: getProductDescription(product)
|
|
90
|
+
});
|
|
91
|
+
return /*#__PURE__*/_jsx(FlexBox, {
|
|
92
|
+
mt: 8,
|
|
93
|
+
alignItems: "center",
|
|
94
|
+
flexDirection: "column",
|
|
95
|
+
children: productFeatures
|
|
96
|
+
});
|
|
109
97
|
};
|
|
110
98
|
export const PricingCard = ({
|
|
111
99
|
product,
|
|
112
100
|
productDetails,
|
|
101
|
+
currency,
|
|
113
102
|
plansByType,
|
|
114
103
|
isUserInIndia,
|
|
115
|
-
|
|
116
|
-
ctaConfig,
|
|
117
|
-
copyConfig
|
|
104
|
+
user
|
|
118
105
|
}) => {
|
|
119
106
|
const isGoldProduct = product === Product.Gold;
|
|
120
|
-
const
|
|
121
|
-
const isLegacyProduct = product === Product.Pro;
|
|
122
|
-
const planDetail = planDetails[product];
|
|
123
|
-
const CardComponent = !isGoldProduct ? StyledCard : StyledCardWithBorder;
|
|
124
|
-
const isMultiple = true;
|
|
107
|
+
const isTeamsProduct = product === Product.Teams;
|
|
125
108
|
return /*#__PURE__*/_jsx(StyledListRow, {
|
|
126
109
|
showRecommendedBadge: product === Product.Gold,
|
|
127
|
-
children: /*#__PURE__*/_jsxs(
|
|
128
|
-
position: "relative",
|
|
110
|
+
children: /*#__PURE__*/_jsxs(StyledCard, {
|
|
129
111
|
display: "flex",
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
112
|
+
shadow: "patternLeft",
|
|
113
|
+
children: [isGoldProduct && /*#__PURE__*/_jsx(RecommendedBadge, {}), /*#__PURE__*/_jsx(ConditionalColorMode, {
|
|
114
|
+
condition: isTeamsProduct,
|
|
115
|
+
children: /*#__PURE__*/_jsxs(CardHeader, {
|
|
116
|
+
variant: product,
|
|
117
|
+
py: 8,
|
|
118
|
+
px: 24,
|
|
119
|
+
alignItems: "center",
|
|
120
|
+
justifyContent: "space-between",
|
|
121
|
+
children: [/*#__PURE__*/_jsx(Text, {
|
|
122
|
+
as: "h2",
|
|
123
|
+
fontSize: {
|
|
124
|
+
_: 22,
|
|
125
|
+
xl: 26
|
|
126
|
+
},
|
|
127
|
+
children: plansByType[product].monthly?.displayName || plansByType[product].annual?.displayName
|
|
128
|
+
}), isGoldProduct && /*#__PURE__*/_jsx(Text, {
|
|
129
|
+
screenreader: true,
|
|
130
|
+
children: "Recommended"
|
|
131
|
+
}), getBadgeText(product) && /*#__PURE__*/_jsx(Badge, {
|
|
132
|
+
variant: "tertiary",
|
|
133
|
+
ml: 12,
|
|
134
|
+
children: getBadgeText(product)
|
|
135
|
+
})]
|
|
136
|
+
})
|
|
152
137
|
}), /*#__PURE__*/_jsxs(Box, {
|
|
153
138
|
px: 24,
|
|
154
139
|
pt: 24,
|
|
155
140
|
children: [/*#__PURE__*/_jsx(FlexBox, {
|
|
156
141
|
center: true,
|
|
157
142
|
alignItems: "flex-start",
|
|
158
|
-
children: /*#__PURE__*/
|
|
159
|
-
|
|
143
|
+
children: product === Product.Enterprise ? /*#__PURE__*/_jsxs(FlexBox, {
|
|
144
|
+
column: true,
|
|
145
|
+
alignItems: "center",
|
|
146
|
+
justifyContent: "center",
|
|
147
|
+
minHeight: 134,
|
|
148
|
+
children: [/*#__PURE__*/_jsx(Text, {
|
|
149
|
+
color: "text-secondary",
|
|
150
|
+
mb: 8,
|
|
151
|
+
textAlign: "center",
|
|
152
|
+
children: "Looking to train a larger team?"
|
|
153
|
+
}), /*#__PURE__*/_jsx(Text, {
|
|
154
|
+
variant: "title-lg",
|
|
155
|
+
textAlign: "center",
|
|
156
|
+
children: "Contact us for a custom quote"
|
|
157
|
+
})]
|
|
158
|
+
}) : /*#__PURE__*/_jsx(PricingAmount, {
|
|
159
|
+
monthlyPrice: plansByType[product].monthly?.price,
|
|
160
160
|
price: productDetails.price,
|
|
161
161
|
product: product,
|
|
162
|
+
currency: currency,
|
|
162
163
|
isUserInIndia: isUserInIndia
|
|
163
164
|
})
|
|
164
165
|
}), /*#__PURE__*/_jsxs(FlexBox, {
|
|
165
166
|
column: true,
|
|
166
167
|
alignItems: "center",
|
|
167
|
-
children: [/*#__PURE__*/_jsx(
|
|
168
|
-
mt: 16
|
|
169
|
-
|
|
170
|
-
}), /*#__PURE__*/_jsxs(FlexBox, {
|
|
168
|
+
children: [/*#__PURE__*/_jsx(Divider, {
|
|
169
|
+
mt: 16
|
|
170
|
+
}), /*#__PURE__*/_jsx(FlexBox, {
|
|
171
171
|
flexDirection: "column",
|
|
172
|
-
|
|
173
|
-
|
|
172
|
+
my: 12
|
|
173
|
+
/**
|
|
174
|
+
* Previously we were arbitrarily sizing cards to a width that was
|
|
175
|
+
* "too small" for the content they contained. The actual affect this was
|
|
176
|
+
* having was shrinking this section down to 256px, so rather than size
|
|
177
|
+
* the card, I'm opting to size this particular container w/i.
|
|
178
|
+
* This allows the cards themselves to be more flexible and simply styled
|
|
179
|
+
* while still maintaining the "proper size" on smaller screens.
|
|
180
|
+
*/,
|
|
174
181
|
width: {
|
|
175
182
|
_: 'unset',
|
|
176
183
|
md: pxRem(256)
|
|
177
184
|
},
|
|
178
|
-
children:
|
|
179
|
-
maxWidth: {
|
|
180
|
-
md: '235px'
|
|
181
|
-
},
|
|
182
|
-
textAlign: "center",
|
|
183
|
-
children: copyConfig?.description || getProductDescription(product)
|
|
184
|
-
}), renderIncludesLabel(product, plansByType)]
|
|
185
|
+
children: renderProductDescription(product)
|
|
185
186
|
})]
|
|
186
187
|
})]
|
|
187
|
-
}), /*#__PURE__*/_jsx(FlexBox, {
|
|
188
|
-
justifyContent: "flex-end",
|
|
189
|
-
alignContent: isMultiple ? 'flex-start' : 'unset',
|
|
190
|
-
alignItems: "center",
|
|
191
|
-
position: "relative",
|
|
192
|
-
px: isMultiple ? 24 : {
|
|
193
|
-
_: 40,
|
|
194
|
-
sm: 0
|
|
195
|
-
},
|
|
196
|
-
py: 24,
|
|
197
|
-
m: "auto",
|
|
198
|
-
flexDirection: "column",
|
|
199
|
-
flexWrap: "wrap",
|
|
200
|
-
flexGrow: isMultiple ? 1 : {
|
|
201
|
-
_: 1,
|
|
202
|
-
sm: 'unset'
|
|
203
|
-
},
|
|
204
|
-
width: isMultiple ? 'auto' : {
|
|
205
|
-
_: '100%',
|
|
206
|
-
sm: '65%'
|
|
207
|
-
},
|
|
208
|
-
children: /*#__PURE__*/_jsx(StyledList, {
|
|
209
|
-
flexDirection: {
|
|
210
|
-
_: 'column'
|
|
211
|
-
},
|
|
212
|
-
flexWrap: isMultiple ? 'unset' : {
|
|
213
|
-
_: 'nowrap',
|
|
214
|
-
xs: 'wrap'
|
|
215
|
-
},
|
|
216
|
-
height: isMultiple ? 'unset' : {
|
|
217
|
-
_: 'unset',
|
|
218
|
-
xs: 188
|
|
219
|
-
},
|
|
220
|
-
width: isMultiple ? 'unset' : {
|
|
221
|
-
_: 'unset',
|
|
222
|
-
xs: '100%'
|
|
223
|
-
},
|
|
224
|
-
children: planDetail.features.map(feature => /*#__PURE__*/_jsx(PlanFeature, {
|
|
225
|
-
feature: feature,
|
|
226
|
-
isMultiple: isMultiple
|
|
227
|
-
}, feature.label))
|
|
228
|
-
})
|
|
229
188
|
}), /*#__PURE__*/_jsx(FlexBox, {
|
|
230
189
|
mb: 24,
|
|
231
190
|
mt: "auto",
|
|
232
191
|
column: true,
|
|
233
192
|
children: /*#__PURE__*/_jsx(ProductCTA, {
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
193
|
+
user: user,
|
|
194
|
+
productDetails: productDetails,
|
|
195
|
+
product: product
|
|
237
196
|
})
|
|
238
197
|
})]
|
|
239
198
|
})
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
export declare enum Currency {
|
|
2
|
+
Aud = "AUD",
|
|
3
|
+
Cad = "CAD",
|
|
4
|
+
Eur = "EUR",
|
|
5
|
+
Gbp = "GBP",
|
|
6
|
+
Inr = "INR",
|
|
7
|
+
Nok = "NOK",
|
|
8
|
+
Pln = "PLN",
|
|
9
|
+
Sek = "SEK",
|
|
10
|
+
Usd = "USD"
|
|
11
|
+
}
|
|
12
|
+
export declare enum CFBCurrency {
|
|
13
|
+
Dkk = "DKK",
|
|
14
|
+
Brl = "BRL",
|
|
15
|
+
Bgn = "BGN",
|
|
16
|
+
Chf = "CHF",
|
|
17
|
+
Czk = "CZK",
|
|
18
|
+
Hrk = "HRK",
|
|
19
|
+
Huf = "HUF",
|
|
20
|
+
Mxn = "MXN",
|
|
21
|
+
Ron = "RON"
|
|
22
|
+
}
|
|
23
|
+
export declare enum CurrencySymbol {
|
|
24
|
+
Dollar = "$",
|
|
25
|
+
Pound = "\u00A3",
|
|
26
|
+
Euro = "\u20AC",
|
|
27
|
+
Rupee = "\u20B9",
|
|
28
|
+
Krona = "kr",
|
|
29
|
+
Krone = "kr",
|
|
30
|
+
ZLOTY = "z\u0142",
|
|
31
|
+
Lev = "\u041B\u0432.",
|
|
32
|
+
Real = "R$",
|
|
33
|
+
Franc = "Fr.",
|
|
34
|
+
Koruna = "K\u010D",
|
|
35
|
+
Kuna = "kn",
|
|
36
|
+
Forint = "Ft",
|
|
37
|
+
Leu = "lei"
|
|
38
|
+
}
|
|
39
|
+
export type AllCurrency = Currency | CFBCurrency;
|
|
40
|
+
export declare const getCurrencySymbol: (currency: AllCurrency) => CurrencySymbol;
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
/* eslint-disable @typescript-eslint/no-duplicate-enum-values */
|
|
2
|
+
export let Currency = /*#__PURE__*/function (Currency) {
|
|
3
|
+
Currency["Aud"] = "AUD";
|
|
4
|
+
Currency["Cad"] = "CAD";
|
|
5
|
+
Currency["Eur"] = "EUR";
|
|
6
|
+
Currency["Gbp"] = "GBP";
|
|
7
|
+
Currency["Inr"] = "INR";
|
|
8
|
+
Currency["Nok"] = "NOK";
|
|
9
|
+
Currency["Pln"] = "PLN";
|
|
10
|
+
Currency["Sek"] = "SEK";
|
|
11
|
+
Currency["Usd"] = "USD";
|
|
12
|
+
return Currency;
|
|
13
|
+
}({});
|
|
14
|
+
export let CFBCurrency = /*#__PURE__*/function (CFBCurrency) {
|
|
15
|
+
CFBCurrency["Dkk"] = "DKK";
|
|
16
|
+
CFBCurrency["Brl"] = "BRL";
|
|
17
|
+
CFBCurrency["Bgn"] = "BGN";
|
|
18
|
+
CFBCurrency["Chf"] = "CHF";
|
|
19
|
+
CFBCurrency["Czk"] = "CZK";
|
|
20
|
+
CFBCurrency["Hrk"] = "HRK";
|
|
21
|
+
CFBCurrency["Huf"] = "HUF";
|
|
22
|
+
CFBCurrency["Mxn"] = "MXN";
|
|
23
|
+
CFBCurrency["Ron"] = "RON";
|
|
24
|
+
return CFBCurrency;
|
|
25
|
+
}({});
|
|
26
|
+
export let CurrencySymbol = /*#__PURE__*/function (CurrencySymbol) {
|
|
27
|
+
CurrencySymbol["Dollar"] = "$";
|
|
28
|
+
CurrencySymbol["Pound"] = "\xA3";
|
|
29
|
+
CurrencySymbol["Euro"] = "\u20AC";
|
|
30
|
+
CurrencySymbol["Rupee"] = "\u20B9";
|
|
31
|
+
CurrencySymbol["Krona"] = "kr";
|
|
32
|
+
CurrencySymbol["Krone"] = "kr";
|
|
33
|
+
CurrencySymbol["ZLOTY"] = "z\u0142";
|
|
34
|
+
CurrencySymbol["Lev"] = "\u041B\u0432.";
|
|
35
|
+
CurrencySymbol["Real"] = "R$";
|
|
36
|
+
CurrencySymbol["Franc"] = "Fr.";
|
|
37
|
+
CurrencySymbol["Koruna"] = "K\u010D";
|
|
38
|
+
CurrencySymbol["Kuna"] = "kn";
|
|
39
|
+
CurrencySymbol["Forint"] = "Ft";
|
|
40
|
+
CurrencySymbol["Leu"] = "lei";
|
|
41
|
+
return CurrencySymbol;
|
|
42
|
+
}({});
|
|
43
|
+
const currencySymbols = {
|
|
44
|
+
[Currency.Usd]: CurrencySymbol.Dollar,
|
|
45
|
+
[Currency.Cad]: CurrencySymbol.Dollar,
|
|
46
|
+
[Currency.Aud]: CurrencySymbol.Dollar,
|
|
47
|
+
[Currency.Eur]: CurrencySymbol.Euro,
|
|
48
|
+
[Currency.Gbp]: CurrencySymbol.Pound,
|
|
49
|
+
[Currency.Inr]: CurrencySymbol.Rupee,
|
|
50
|
+
[Currency.Sek]: CurrencySymbol.Krona,
|
|
51
|
+
[Currency.Nok]: CurrencySymbol.Krone,
|
|
52
|
+
[Currency.Pln]: CurrencySymbol.ZLOTY,
|
|
53
|
+
[CFBCurrency.Bgn]: CurrencySymbol.Lev,
|
|
54
|
+
[CFBCurrency.Brl]: CurrencySymbol.Real,
|
|
55
|
+
[CFBCurrency.Chf]: CurrencySymbol.Franc,
|
|
56
|
+
[CFBCurrency.Czk]: CurrencySymbol.Koruna,
|
|
57
|
+
[CFBCurrency.Dkk]: CurrencySymbol.Krone,
|
|
58
|
+
[CFBCurrency.Hrk]: CurrencySymbol.Kuna,
|
|
59
|
+
[CFBCurrency.Huf]: CurrencySymbol.Forint,
|
|
60
|
+
[CFBCurrency.Mxn]: CurrencySymbol.Dollar,
|
|
61
|
+
[CFBCurrency.Ron]: CurrencySymbol.Leu
|
|
62
|
+
};
|
|
63
|
+
export const getCurrencySymbol = currency => currencySymbols[currency];
|