@ultraviolet/plus 0.15.1 → 0.15.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (96) hide show
  1. package/dist/components/CodeEditor/CodeEditor.cjs +39 -0
  2. package/dist/components/CodeEditor/CodeEditor.js +17 -24
  3. package/dist/components/ContentCard/Skeleton.cjs +43 -0
  4. package/dist/components/ContentCard/Skeleton.js +28 -21
  5. package/dist/components/ContentCard/index.cjs +154 -0
  6. package/dist/components/ContentCard/index.js +143 -149
  7. package/dist/components/ContentCardGroup/Card.cjs +93 -0
  8. package/dist/components/ContentCardGroup/Card.js +84 -76
  9. package/dist/components/ContentCardGroup/SkeletonCard.cjs +34 -0
  10. package/dist/components/ContentCardGroup/SkeletonCard.js +19 -13
  11. package/dist/components/ContentCardGroup/index.cjs +33 -0
  12. package/dist/components/ContentCardGroup/index.js +17 -8
  13. package/dist/components/Conversation/index.cjs +128 -0
  14. package/dist/components/Conversation/index.js +100 -42
  15. package/dist/components/CustomerSatisfaction/assets/1-5.svg.cjs +3 -0
  16. package/dist/components/CustomerSatisfaction/assets/1-5NB.svg.cjs +3 -0
  17. package/dist/components/CustomerSatisfaction/assets/2-5.svg.cjs +3 -0
  18. package/dist/components/CustomerSatisfaction/assets/2-5NB.svg.cjs +3 -0
  19. package/dist/components/CustomerSatisfaction/assets/3-5.svg.cjs +3 -0
  20. package/dist/components/CustomerSatisfaction/assets/3-5NB.svg.cjs +3 -0
  21. package/dist/components/CustomerSatisfaction/assets/4-5.svg.cjs +3 -0
  22. package/dist/components/CustomerSatisfaction/assets/4-5NB.svg.cjs +3 -0
  23. package/dist/components/CustomerSatisfaction/assets/5-5.svg.cjs +3 -0
  24. package/dist/components/CustomerSatisfaction/assets/5-5NB.svg.cjs +3 -0
  25. package/dist/components/CustomerSatisfaction/index.cjs +123 -0
  26. package/dist/components/CustomerSatisfaction/index.js +58 -81
  27. package/dist/components/EstimateCost/Components/CustomUnitInput.cjs +33 -0
  28. package/dist/components/EstimateCost/Components/CustomUnitInput.js +19 -34
  29. package/dist/components/EstimateCost/Components/Item.cjs +303 -0
  30. package/dist/components/EstimateCost/Components/Item.js +182 -199
  31. package/dist/components/EstimateCost/Components/LineThrough.cjs +18 -0
  32. package/dist/components/EstimateCost/Components/LineThrough.js +12 -9
  33. package/dist/components/EstimateCost/Components/NumberInput.cjs +29 -0
  34. package/dist/components/EstimateCost/Components/NumberInput.js +8 -15
  35. package/dist/components/EstimateCost/Components/Region.cjs +41 -0
  36. package/dist/components/EstimateCost/Components/Region.js +16 -27
  37. package/dist/components/EstimateCost/Components/Regular.cjs +35 -0
  38. package/dist/components/EstimateCost/Components/Regular.js +20 -28
  39. package/dist/components/EstimateCost/Components/Strong.cjs +28 -0
  40. package/dist/components/EstimateCost/Components/Strong.js +15 -12
  41. package/dist/components/EstimateCost/Components/Unit.cjs +50 -0
  42. package/dist/components/EstimateCost/Components/Unit.js +27 -33
  43. package/dist/components/EstimateCost/Components/UnitInput.cjs +90 -0
  44. package/dist/components/EstimateCost/Components/UnitInput.js +39 -87
  45. package/dist/components/EstimateCost/Components/Zone.cjs +40 -0
  46. package/dist/components/EstimateCost/Components/Zone.js +16 -26
  47. package/dist/components/EstimateCost/EstimateCost.cjs +88 -0
  48. package/dist/components/EstimateCost/EstimateCost.js +24 -63
  49. package/dist/components/EstimateCost/EstimateCostContent.cjs +245 -0
  50. package/dist/components/EstimateCost/EstimateCostContent.js +148 -208
  51. package/dist/components/EstimateCost/EstimateCostProvider.cjs +36 -0
  52. package/dist/components/EstimateCost/EstimateCostProvider.js +20 -23
  53. package/dist/components/EstimateCost/OverlayComponent.cjs +122 -0
  54. package/dist/components/EstimateCost/OverlayComponent.js +79 -73
  55. package/dist/components/EstimateCost/OverlayContext.cjs +14 -0
  56. package/dist/components/EstimateCost/OverlayContext.js +3 -1
  57. package/dist/components/EstimateCost/componentStyle.cjs +230 -0
  58. package/dist/components/EstimateCost/componentStyle.js +206 -148
  59. package/dist/components/EstimateCost/constants.cjs +30 -0
  60. package/dist/components/EstimateCost/helper.cjs +19 -0
  61. package/dist/components/EstimateCost/locales/en.cjs +23 -0
  62. package/dist/components/FAQ/index.cjs +36 -0
  63. package/dist/components/FAQ/index.js +9 -5
  64. package/dist/components/Navigation/Navigation.cjs +20 -0
  65. package/dist/components/Navigation/Navigation.d.ts +4 -3
  66. package/dist/components/Navigation/Navigation.js +1 -10
  67. package/dist/components/Navigation/NavigationContent.cjs +216 -0
  68. package/dist/components/Navigation/NavigationContent.js +96 -146
  69. package/dist/components/Navigation/NavigationProvider.cjs +111 -0
  70. package/dist/components/Navigation/NavigationProvider.d.ts +7 -18
  71. package/dist/components/Navigation/NavigationProvider.js +67 -98
  72. package/dist/components/Navigation/components/Group.cjs +57 -0
  73. package/dist/components/Navigation/components/Group.js +37 -41
  74. package/dist/components/Navigation/components/Item.cjs +438 -0
  75. package/dist/components/Navigation/components/Item.d.ts +4 -3
  76. package/dist/components/Navigation/components/Item.js +296 -555
  77. package/dist/components/Navigation/components/PinnedItems.cjs +108 -0
  78. package/dist/components/Navigation/components/PinnedItems.d.ts +5 -1
  79. package/dist/components/Navigation/components/PinnedItems.js +81 -119
  80. package/dist/components/Navigation/components/Separator.cjs +17 -0
  81. package/dist/components/Navigation/components/Separator.js +9 -4
  82. package/dist/components/Navigation/constants.cjs +37 -0
  83. package/dist/components/Navigation/constants.d.ts +14 -0
  84. package/dist/components/Navigation/index.d.ts +1 -0
  85. package/dist/components/Navigation/locales/en.cjs +11 -0
  86. package/dist/components/SteppedListCard/Step.cjs +37 -0
  87. package/dist/components/SteppedListCard/Step.js +19 -38
  88. package/dist/components/SteppedListCard/SteppedListContainer.cjs +80 -0
  89. package/dist/components/SteppedListCard/SteppedListContainer.js +39 -48
  90. package/dist/components/SteppedListCard/SteppedListContent.cjs +67 -0
  91. package/dist/components/SteppedListCard/SteppedListContent.js +46 -24
  92. package/dist/components/SteppedListCard/helper.cjs +37 -0
  93. package/dist/components/SteppedListCard/helper.js +1 -3
  94. package/dist/components/index.d.ts +1 -0
  95. package/dist/index.cjs +25 -0
  96. package/package.json +24 -9
@@ -1,5 +1,5 @@
1
1
  import { jsxs, jsx, Fragment } from "@emotion/react/jsx-runtime";
2
- import styled from "@emotion/styled";
2
+ import _styled from "@emotion/styled/base";
3
3
  import { Text, Icon, Stack, Alert } from "@ultraviolet/ui";
4
4
  import { useState, useMemo, useEffect, Children, isValidElement, cloneElement } from "react";
5
5
  import { useInView } from "react-intersection-observer";
@@ -13,23 +13,61 @@ import { Cell, PriceCell, StyledTable, PriceCol, Title, TimeCell, EmptyTable, Em
13
13
  import { maximumFractionDigitsLong, maximumFractionDigits } from "./constants.js";
14
14
  import { calculatePrice } from "./helper.js";
15
15
  import EstimateCostLocales from "./locales/en.js";
16
- const FeesText = styled(Text)`
17
- margin-top: ${({ theme }) => theme.space["3"]};
18
- `;
19
- const StyledText = styled(Text)`
20
- text-align: right;
21
- ${({ isBeta, theme }) => isBeta ? `margin-left: ${theme.space["2"]};` : null}
22
- `;
23
- const RightAlignedText = styled(Text)`
24
- text-align: right;
25
- `;
26
- const StyledIcon = styled(Icon)`
27
- margin-right: ${({ theme }) => theme.space["1"]};
28
- `;
29
- const StyledPriceCell = styled(Cell.withComponent("th"))`
30
- ${({ theme }) => PriceCell(theme)}
31
- padding: 0;
32
- `;
16
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() {
17
+ 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).";
18
+ }
19
+ const FeesText = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
20
+ target: "excc3v74"
21
+ } : {
22
+ target: "excc3v74",
23
+ label: "FeesText"
24
+ })("margin-top:", ({
25
+ theme
26
+ }) => theme.space["3"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx"],"names":[],"mappings":"AA0C6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react'\nimport { useInView } from 'react-intersection-observer'\nimport { CustomUnitInput } from './Components/CustomUnitInput'\nimport { Item } from './Components/Item'\nimport { LineThrough } from './Components/LineThrough'\nimport { useEstimateCost } from './EstimateCostProvider'\nimport { OverlayComponent } from './OverlayComponent'\nimport { OverlayContextProvider } from './OverlayContext'\nimport {\n  BadgeBeta,\n  Cell,\n  EmptyCell,\n  EmptyTable,\n  PriceCell,\n  PriceCol,\n  StyledFeesTable,\n  StyledTable,\n  TimeCell,\n  Title,\n  TotalPriceCell,\n} from './componentStyle'\nimport { maximumFractionDigits, maximumFractionDigitsLong } from './constants'\nimport { calculatePrice } from './helper'\nimport EstimateCostLocales from './locales/en'\nimport type {\n  BareEstimateProduct,\n  EstimateCostProps,\n  EstimateProduct,\n  Iteration,\n  Units,\n} from './types'\n\nconst FeesText = styled(Text)`\n  margin-top: ${({ theme }) => theme.space['3']};\n`\n\nconst StyledText = styled(Text)<{\n  isBeta?: boolean\n}>`\n  text-align: right;\n  ${({ isBeta, theme }) =>\n    isBeta ? `margin-left: ${theme.space['2']};` : null}\n`\n\nconst RightAlignedText = styled(Text)`\n  text-align: right;\n`\n\nconst StyledIcon = styled(Icon)`\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledPriceCell = styled(Cell.withComponent('th'))`\n  ${({ theme }) => PriceCell(theme)}\n  padding: 0;\n`\n\nconst DEFAULT_UNIT_LIST: Units[] = ['hours', 'days', 'months']\n\ntype ExtraProps = {\n  isLastElement?: boolean\n  productsCallback?: {\n    add: (product: EstimateProduct) => void\n    remove: (product: BareEstimateProduct) => void\n  }\n  iteration?: Iteration\n  discount?: number\n}\n\nconst DescriptionComponent = ({\n  description,\n  locales,\n}: {\n  description: ReactNode\n  locales: typeof EstimateCostLocales\n}) =>\n  description === undefined || typeof description === 'string' ? (\n    <Text as=\"span\" variant=\"body\">\n      {description || locales['estimate.cost.description']}\n    </Text>\n  ) : (\n    description\n  )\n\nexport const EstimateCostContent = ({\n  description,\n  alert,\n  alertTitle,\n  alertVariant = 'warning',\n  defaultTimeUnit = 'hours',\n  timeUnits = DEFAULT_UNIT_LIST,\n  hideOverlay = false,\n  disableOverlayLeft = false,\n  disableOverlayRight = false,\n  hideTimeUnit = false,\n  hideTotal = false,\n  hideHourlyPriceOnTotal = false,\n  discount = 0,\n  OverlayRight,\n  OverlayLeft,\n  isBeta = false,\n  commitmentFees,\n  commitmentFeesContent,\n  monthlyFees,\n  monthlyFeesLabel,\n  monthlyFeesContent,\n  overlayUnit = 'hours',\n  children = null,\n  locales = EstimateCostLocales,\n  overlayMargin,\n}: EstimateCostProps) => {\n  const { formatNumber } = useEstimateCost()\n  const [ref, inView] = useInView()\n  const [products, setProducts] = useState<EstimateProduct[]>([]) // product is used to store each items with their price and amount\n  const [totalPrice, setTotalPrice] = useState({\n    overlayHourly: 0,\n    maxOverlayHourly: 0,\n    hourly: 0,\n    maxHourly: 0,\n    total: 0,\n    maxTotal: 0,\n  })\n  const [iteration, setIteration] = useState<Iteration>({\n    value: 1,\n    unit: defaultTimeUnit ?? 'hours',\n  })\n\n  const [isLongFractionDigits, setIsLongFractionDigits] = useState(false)\n  const providerValue = useMemo(() => ({ isOverlay: false }), [])\n\n  const productsCallback = useMemo(\n    () => ({\n      add: (newProduct: EstimateProduct) => {\n        setProducts(total => {\n          if (total.find(product => product.id === newProduct.id)) {\n            return total.map(product =>\n              product.id === newProduct.id ? newProduct : product,\n            )\n          }\n\n          return [...total, newProduct]\n        })\n      },\n\n      remove: ({ id }: BareEstimateProduct) => {\n        setProducts(total => total.filter(product => product.id !== id))\n      },\n    }),\n    [setProducts],\n  )\n\n  useEffect(() => {\n    // this variable check if there is a maxAmount in each product\n    // if not we do not need to calculate maxTotal, maxHourly, maxOverlayHourly\n    const isMaxAmountInProducts = products.find(product => product.maxAmount)\n    setIsLongFractionDigits(\n      !!products.find(product => product.longFractionDigits),\n    )\n    setTotalPrice({\n      total: !hideTotal\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.amount,\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      maxTotal: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.maxAmount || product.amount, // Not all products have maxAmount, so we need to check both\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      hourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc && product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0),\n\n            0,\n          )\n        : 0,\n      overlayHourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxOverlayHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              (product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0)),\n            0,\n          )\n        : 0,\n    })\n  }, [hideTotal, products, iteration, setTotalPrice])\n\n  useEffect(() => {\n    if (\n      hideTimeUnit &&\n      (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? 'hours'))\n    ) {\n      setIteration({ unit: defaultTimeUnit ?? 'hours', value: 1 })\n    }\n  }, [hideTimeUnit, iteration, defaultTimeUnit])\n\n  return (\n    <Stack gap={2}>\n      {!hideOverlay ? (\n        <OverlayComponent\n          inView={inView}\n          totalPrice={totalPrice}\n          disableOverlayLeft={disableOverlayLeft}\n          disableOverlayRight={disableOverlayRight}\n          OverlayLeft={OverlayLeft}\n          OverlayRight={OverlayRight}\n          isBeta={isBeta}\n          discount={discount}\n          unit={overlayUnit ?? 'hours'}\n          overlayMargin={overlayMargin}\n        >\n          {children}\n        </OverlayComponent>\n      ) : null}\n      {description === false ? null : (\n        <DescriptionComponent description={description} locales={locales} />\n      )}\n      {alert ? (\n        <Alert sentiment={alertVariant} title={alertTitle}>\n          {alert}\n        </Alert>\n      ) : null}\n      <OverlayContextProvider value={providerValue}>\n        <div>\n          {children ? (\n            <StyledTable\n              cellPadding=\"0\"\n              cellSpacing=\"0\"\n              ref={ref}\n              data-testid=\"summary\"\n              noTotal={hideTotal}\n            >\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              {!hideTimeUnit ? (\n                <thead>\n                  <tr>\n                    <th>\n                      <Title>\n                        <StyledIcon\n                          name=\"calculator\"\n                          color=\"primary\"\n                          size={20}\n                        />\n                        {locales['estimate.cost.label']}\n                      </Title>\n                    </th>\n                    <StyledPriceCell>\n                      <TimeCell>\n                        <CustomUnitInput\n                          defaultTimeUnit={defaultTimeUnit}\n                          setIteration={setIteration}\n                          iteration={iteration}\n                          timeUnits={timeUnits}\n                        />\n                      </TimeCell>\n                    </StyledPriceCell>\n                  </tr>\n                </thead>\n              ) : null}\n              <tbody>\n                {Children.map(children, (child, index) =>\n                  isValidElement<ExtraProps>(child)\n                    ? cloneElement(child, {\n                        isLastElement: index === Children.count(children) - 1,\n                        productsCallback,\n                        iteration,\n                        discount:\n                          discount &&\n                          !(\n                            (\n                              child as {\n                                props: Record<string, unknown>\n                              }\n                            ).props as {\n                              discount?: number\n                            }\n                          ).discount\n                            ? discount\n                            : (\n                                (\n                                  child as {\n                                    props: Record<string, unknown>\n                                  }\n                                ).props as {\n                                  discount?: number\n                                }\n                              ).discount,\n                      })\n                    : child,\n                )}\n              </tbody>\n            </StyledTable>\n          ) : null}\n          {!hideTotal ? (\n            <EmptyTable cellPadding=\"0\" cellSpacing=\"0\">\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              <tbody>\n                <tr>\n                  <EmptyCell aria-label=\"control\" />\n                  <TotalPriceCell hasBorder={false}>\n                    {isBeta ? (\n                      <BadgeBeta\n                        prominence=\"strong\"\n                        long={\n                          locales[\n                            `estimate.cost.beta.${\n                              discount > 0 ? 'discount' : 'free'\n                            }`\n                          ].length > 25\n                        }\n                        sentiment=\"warning\"\n                      >\n                        {`${discount > 0 ? discount * 100 : ''}\n                          ${\n                            locales[\n                              `estimate.cost.beta.${\n                                discount > 0 ? 'discount' : 'free'\n                              }`\n                            ]\n                          }`}\n                      </BadgeBeta>\n                    ) : null}\n                    <StyledText\n                      as=\"h3\"\n                      variant=\"heading\"\n                      color=\"primary\"\n                      isBeta={isBeta}\n                    >\n                      <LineThrough\n                        isActive={isBeta && (discount === 0 || discount >= 1)}\n                      >\n                        {formatNumber(totalPrice.total, {\n                          maximumFractionDigits: isLongFractionDigits\n                            ? maximumFractionDigitsLong[iteration.unit]\n                            : maximumFractionDigits[iteration.unit],\n                        })}\n                        {totalPrice.maxTotal > 0\n                          ? ` - ${formatNumber(totalPrice.maxTotal, {\n                              maximumFractionDigits: isLongFractionDigits\n                                ? maximumFractionDigitsLong[iteration.unit]\n                                : maximumFractionDigits[iteration.unit],\n                            })}`\n                          : null}\n                      </LineThrough>\n                    </StyledText>\n                    {hideHourlyPriceOnTotal &&\n                    totalPrice.hourly > 0 &&\n                    totalPrice.hourly !== totalPrice.total &&\n                    totalPrice.total > 0 ? (\n                      <RightAlignedText as=\"p\" variant=\"body\">\n                        <LineThrough\n                          isActive={isBeta && (discount === 0 || discount >= 1)}\n                        >\n                          {formatNumber(totalPrice.hourly, {\n                            maximumFractionDigits: isLongFractionDigits\n                              ? maximumFractionDigitsLong.hours\n                              : maximumFractionDigits.hours,\n                          })}\n                          {totalPrice.maxHourly > 0\n                            ? ` - ${formatNumber(totalPrice.maxHourly, {\n                                maximumFractionDigits: isLongFractionDigits\n                                  ? maximumFractionDigitsLong.hours\n                                  : maximumFractionDigits.hours,\n                              })}`\n                            : null}\n                          /\n                          {locales[\n                            `estimate.cost.units.hours.label`\n                          ].toLowerCase()}\n                        </LineThrough>\n                      </RightAlignedText>\n                    ) : null}\n                  </TotalPriceCell>\n                </tr>\n              </tbody>\n            </EmptyTable>\n          ) : null}\n          {commitmentFees !== undefined || monthlyFees !== undefined ? (\n            <>\n              <FeesText as=\"h3\" variant=\"headingSmall\">\n                {\n                  locales[\n                    `estimate.cost.fees.${\n                      commitmentFees ? 'oneTime' : 'monthly'\n                    }.title`\n                  ]\n                }\n              </FeesText>\n              <StyledFeesTable>\n                <tbody>\n                  <Item\n                    label={\n                      commitmentFees\n                        ? locales['estimate.cost.fees.commitment']\n                        : monthlyFeesLabel\n                    }\n                    noIteration\n                    isLastElement\n                    price={commitmentFees || monthlyFees}\n                    productsCallback={{\n                      add: () => {},\n                      remove: () => {},\n                    }}\n                  >\n                    {commitmentFees\n                      ? commitmentFeesContent\n                      : monthlyFeesContent}\n                  </Item>\n                </tbody>\n              </StyledFeesTable>\n            </>\n          ) : null}\n        </div>\n      </OverlayContextProvider>\n    </Stack>\n  )\n}\n"]} */"));
27
+ const StyledText = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
28
+ target: "excc3v73"
29
+ } : {
30
+ target: "excc3v73",
31
+ label: "StyledText"
32
+ })("text-align:right;", ({
33
+ isBeta,
34
+ theme
35
+ }) => isBeta ? `margin-left: ${theme.space["2"]};` : null, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx"],"names":[],"mappings":"AAgDE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react'\nimport { useInView } from 'react-intersection-observer'\nimport { CustomUnitInput } from './Components/CustomUnitInput'\nimport { Item } from './Components/Item'\nimport { LineThrough } from './Components/LineThrough'\nimport { useEstimateCost } from './EstimateCostProvider'\nimport { OverlayComponent } from './OverlayComponent'\nimport { OverlayContextProvider } from './OverlayContext'\nimport {\n  BadgeBeta,\n  Cell,\n  EmptyCell,\n  EmptyTable,\n  PriceCell,\n  PriceCol,\n  StyledFeesTable,\n  StyledTable,\n  TimeCell,\n  Title,\n  TotalPriceCell,\n} from './componentStyle'\nimport { maximumFractionDigits, maximumFractionDigitsLong } from './constants'\nimport { calculatePrice } from './helper'\nimport EstimateCostLocales from './locales/en'\nimport type {\n  BareEstimateProduct,\n  EstimateCostProps,\n  EstimateProduct,\n  Iteration,\n  Units,\n} from './types'\n\nconst FeesText = styled(Text)`\n  margin-top: ${({ theme }) => theme.space['3']};\n`\n\nconst StyledText = styled(Text)<{\n  isBeta?: boolean\n}>`\n  text-align: right;\n  ${({ isBeta, theme }) =>\n    isBeta ? `margin-left: ${theme.space['2']};` : null}\n`\n\nconst RightAlignedText = styled(Text)`\n  text-align: right;\n`\n\nconst StyledIcon = styled(Icon)`\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledPriceCell = styled(Cell.withComponent('th'))`\n  ${({ theme }) => PriceCell(theme)}\n  padding: 0;\n`\n\nconst DEFAULT_UNIT_LIST: Units[] = ['hours', 'days', 'months']\n\ntype ExtraProps = {\n  isLastElement?: boolean\n  productsCallback?: {\n    add: (product: EstimateProduct) => void\n    remove: (product: BareEstimateProduct) => void\n  }\n  iteration?: Iteration\n  discount?: number\n}\n\nconst DescriptionComponent = ({\n  description,\n  locales,\n}: {\n  description: ReactNode\n  locales: typeof EstimateCostLocales\n}) =>\n  description === undefined || typeof description === 'string' ? (\n    <Text as=\"span\" variant=\"body\">\n      {description || locales['estimate.cost.description']}\n    </Text>\n  ) : (\n    description\n  )\n\nexport const EstimateCostContent = ({\n  description,\n  alert,\n  alertTitle,\n  alertVariant = 'warning',\n  defaultTimeUnit = 'hours',\n  timeUnits = DEFAULT_UNIT_LIST,\n  hideOverlay = false,\n  disableOverlayLeft = false,\n  disableOverlayRight = false,\n  hideTimeUnit = false,\n  hideTotal = false,\n  hideHourlyPriceOnTotal = false,\n  discount = 0,\n  OverlayRight,\n  OverlayLeft,\n  isBeta = false,\n  commitmentFees,\n  commitmentFeesContent,\n  monthlyFees,\n  monthlyFeesLabel,\n  monthlyFeesContent,\n  overlayUnit = 'hours',\n  children = null,\n  locales = EstimateCostLocales,\n  overlayMargin,\n}: EstimateCostProps) => {\n  const { formatNumber } = useEstimateCost()\n  const [ref, inView] = useInView()\n  const [products, setProducts] = useState<EstimateProduct[]>([]) // product is used to store each items with their price and amount\n  const [totalPrice, setTotalPrice] = useState({\n    overlayHourly: 0,\n    maxOverlayHourly: 0,\n    hourly: 0,\n    maxHourly: 0,\n    total: 0,\n    maxTotal: 0,\n  })\n  const [iteration, setIteration] = useState<Iteration>({\n    value: 1,\n    unit: defaultTimeUnit ?? 'hours',\n  })\n\n  const [isLongFractionDigits, setIsLongFractionDigits] = useState(false)\n  const providerValue = useMemo(() => ({ isOverlay: false }), [])\n\n  const productsCallback = useMemo(\n    () => ({\n      add: (newProduct: EstimateProduct) => {\n        setProducts(total => {\n          if (total.find(product => product.id === newProduct.id)) {\n            return total.map(product =>\n              product.id === newProduct.id ? newProduct : product,\n            )\n          }\n\n          return [...total, newProduct]\n        })\n      },\n\n      remove: ({ id }: BareEstimateProduct) => {\n        setProducts(total => total.filter(product => product.id !== id))\n      },\n    }),\n    [setProducts],\n  )\n\n  useEffect(() => {\n    // this variable check if there is a maxAmount in each product\n    // if not we do not need to calculate maxTotal, maxHourly, maxOverlayHourly\n    const isMaxAmountInProducts = products.find(product => product.maxAmount)\n    setIsLongFractionDigits(\n      !!products.find(product => product.longFractionDigits),\n    )\n    setTotalPrice({\n      total: !hideTotal\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.amount,\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      maxTotal: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.maxAmount || product.amount, // Not all products have maxAmount, so we need to check both\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      hourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc && product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0),\n\n            0,\n          )\n        : 0,\n      overlayHourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxOverlayHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              (product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0)),\n            0,\n          )\n        : 0,\n    })\n  }, [hideTotal, products, iteration, setTotalPrice])\n\n  useEffect(() => {\n    if (\n      hideTimeUnit &&\n      (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? 'hours'))\n    ) {\n      setIteration({ unit: defaultTimeUnit ?? 'hours', value: 1 })\n    }\n  }, [hideTimeUnit, iteration, defaultTimeUnit])\n\n  return (\n    <Stack gap={2}>\n      {!hideOverlay ? (\n        <OverlayComponent\n          inView={inView}\n          totalPrice={totalPrice}\n          disableOverlayLeft={disableOverlayLeft}\n          disableOverlayRight={disableOverlayRight}\n          OverlayLeft={OverlayLeft}\n          OverlayRight={OverlayRight}\n          isBeta={isBeta}\n          discount={discount}\n          unit={overlayUnit ?? 'hours'}\n          overlayMargin={overlayMargin}\n        >\n          {children}\n        </OverlayComponent>\n      ) : null}\n      {description === false ? null : (\n        <DescriptionComponent description={description} locales={locales} />\n      )}\n      {alert ? (\n        <Alert sentiment={alertVariant} title={alertTitle}>\n          {alert}\n        </Alert>\n      ) : null}\n      <OverlayContextProvider value={providerValue}>\n        <div>\n          {children ? (\n            <StyledTable\n              cellPadding=\"0\"\n              cellSpacing=\"0\"\n              ref={ref}\n              data-testid=\"summary\"\n              noTotal={hideTotal}\n            >\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              {!hideTimeUnit ? (\n                <thead>\n                  <tr>\n                    <th>\n                      <Title>\n                        <StyledIcon\n                          name=\"calculator\"\n                          color=\"primary\"\n                          size={20}\n                        />\n                        {locales['estimate.cost.label']}\n                      </Title>\n                    </th>\n                    <StyledPriceCell>\n                      <TimeCell>\n                        <CustomUnitInput\n                          defaultTimeUnit={defaultTimeUnit}\n                          setIteration={setIteration}\n                          iteration={iteration}\n                          timeUnits={timeUnits}\n                        />\n                      </TimeCell>\n                    </StyledPriceCell>\n                  </tr>\n                </thead>\n              ) : null}\n              <tbody>\n                {Children.map(children, (child, index) =>\n                  isValidElement<ExtraProps>(child)\n                    ? cloneElement(child, {\n                        isLastElement: index === Children.count(children) - 1,\n                        productsCallback,\n                        iteration,\n                        discount:\n                          discount &&\n                          !(\n                            (\n                              child as {\n                                props: Record<string, unknown>\n                              }\n                            ).props as {\n                              discount?: number\n                            }\n                          ).discount\n                            ? discount\n                            : (\n                                (\n                                  child as {\n                                    props: Record<string, unknown>\n                                  }\n                                ).props as {\n                                  discount?: number\n                                }\n                              ).discount,\n                      })\n                    : child,\n                )}\n              </tbody>\n            </StyledTable>\n          ) : null}\n          {!hideTotal ? (\n            <EmptyTable cellPadding=\"0\" cellSpacing=\"0\">\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              <tbody>\n                <tr>\n                  <EmptyCell aria-label=\"control\" />\n                  <TotalPriceCell hasBorder={false}>\n                    {isBeta ? (\n                      <BadgeBeta\n                        prominence=\"strong\"\n                        long={\n                          locales[\n                            `estimate.cost.beta.${\n                              discount > 0 ? 'discount' : 'free'\n                            }`\n                          ].length > 25\n                        }\n                        sentiment=\"warning\"\n                      >\n                        {`${discount > 0 ? discount * 100 : ''}\n                          ${\n                            locales[\n                              `estimate.cost.beta.${\n                                discount > 0 ? 'discount' : 'free'\n                              }`\n                            ]\n                          }`}\n                      </BadgeBeta>\n                    ) : null}\n                    <StyledText\n                      as=\"h3\"\n                      variant=\"heading\"\n                      color=\"primary\"\n                      isBeta={isBeta}\n                    >\n                      <LineThrough\n                        isActive={isBeta && (discount === 0 || discount >= 1)}\n                      >\n                        {formatNumber(totalPrice.total, {\n                          maximumFractionDigits: isLongFractionDigits\n                            ? maximumFractionDigitsLong[iteration.unit]\n                            : maximumFractionDigits[iteration.unit],\n                        })}\n                        {totalPrice.maxTotal > 0\n                          ? ` - ${formatNumber(totalPrice.maxTotal, {\n                              maximumFractionDigits: isLongFractionDigits\n                                ? maximumFractionDigitsLong[iteration.unit]\n                                : maximumFractionDigits[iteration.unit],\n                            })}`\n                          : null}\n                      </LineThrough>\n                    </StyledText>\n                    {hideHourlyPriceOnTotal &&\n                    totalPrice.hourly > 0 &&\n                    totalPrice.hourly !== totalPrice.total &&\n                    totalPrice.total > 0 ? (\n                      <RightAlignedText as=\"p\" variant=\"body\">\n                        <LineThrough\n                          isActive={isBeta && (discount === 0 || discount >= 1)}\n                        >\n                          {formatNumber(totalPrice.hourly, {\n                            maximumFractionDigits: isLongFractionDigits\n                              ? maximumFractionDigitsLong.hours\n                              : maximumFractionDigits.hours,\n                          })}\n                          {totalPrice.maxHourly > 0\n                            ? ` - ${formatNumber(totalPrice.maxHourly, {\n                                maximumFractionDigits: isLongFractionDigits\n                                  ? maximumFractionDigitsLong.hours\n                                  : maximumFractionDigits.hours,\n                              })}`\n                            : null}\n                          /\n                          {locales[\n                            `estimate.cost.units.hours.label`\n                          ].toLowerCase()}\n                        </LineThrough>\n                      </RightAlignedText>\n                    ) : null}\n                  </TotalPriceCell>\n                </tr>\n              </tbody>\n            </EmptyTable>\n          ) : null}\n          {commitmentFees !== undefined || monthlyFees !== undefined ? (\n            <>\n              <FeesText as=\"h3\" variant=\"headingSmall\">\n                {\n                  locales[\n                    `estimate.cost.fees.${\n                      commitmentFees ? 'oneTime' : 'monthly'\n                    }.title`\n                  ]\n                }\n              </FeesText>\n              <StyledFeesTable>\n                <tbody>\n                  <Item\n                    label={\n                      commitmentFees\n                        ? locales['estimate.cost.fees.commitment']\n                        : monthlyFeesLabel\n                    }\n                    noIteration\n                    isLastElement\n                    price={commitmentFees || monthlyFees}\n                    productsCallback={{\n                      add: () => {},\n                      remove: () => {},\n                    }}\n                  >\n                    {commitmentFees\n                      ? commitmentFeesContent\n                      : monthlyFeesContent}\n                  </Item>\n                </tbody>\n              </StyledFeesTable>\n            </>\n          ) : null}\n        </div>\n      </OverlayContextProvider>\n    </Stack>\n  )\n}\n"]} */"));
36
+ const RightAlignedText = /* @__PURE__ */ _styled(Text, process.env.NODE_ENV === "production" ? {
37
+ target: "excc3v72"
38
+ } : {
39
+ target: "excc3v72",
40
+ label: "RightAlignedText"
41
+ })(process.env.NODE_ENV === "production" ? {
42
+ name: "2qga7i",
43
+ styles: "text-align:right"
44
+ } : {
45
+ name: "2qga7i",
46
+ styles: "text-align:right",
47
+ map: "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx"],"names":[],"mappings":"AAsDqC","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react'\nimport { useInView } from 'react-intersection-observer'\nimport { CustomUnitInput } from './Components/CustomUnitInput'\nimport { Item } from './Components/Item'\nimport { LineThrough } from './Components/LineThrough'\nimport { useEstimateCost } from './EstimateCostProvider'\nimport { OverlayComponent } from './OverlayComponent'\nimport { OverlayContextProvider } from './OverlayContext'\nimport {\n  BadgeBeta,\n  Cell,\n  EmptyCell,\n  EmptyTable,\n  PriceCell,\n  PriceCol,\n  StyledFeesTable,\n  StyledTable,\n  TimeCell,\n  Title,\n  TotalPriceCell,\n} from './componentStyle'\nimport { maximumFractionDigits, maximumFractionDigitsLong } from './constants'\nimport { calculatePrice } from './helper'\nimport EstimateCostLocales from './locales/en'\nimport type {\n  BareEstimateProduct,\n  EstimateCostProps,\n  EstimateProduct,\n  Iteration,\n  Units,\n} from './types'\n\nconst FeesText = styled(Text)`\n  margin-top: ${({ theme }) => theme.space['3']};\n`\n\nconst StyledText = styled(Text)<{\n  isBeta?: boolean\n}>`\n  text-align: right;\n  ${({ isBeta, theme }) =>\n    isBeta ? `margin-left: ${theme.space['2']};` : null}\n`\n\nconst RightAlignedText = styled(Text)`\n  text-align: right;\n`\n\nconst StyledIcon = styled(Icon)`\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledPriceCell = styled(Cell.withComponent('th'))`\n  ${({ theme }) => PriceCell(theme)}\n  padding: 0;\n`\n\nconst DEFAULT_UNIT_LIST: Units[] = ['hours', 'days', 'months']\n\ntype ExtraProps = {\n  isLastElement?: boolean\n  productsCallback?: {\n    add: (product: EstimateProduct) => void\n    remove: (product: BareEstimateProduct) => void\n  }\n  iteration?: Iteration\n  discount?: number\n}\n\nconst DescriptionComponent = ({\n  description,\n  locales,\n}: {\n  description: ReactNode\n  locales: typeof EstimateCostLocales\n}) =>\n  description === undefined || typeof description === 'string' ? (\n    <Text as=\"span\" variant=\"body\">\n      {description || locales['estimate.cost.description']}\n    </Text>\n  ) : (\n    description\n  )\n\nexport const EstimateCostContent = ({\n  description,\n  alert,\n  alertTitle,\n  alertVariant = 'warning',\n  defaultTimeUnit = 'hours',\n  timeUnits = DEFAULT_UNIT_LIST,\n  hideOverlay = false,\n  disableOverlayLeft = false,\n  disableOverlayRight = false,\n  hideTimeUnit = false,\n  hideTotal = false,\n  hideHourlyPriceOnTotal = false,\n  discount = 0,\n  OverlayRight,\n  OverlayLeft,\n  isBeta = false,\n  commitmentFees,\n  commitmentFeesContent,\n  monthlyFees,\n  monthlyFeesLabel,\n  monthlyFeesContent,\n  overlayUnit = 'hours',\n  children = null,\n  locales = EstimateCostLocales,\n  overlayMargin,\n}: EstimateCostProps) => {\n  const { formatNumber } = useEstimateCost()\n  const [ref, inView] = useInView()\n  const [products, setProducts] = useState<EstimateProduct[]>([]) // product is used to store each items with their price and amount\n  const [totalPrice, setTotalPrice] = useState({\n    overlayHourly: 0,\n    maxOverlayHourly: 0,\n    hourly: 0,\n    maxHourly: 0,\n    total: 0,\n    maxTotal: 0,\n  })\n  const [iteration, setIteration] = useState<Iteration>({\n    value: 1,\n    unit: defaultTimeUnit ?? 'hours',\n  })\n\n  const [isLongFractionDigits, setIsLongFractionDigits] = useState(false)\n  const providerValue = useMemo(() => ({ isOverlay: false }), [])\n\n  const productsCallback = useMemo(\n    () => ({\n      add: (newProduct: EstimateProduct) => {\n        setProducts(total => {\n          if (total.find(product => product.id === newProduct.id)) {\n            return total.map(product =>\n              product.id === newProduct.id ? newProduct : product,\n            )\n          }\n\n          return [...total, newProduct]\n        })\n      },\n\n      remove: ({ id }: BareEstimateProduct) => {\n        setProducts(total => total.filter(product => product.id !== id))\n      },\n    }),\n    [setProducts],\n  )\n\n  useEffect(() => {\n    // this variable check if there is a maxAmount in each product\n    // if not we do not need to calculate maxTotal, maxHourly, maxOverlayHourly\n    const isMaxAmountInProducts = products.find(product => product.maxAmount)\n    setIsLongFractionDigits(\n      !!products.find(product => product.longFractionDigits),\n    )\n    setTotalPrice({\n      total: !hideTotal\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.amount,\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      maxTotal: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.maxAmount || product.amount, // Not all products have maxAmount, so we need to check both\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      hourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc && product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0),\n\n            0,\n          )\n        : 0,\n      overlayHourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxOverlayHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              (product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0)),\n            0,\n          )\n        : 0,\n    })\n  }, [hideTotal, products, iteration, setTotalPrice])\n\n  useEffect(() => {\n    if (\n      hideTimeUnit &&\n      (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? 'hours'))\n    ) {\n      setIteration({ unit: defaultTimeUnit ?? 'hours', value: 1 })\n    }\n  }, [hideTimeUnit, iteration, defaultTimeUnit])\n\n  return (\n    <Stack gap={2}>\n      {!hideOverlay ? (\n        <OverlayComponent\n          inView={inView}\n          totalPrice={totalPrice}\n          disableOverlayLeft={disableOverlayLeft}\n          disableOverlayRight={disableOverlayRight}\n          OverlayLeft={OverlayLeft}\n          OverlayRight={OverlayRight}\n          isBeta={isBeta}\n          discount={discount}\n          unit={overlayUnit ?? 'hours'}\n          overlayMargin={overlayMargin}\n        >\n          {children}\n        </OverlayComponent>\n      ) : null}\n      {description === false ? null : (\n        <DescriptionComponent description={description} locales={locales} />\n      )}\n      {alert ? (\n        <Alert sentiment={alertVariant} title={alertTitle}>\n          {alert}\n        </Alert>\n      ) : null}\n      <OverlayContextProvider value={providerValue}>\n        <div>\n          {children ? (\n            <StyledTable\n              cellPadding=\"0\"\n              cellSpacing=\"0\"\n              ref={ref}\n              data-testid=\"summary\"\n              noTotal={hideTotal}\n            >\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              {!hideTimeUnit ? (\n                <thead>\n                  <tr>\n                    <th>\n                      <Title>\n                        <StyledIcon\n                          name=\"calculator\"\n                          color=\"primary\"\n                          size={20}\n                        />\n                        {locales['estimate.cost.label']}\n                      </Title>\n                    </th>\n                    <StyledPriceCell>\n                      <TimeCell>\n                        <CustomUnitInput\n                          defaultTimeUnit={defaultTimeUnit}\n                          setIteration={setIteration}\n                          iteration={iteration}\n                          timeUnits={timeUnits}\n                        />\n                      </TimeCell>\n                    </StyledPriceCell>\n                  </tr>\n                </thead>\n              ) : null}\n              <tbody>\n                {Children.map(children, (child, index) =>\n                  isValidElement<ExtraProps>(child)\n                    ? cloneElement(child, {\n                        isLastElement: index === Children.count(children) - 1,\n                        productsCallback,\n                        iteration,\n                        discount:\n                          discount &&\n                          !(\n                            (\n                              child as {\n                                props: Record<string, unknown>\n                              }\n                            ).props as {\n                              discount?: number\n                            }\n                          ).discount\n                            ? discount\n                            : (\n                                (\n                                  child as {\n                                    props: Record<string, unknown>\n                                  }\n                                ).props as {\n                                  discount?: number\n                                }\n                              ).discount,\n                      })\n                    : child,\n                )}\n              </tbody>\n            </StyledTable>\n          ) : null}\n          {!hideTotal ? (\n            <EmptyTable cellPadding=\"0\" cellSpacing=\"0\">\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              <tbody>\n                <tr>\n                  <EmptyCell aria-label=\"control\" />\n                  <TotalPriceCell hasBorder={false}>\n                    {isBeta ? (\n                      <BadgeBeta\n                        prominence=\"strong\"\n                        long={\n                          locales[\n                            `estimate.cost.beta.${\n                              discount > 0 ? 'discount' : 'free'\n                            }`\n                          ].length > 25\n                        }\n                        sentiment=\"warning\"\n                      >\n                        {`${discount > 0 ? discount * 100 : ''}\n                          ${\n                            locales[\n                              `estimate.cost.beta.${\n                                discount > 0 ? 'discount' : 'free'\n                              }`\n                            ]\n                          }`}\n                      </BadgeBeta>\n                    ) : null}\n                    <StyledText\n                      as=\"h3\"\n                      variant=\"heading\"\n                      color=\"primary\"\n                      isBeta={isBeta}\n                    >\n                      <LineThrough\n                        isActive={isBeta && (discount === 0 || discount >= 1)}\n                      >\n                        {formatNumber(totalPrice.total, {\n                          maximumFractionDigits: isLongFractionDigits\n                            ? maximumFractionDigitsLong[iteration.unit]\n                            : maximumFractionDigits[iteration.unit],\n                        })}\n                        {totalPrice.maxTotal > 0\n                          ? ` - ${formatNumber(totalPrice.maxTotal, {\n                              maximumFractionDigits: isLongFractionDigits\n                                ? maximumFractionDigitsLong[iteration.unit]\n                                : maximumFractionDigits[iteration.unit],\n                            })}`\n                          : null}\n                      </LineThrough>\n                    </StyledText>\n                    {hideHourlyPriceOnTotal &&\n                    totalPrice.hourly > 0 &&\n                    totalPrice.hourly !== totalPrice.total &&\n                    totalPrice.total > 0 ? (\n                      <RightAlignedText as=\"p\" variant=\"body\">\n                        <LineThrough\n                          isActive={isBeta && (discount === 0 || discount >= 1)}\n                        >\n                          {formatNumber(totalPrice.hourly, {\n                            maximumFractionDigits: isLongFractionDigits\n                              ? maximumFractionDigitsLong.hours\n                              : maximumFractionDigits.hours,\n                          })}\n                          {totalPrice.maxHourly > 0\n                            ? ` - ${formatNumber(totalPrice.maxHourly, {\n                                maximumFractionDigits: isLongFractionDigits\n                                  ? maximumFractionDigitsLong.hours\n                                  : maximumFractionDigits.hours,\n                              })}`\n                            : null}\n                          /\n                          {locales[\n                            `estimate.cost.units.hours.label`\n                          ].toLowerCase()}\n                        </LineThrough>\n                      </RightAlignedText>\n                    ) : null}\n                  </TotalPriceCell>\n                </tr>\n              </tbody>\n            </EmptyTable>\n          ) : null}\n          {commitmentFees !== undefined || monthlyFees !== undefined ? (\n            <>\n              <FeesText as=\"h3\" variant=\"headingSmall\">\n                {\n                  locales[\n                    `estimate.cost.fees.${\n                      commitmentFees ? 'oneTime' : 'monthly'\n                    }.title`\n                  ]\n                }\n              </FeesText>\n              <StyledFeesTable>\n                <tbody>\n                  <Item\n                    label={\n                      commitmentFees\n                        ? locales['estimate.cost.fees.commitment']\n                        : monthlyFeesLabel\n                    }\n                    noIteration\n                    isLastElement\n                    price={commitmentFees || monthlyFees}\n                    productsCallback={{\n                      add: () => {},\n                      remove: () => {},\n                    }}\n                  >\n                    {commitmentFees\n                      ? commitmentFeesContent\n                      : monthlyFeesContent}\n                  </Item>\n                </tbody>\n              </StyledFeesTable>\n            </>\n          ) : null}\n        </div>\n      </OverlayContextProvider>\n    </Stack>\n  )\n}\n"]} */",
48
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
49
+ });
50
+ const StyledIcon = /* @__PURE__ */ _styled(Icon, process.env.NODE_ENV === "production" ? {
51
+ target: "excc3v71"
52
+ } : {
53
+ target: "excc3v71",
54
+ label: "StyledIcon"
55
+ })("margin-right:", ({
56
+ theme
57
+ }) => theme.space["1"], ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx"],"names":[],"mappings":"AA0D+B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react'\nimport { useInView } from 'react-intersection-observer'\nimport { CustomUnitInput } from './Components/CustomUnitInput'\nimport { Item } from './Components/Item'\nimport { LineThrough } from './Components/LineThrough'\nimport { useEstimateCost } from './EstimateCostProvider'\nimport { OverlayComponent } from './OverlayComponent'\nimport { OverlayContextProvider } from './OverlayContext'\nimport {\n  BadgeBeta,\n  Cell,\n  EmptyCell,\n  EmptyTable,\n  PriceCell,\n  PriceCol,\n  StyledFeesTable,\n  StyledTable,\n  TimeCell,\n  Title,\n  TotalPriceCell,\n} from './componentStyle'\nimport { maximumFractionDigits, maximumFractionDigitsLong } from './constants'\nimport { calculatePrice } from './helper'\nimport EstimateCostLocales from './locales/en'\nimport type {\n  BareEstimateProduct,\n  EstimateCostProps,\n  EstimateProduct,\n  Iteration,\n  Units,\n} from './types'\n\nconst FeesText = styled(Text)`\n  margin-top: ${({ theme }) => theme.space['3']};\n`\n\nconst StyledText = styled(Text)<{\n  isBeta?: boolean\n}>`\n  text-align: right;\n  ${({ isBeta, theme }) =>\n    isBeta ? `margin-left: ${theme.space['2']};` : null}\n`\n\nconst RightAlignedText = styled(Text)`\n  text-align: right;\n`\n\nconst StyledIcon = styled(Icon)`\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledPriceCell = styled(Cell.withComponent('th'))`\n  ${({ theme }) => PriceCell(theme)}\n  padding: 0;\n`\n\nconst DEFAULT_UNIT_LIST: Units[] = ['hours', 'days', 'months']\n\ntype ExtraProps = {\n  isLastElement?: boolean\n  productsCallback?: {\n    add: (product: EstimateProduct) => void\n    remove: (product: BareEstimateProduct) => void\n  }\n  iteration?: Iteration\n  discount?: number\n}\n\nconst DescriptionComponent = ({\n  description,\n  locales,\n}: {\n  description: ReactNode\n  locales: typeof EstimateCostLocales\n}) =>\n  description === undefined || typeof description === 'string' ? (\n    <Text as=\"span\" variant=\"body\">\n      {description || locales['estimate.cost.description']}\n    </Text>\n  ) : (\n    description\n  )\n\nexport const EstimateCostContent = ({\n  description,\n  alert,\n  alertTitle,\n  alertVariant = 'warning',\n  defaultTimeUnit = 'hours',\n  timeUnits = DEFAULT_UNIT_LIST,\n  hideOverlay = false,\n  disableOverlayLeft = false,\n  disableOverlayRight = false,\n  hideTimeUnit = false,\n  hideTotal = false,\n  hideHourlyPriceOnTotal = false,\n  discount = 0,\n  OverlayRight,\n  OverlayLeft,\n  isBeta = false,\n  commitmentFees,\n  commitmentFeesContent,\n  monthlyFees,\n  monthlyFeesLabel,\n  monthlyFeesContent,\n  overlayUnit = 'hours',\n  children = null,\n  locales = EstimateCostLocales,\n  overlayMargin,\n}: EstimateCostProps) => {\n  const { formatNumber } = useEstimateCost()\n  const [ref, inView] = useInView()\n  const [products, setProducts] = useState<EstimateProduct[]>([]) // product is used to store each items with their price and amount\n  const [totalPrice, setTotalPrice] = useState({\n    overlayHourly: 0,\n    maxOverlayHourly: 0,\n    hourly: 0,\n    maxHourly: 0,\n    total: 0,\n    maxTotal: 0,\n  })\n  const [iteration, setIteration] = useState<Iteration>({\n    value: 1,\n    unit: defaultTimeUnit ?? 'hours',\n  })\n\n  const [isLongFractionDigits, setIsLongFractionDigits] = useState(false)\n  const providerValue = useMemo(() => ({ isOverlay: false }), [])\n\n  const productsCallback = useMemo(\n    () => ({\n      add: (newProduct: EstimateProduct) => {\n        setProducts(total => {\n          if (total.find(product => product.id === newProduct.id)) {\n            return total.map(product =>\n              product.id === newProduct.id ? newProduct : product,\n            )\n          }\n\n          return [...total, newProduct]\n        })\n      },\n\n      remove: ({ id }: BareEstimateProduct) => {\n        setProducts(total => total.filter(product => product.id !== id))\n      },\n    }),\n    [setProducts],\n  )\n\n  useEffect(() => {\n    // this variable check if there is a maxAmount in each product\n    // if not we do not need to calculate maxTotal, maxHourly, maxOverlayHourly\n    const isMaxAmountInProducts = products.find(product => product.maxAmount)\n    setIsLongFractionDigits(\n      !!products.find(product => product.longFractionDigits),\n    )\n    setTotalPrice({\n      total: !hideTotal\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.amount,\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      maxTotal: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.maxAmount || product.amount, // Not all products have maxAmount, so we need to check both\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      hourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc && product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0),\n\n            0,\n          )\n        : 0,\n      overlayHourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxOverlayHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              (product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0)),\n            0,\n          )\n        : 0,\n    })\n  }, [hideTotal, products, iteration, setTotalPrice])\n\n  useEffect(() => {\n    if (\n      hideTimeUnit &&\n      (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? 'hours'))\n    ) {\n      setIteration({ unit: defaultTimeUnit ?? 'hours', value: 1 })\n    }\n  }, [hideTimeUnit, iteration, defaultTimeUnit])\n\n  return (\n    <Stack gap={2}>\n      {!hideOverlay ? (\n        <OverlayComponent\n          inView={inView}\n          totalPrice={totalPrice}\n          disableOverlayLeft={disableOverlayLeft}\n          disableOverlayRight={disableOverlayRight}\n          OverlayLeft={OverlayLeft}\n          OverlayRight={OverlayRight}\n          isBeta={isBeta}\n          discount={discount}\n          unit={overlayUnit ?? 'hours'}\n          overlayMargin={overlayMargin}\n        >\n          {children}\n        </OverlayComponent>\n      ) : null}\n      {description === false ? null : (\n        <DescriptionComponent description={description} locales={locales} />\n      )}\n      {alert ? (\n        <Alert sentiment={alertVariant} title={alertTitle}>\n          {alert}\n        </Alert>\n      ) : null}\n      <OverlayContextProvider value={providerValue}>\n        <div>\n          {children ? (\n            <StyledTable\n              cellPadding=\"0\"\n              cellSpacing=\"0\"\n              ref={ref}\n              data-testid=\"summary\"\n              noTotal={hideTotal}\n            >\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              {!hideTimeUnit ? (\n                <thead>\n                  <tr>\n                    <th>\n                      <Title>\n                        <StyledIcon\n                          name=\"calculator\"\n                          color=\"primary\"\n                          size={20}\n                        />\n                        {locales['estimate.cost.label']}\n                      </Title>\n                    </th>\n                    <StyledPriceCell>\n                      <TimeCell>\n                        <CustomUnitInput\n                          defaultTimeUnit={defaultTimeUnit}\n                          setIteration={setIteration}\n                          iteration={iteration}\n                          timeUnits={timeUnits}\n                        />\n                      </TimeCell>\n                    </StyledPriceCell>\n                  </tr>\n                </thead>\n              ) : null}\n              <tbody>\n                {Children.map(children, (child, index) =>\n                  isValidElement<ExtraProps>(child)\n                    ? cloneElement(child, {\n                        isLastElement: index === Children.count(children) - 1,\n                        productsCallback,\n                        iteration,\n                        discount:\n                          discount &&\n                          !(\n                            (\n                              child as {\n                                props: Record<string, unknown>\n                              }\n                            ).props as {\n                              discount?: number\n                            }\n                          ).discount\n                            ? discount\n                            : (\n                                (\n                                  child as {\n                                    props: Record<string, unknown>\n                                  }\n                                ).props as {\n                                  discount?: number\n                                }\n                              ).discount,\n                      })\n                    : child,\n                )}\n              </tbody>\n            </StyledTable>\n          ) : null}\n          {!hideTotal ? (\n            <EmptyTable cellPadding=\"0\" cellSpacing=\"0\">\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              <tbody>\n                <tr>\n                  <EmptyCell aria-label=\"control\" />\n                  <TotalPriceCell hasBorder={false}>\n                    {isBeta ? (\n                      <BadgeBeta\n                        prominence=\"strong\"\n                        long={\n                          locales[\n                            `estimate.cost.beta.${\n                              discount > 0 ? 'discount' : 'free'\n                            }`\n                          ].length > 25\n                        }\n                        sentiment=\"warning\"\n                      >\n                        {`${discount > 0 ? discount * 100 : ''}\n                          ${\n                            locales[\n                              `estimate.cost.beta.${\n                                discount > 0 ? 'discount' : 'free'\n                              }`\n                            ]\n                          }`}\n                      </BadgeBeta>\n                    ) : null}\n                    <StyledText\n                      as=\"h3\"\n                      variant=\"heading\"\n                      color=\"primary\"\n                      isBeta={isBeta}\n                    >\n                      <LineThrough\n                        isActive={isBeta && (discount === 0 || discount >= 1)}\n                      >\n                        {formatNumber(totalPrice.total, {\n                          maximumFractionDigits: isLongFractionDigits\n                            ? maximumFractionDigitsLong[iteration.unit]\n                            : maximumFractionDigits[iteration.unit],\n                        })}\n                        {totalPrice.maxTotal > 0\n                          ? ` - ${formatNumber(totalPrice.maxTotal, {\n                              maximumFractionDigits: isLongFractionDigits\n                                ? maximumFractionDigitsLong[iteration.unit]\n                                : maximumFractionDigits[iteration.unit],\n                            })}`\n                          : null}\n                      </LineThrough>\n                    </StyledText>\n                    {hideHourlyPriceOnTotal &&\n                    totalPrice.hourly > 0 &&\n                    totalPrice.hourly !== totalPrice.total &&\n                    totalPrice.total > 0 ? (\n                      <RightAlignedText as=\"p\" variant=\"body\">\n                        <LineThrough\n                          isActive={isBeta && (discount === 0 || discount >= 1)}\n                        >\n                          {formatNumber(totalPrice.hourly, {\n                            maximumFractionDigits: isLongFractionDigits\n                              ? maximumFractionDigitsLong.hours\n                              : maximumFractionDigits.hours,\n                          })}\n                          {totalPrice.maxHourly > 0\n                            ? ` - ${formatNumber(totalPrice.maxHourly, {\n                                maximumFractionDigits: isLongFractionDigits\n                                  ? maximumFractionDigitsLong.hours\n                                  : maximumFractionDigits.hours,\n                              })}`\n                            : null}\n                          /\n                          {locales[\n                            `estimate.cost.units.hours.label`\n                          ].toLowerCase()}\n                        </LineThrough>\n                      </RightAlignedText>\n                    ) : null}\n                  </TotalPriceCell>\n                </tr>\n              </tbody>\n            </EmptyTable>\n          ) : null}\n          {commitmentFees !== undefined || monthlyFees !== undefined ? (\n            <>\n              <FeesText as=\"h3\" variant=\"headingSmall\">\n                {\n                  locales[\n                    `estimate.cost.fees.${\n                      commitmentFees ? 'oneTime' : 'monthly'\n                    }.title`\n                  ]\n                }\n              </FeesText>\n              <StyledFeesTable>\n                <tbody>\n                  <Item\n                    label={\n                      commitmentFees\n                        ? locales['estimate.cost.fees.commitment']\n                        : monthlyFeesLabel\n                    }\n                    noIteration\n                    isLastElement\n                    price={commitmentFees || monthlyFees}\n                    productsCallback={{\n                      add: () => {},\n                      remove: () => {},\n                    }}\n                  >\n                    {commitmentFees\n                      ? commitmentFeesContent\n                      : monthlyFeesContent}\n                  </Item>\n                </tbody>\n              </StyledFeesTable>\n            </>\n          ) : null}\n        </div>\n      </OverlayContextProvider>\n    </Stack>\n  )\n}\n"]} */"));
58
+ const StyledPriceCell = /* @__PURE__ */ _styled(Cell.withComponent("th", process.env.NODE_ENV === "production" ? {
59
+ target: "excc3v75"
60
+ } : {
61
+ target: "excc3v75",
62
+ label: "StyledPriceCell"
63
+ }), process.env.NODE_ENV === "production" ? {
64
+ target: "excc3v70"
65
+ } : {
66
+ target: "excc3v70",
67
+ label: "StyledPriceCell"
68
+ })(({
69
+ theme
70
+ }) => PriceCell(theme), " padding:0;" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx"],"names":[],"mappings":"AA8DwD","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["import styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  useEffect,\n  useMemo,\n  useState,\n} from 'react'\nimport { useInView } from 'react-intersection-observer'\nimport { CustomUnitInput } from './Components/CustomUnitInput'\nimport { Item } from './Components/Item'\nimport { LineThrough } from './Components/LineThrough'\nimport { useEstimateCost } from './EstimateCostProvider'\nimport { OverlayComponent } from './OverlayComponent'\nimport { OverlayContextProvider } from './OverlayContext'\nimport {\n  BadgeBeta,\n  Cell,\n  EmptyCell,\n  EmptyTable,\n  PriceCell,\n  PriceCol,\n  StyledFeesTable,\n  StyledTable,\n  TimeCell,\n  Title,\n  TotalPriceCell,\n} from './componentStyle'\nimport { maximumFractionDigits, maximumFractionDigitsLong } from './constants'\nimport { calculatePrice } from './helper'\nimport EstimateCostLocales from './locales/en'\nimport type {\n  BareEstimateProduct,\n  EstimateCostProps,\n  EstimateProduct,\n  Iteration,\n  Units,\n} from './types'\n\nconst FeesText = styled(Text)`\n  margin-top: ${({ theme }) => theme.space['3']};\n`\n\nconst StyledText = styled(Text)<{\n  isBeta?: boolean\n}>`\n  text-align: right;\n  ${({ isBeta, theme }) =>\n    isBeta ? `margin-left: ${theme.space['2']};` : null}\n`\n\nconst RightAlignedText = styled(Text)`\n  text-align: right;\n`\n\nconst StyledIcon = styled(Icon)`\n  margin-right: ${({ theme }) => theme.space['1']};\n`\n\nconst StyledPriceCell = styled(Cell.withComponent('th'))`\n  ${({ theme }) => PriceCell(theme)}\n  padding: 0;\n`\n\nconst DEFAULT_UNIT_LIST: Units[] = ['hours', 'days', 'months']\n\ntype ExtraProps = {\n  isLastElement?: boolean\n  productsCallback?: {\n    add: (product: EstimateProduct) => void\n    remove: (product: BareEstimateProduct) => void\n  }\n  iteration?: Iteration\n  discount?: number\n}\n\nconst DescriptionComponent = ({\n  description,\n  locales,\n}: {\n  description: ReactNode\n  locales: typeof EstimateCostLocales\n}) =>\n  description === undefined || typeof description === 'string' ? (\n    <Text as=\"span\" variant=\"body\">\n      {description || locales['estimate.cost.description']}\n    </Text>\n  ) : (\n    description\n  )\n\nexport const EstimateCostContent = ({\n  description,\n  alert,\n  alertTitle,\n  alertVariant = 'warning',\n  defaultTimeUnit = 'hours',\n  timeUnits = DEFAULT_UNIT_LIST,\n  hideOverlay = false,\n  disableOverlayLeft = false,\n  disableOverlayRight = false,\n  hideTimeUnit = false,\n  hideTotal = false,\n  hideHourlyPriceOnTotal = false,\n  discount = 0,\n  OverlayRight,\n  OverlayLeft,\n  isBeta = false,\n  commitmentFees,\n  commitmentFeesContent,\n  monthlyFees,\n  monthlyFeesLabel,\n  monthlyFeesContent,\n  overlayUnit = 'hours',\n  children = null,\n  locales = EstimateCostLocales,\n  overlayMargin,\n}: EstimateCostProps) => {\n  const { formatNumber } = useEstimateCost()\n  const [ref, inView] = useInView()\n  const [products, setProducts] = useState<EstimateProduct[]>([]) // product is used to store each items with their price and amount\n  const [totalPrice, setTotalPrice] = useState({\n    overlayHourly: 0,\n    maxOverlayHourly: 0,\n    hourly: 0,\n    maxHourly: 0,\n    total: 0,\n    maxTotal: 0,\n  })\n  const [iteration, setIteration] = useState<Iteration>({\n    value: 1,\n    unit: defaultTimeUnit ?? 'hours',\n  })\n\n  const [isLongFractionDigits, setIsLongFractionDigits] = useState(false)\n  const providerValue = useMemo(() => ({ isOverlay: false }), [])\n\n  const productsCallback = useMemo(\n    () => ({\n      add: (newProduct: EstimateProduct) => {\n        setProducts(total => {\n          if (total.find(product => product.id === newProduct.id)) {\n            return total.map(product =>\n              product.id === newProduct.id ? newProduct : product,\n            )\n          }\n\n          return [...total, newProduct]\n        })\n      },\n\n      remove: ({ id }: BareEstimateProduct) => {\n        setProducts(total => total.filter(product => product.id !== id))\n      },\n    }),\n    [setProducts],\n  )\n\n  useEffect(() => {\n    // this variable check if there is a maxAmount in each product\n    // if not we do not need to calculate maxTotal, maxHourly, maxOverlayHourly\n    const isMaxAmountInProducts = products.find(product => product.maxAmount)\n    setIsLongFractionDigits(\n      !!products.find(product => product.longFractionDigits),\n    )\n    setTotalPrice({\n      total: !hideTotal\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.amount,\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      maxTotal: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              calculatePrice({\n                price: product.price,\n                amount: product.maxAmount || product.amount, // Not all products have maxAmount, so we need to check both\n                amountFree: product.amountFree,\n                timeUnit: product.noIteration ? 'hours' : iteration.unit,\n                timeAmount: product.noIteration ? 1 : iteration.value,\n                discount: product.discount,\n              }),\n            0,\n          )\n        : 0,\n      hourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc && product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0),\n\n            0,\n          )\n        : 0,\n      overlayHourly: products.reduce(\n        (acc, product) =>\n          acc +\n          (product.noIteration\n            ? 0\n            : (product.price - product.price * product.discount) *\n              Math.max(product.amount - product.amountFree, 0)),\n        0,\n      ),\n      maxOverlayHourly: isMaxAmountInProducts\n        ? products.reduce(\n            (acc, product) =>\n              acc +\n              (product.noIteration\n                ? 0\n                : (product.price - product.price * product.discount) *\n                  Math.max(product.maxAmount - product.amountFree, 0)),\n            0,\n          )\n        : 0,\n    })\n  }, [hideTotal, products, iteration, setTotalPrice])\n\n  useEffect(() => {\n    if (\n      hideTimeUnit &&\n      (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? 'hours'))\n    ) {\n      setIteration({ unit: defaultTimeUnit ?? 'hours', value: 1 })\n    }\n  }, [hideTimeUnit, iteration, defaultTimeUnit])\n\n  return (\n    <Stack gap={2}>\n      {!hideOverlay ? (\n        <OverlayComponent\n          inView={inView}\n          totalPrice={totalPrice}\n          disableOverlayLeft={disableOverlayLeft}\n          disableOverlayRight={disableOverlayRight}\n          OverlayLeft={OverlayLeft}\n          OverlayRight={OverlayRight}\n          isBeta={isBeta}\n          discount={discount}\n          unit={overlayUnit ?? 'hours'}\n          overlayMargin={overlayMargin}\n        >\n          {children}\n        </OverlayComponent>\n      ) : null}\n      {description === false ? null : (\n        <DescriptionComponent description={description} locales={locales} />\n      )}\n      {alert ? (\n        <Alert sentiment={alertVariant} title={alertTitle}>\n          {alert}\n        </Alert>\n      ) : null}\n      <OverlayContextProvider value={providerValue}>\n        <div>\n          {children ? (\n            <StyledTable\n              cellPadding=\"0\"\n              cellSpacing=\"0\"\n              ref={ref}\n              data-testid=\"summary\"\n              noTotal={hideTotal}\n            >\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              {!hideTimeUnit ? (\n                <thead>\n                  <tr>\n                    <th>\n                      <Title>\n                        <StyledIcon\n                          name=\"calculator\"\n                          color=\"primary\"\n                          size={20}\n                        />\n                        {locales['estimate.cost.label']}\n                      </Title>\n                    </th>\n                    <StyledPriceCell>\n                      <TimeCell>\n                        <CustomUnitInput\n                          defaultTimeUnit={defaultTimeUnit}\n                          setIteration={setIteration}\n                          iteration={iteration}\n                          timeUnits={timeUnits}\n                        />\n                      </TimeCell>\n                    </StyledPriceCell>\n                  </tr>\n                </thead>\n              ) : null}\n              <tbody>\n                {Children.map(children, (child, index) =>\n                  isValidElement<ExtraProps>(child)\n                    ? cloneElement(child, {\n                        isLastElement: index === Children.count(children) - 1,\n                        productsCallback,\n                        iteration,\n                        discount:\n                          discount &&\n                          !(\n                            (\n                              child as {\n                                props: Record<string, unknown>\n                              }\n                            ).props as {\n                              discount?: number\n                            }\n                          ).discount\n                            ? discount\n                            : (\n                                (\n                                  child as {\n                                    props: Record<string, unknown>\n                                  }\n                                ).props as {\n                                  discount?: number\n                                }\n                              ).discount,\n                      })\n                    : child,\n                )}\n              </tbody>\n            </StyledTable>\n          ) : null}\n          {!hideTotal ? (\n            <EmptyTable cellPadding=\"0\" cellSpacing=\"0\">\n              <colgroup>\n                <col />\n                <PriceCol />\n              </colgroup>\n              <tbody>\n                <tr>\n                  <EmptyCell aria-label=\"control\" />\n                  <TotalPriceCell hasBorder={false}>\n                    {isBeta ? (\n                      <BadgeBeta\n                        prominence=\"strong\"\n                        long={\n                          locales[\n                            `estimate.cost.beta.${\n                              discount > 0 ? 'discount' : 'free'\n                            }`\n                          ].length > 25\n                        }\n                        sentiment=\"warning\"\n                      >\n                        {`${discount > 0 ? discount * 100 : ''}\n                          ${\n                            locales[\n                              `estimate.cost.beta.${\n                                discount > 0 ? 'discount' : 'free'\n                              }`\n                            ]\n                          }`}\n                      </BadgeBeta>\n                    ) : null}\n                    <StyledText\n                      as=\"h3\"\n                      variant=\"heading\"\n                      color=\"primary\"\n                      isBeta={isBeta}\n                    >\n                      <LineThrough\n                        isActive={isBeta && (discount === 0 || discount >= 1)}\n                      >\n                        {formatNumber(totalPrice.total, {\n                          maximumFractionDigits: isLongFractionDigits\n                            ? maximumFractionDigitsLong[iteration.unit]\n                            : maximumFractionDigits[iteration.unit],\n                        })}\n                        {totalPrice.maxTotal > 0\n                          ? ` - ${formatNumber(totalPrice.maxTotal, {\n                              maximumFractionDigits: isLongFractionDigits\n                                ? maximumFractionDigitsLong[iteration.unit]\n                                : maximumFractionDigits[iteration.unit],\n                            })}`\n                          : null}\n                      </LineThrough>\n                    </StyledText>\n                    {hideHourlyPriceOnTotal &&\n                    totalPrice.hourly > 0 &&\n                    totalPrice.hourly !== totalPrice.total &&\n                    totalPrice.total > 0 ? (\n                      <RightAlignedText as=\"p\" variant=\"body\">\n                        <LineThrough\n                          isActive={isBeta && (discount === 0 || discount >= 1)}\n                        >\n                          {formatNumber(totalPrice.hourly, {\n                            maximumFractionDigits: isLongFractionDigits\n                              ? maximumFractionDigitsLong.hours\n                              : maximumFractionDigits.hours,\n                          })}\n                          {totalPrice.maxHourly > 0\n                            ? ` - ${formatNumber(totalPrice.maxHourly, {\n                                maximumFractionDigits: isLongFractionDigits\n                                  ? maximumFractionDigitsLong.hours\n                                  : maximumFractionDigits.hours,\n                              })}`\n                            : null}\n                          /\n                          {locales[\n                            `estimate.cost.units.hours.label`\n                          ].toLowerCase()}\n                        </LineThrough>\n                      </RightAlignedText>\n                    ) : null}\n                  </TotalPriceCell>\n                </tr>\n              </tbody>\n            </EmptyTable>\n          ) : null}\n          {commitmentFees !== undefined || monthlyFees !== undefined ? (\n            <>\n              <FeesText as=\"h3\" variant=\"headingSmall\">\n                {\n                  locales[\n                    `estimate.cost.fees.${\n                      commitmentFees ? 'oneTime' : 'monthly'\n                    }.title`\n                  ]\n                }\n              </FeesText>\n              <StyledFeesTable>\n                <tbody>\n                  <Item\n                    label={\n                      commitmentFees\n                        ? locales['estimate.cost.fees.commitment']\n                        : monthlyFeesLabel\n                    }\n                    noIteration\n                    isLastElement\n                    price={commitmentFees || monthlyFees}\n                    productsCallback={{\n                      add: () => {},\n                      remove: () => {},\n                    }}\n                  >\n                    {commitmentFees\n                      ? commitmentFeesContent\n                      : monthlyFeesContent}\n                  </Item>\n                </tbody>\n              </StyledFeesTable>\n            </>\n          ) : null}\n        </div>\n      </OverlayContextProvider>\n    </Stack>\n  )\n}\n"]} */"));
33
71
  const DEFAULT_UNIT_LIST = ["hours", "days", "months"];
34
72
  const DescriptionComponent = ({
35
73
  description,
@@ -62,7 +100,9 @@ const EstimateCostContent = ({
62
100
  locales = EstimateCostLocales,
63
101
  overlayMargin
64
102
  }) => {
65
- const { formatNumber } = useEstimateCost();
103
+ const {
104
+ formatNumber
105
+ } = useEstimateCost();
66
106
  const [ref, inView] = useInView();
67
107
  const [products, setProducts] = useState([]);
68
108
  const [totalPrice, setTotalPrice] = useState({
@@ -78,144 +118,83 @@ const EstimateCostContent = ({
78
118
  unit: defaultTimeUnit ?? "hours"
79
119
  });
80
120
  const [isLongFractionDigits, setIsLongFractionDigits] = useState(false);
81
- const providerValue = useMemo(() => ({ isOverlay: false }), []);
82
- const productsCallback = useMemo(
83
- () => ({
84
- add: (newProduct) => {
85
- setProducts((total) => {
86
- if (total.find((product) => product.id === newProduct.id)) {
87
- return total.map(
88
- (product) => product.id === newProduct.id ? newProduct : product
89
- );
90
- }
91
- return [...total, newProduct];
92
- });
93
- },
94
- remove: ({ id }) => {
95
- setProducts((total) => total.filter((product) => product.id !== id));
96
- }
97
- }),
98
- [setProducts]
99
- );
121
+ const providerValue = useMemo(() => ({
122
+ isOverlay: false
123
+ }), []);
124
+ const productsCallback = useMemo(() => ({
125
+ add: (newProduct) => {
126
+ setProducts((total) => {
127
+ if (total.find((product) => product.id === newProduct.id)) {
128
+ return total.map((product) => product.id === newProduct.id ? newProduct : product);
129
+ }
130
+ return [...total, newProduct];
131
+ });
132
+ },
133
+ remove: ({
134
+ id
135
+ }) => {
136
+ setProducts((total) => total.filter((product) => product.id !== id));
137
+ }
138
+ }), [setProducts]);
100
139
  useEffect(() => {
101
140
  const isMaxAmountInProducts = products.find((product) => product.maxAmount);
102
- setIsLongFractionDigits(
103
- !!products.find((product) => product.longFractionDigits)
104
- );
141
+ setIsLongFractionDigits(!!products.find((product) => product.longFractionDigits));
105
142
  setTotalPrice({
106
- total: !hideTotal ? products.reduce(
107
- (acc, product) => acc + calculatePrice({
108
- price: product.price,
109
- amount: product.amount,
110
- amountFree: product.amountFree,
111
- timeUnit: product.noIteration ? "hours" : iteration.unit,
112
- timeAmount: product.noIteration ? 1 : iteration.value,
113
- discount: product.discount
114
- }),
115
- 0
116
- ) : 0,
117
- maxTotal: isMaxAmountInProducts ? products.reduce(
118
- (acc, product) => acc + calculatePrice({
119
- price: product.price,
120
- amount: product.maxAmount || product.amount,
121
- // Not all products have maxAmount, so we need to check both
122
- amountFree: product.amountFree,
123
- timeUnit: product.noIteration ? "hours" : iteration.unit,
124
- timeAmount: product.noIteration ? 1 : iteration.value,
125
- discount: product.discount
126
- }),
127
- 0
128
- ) : 0,
129
- hourly: products.reduce(
130
- (acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.amount - product.amountFree, 0)),
131
- 0
132
- ),
133
- maxHourly: isMaxAmountInProducts ? products.reduce(
134
- (acc, product) => acc && product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.maxAmount - product.amountFree, 0),
135
- 0
136
- ) : 0,
137
- overlayHourly: products.reduce(
138
- (acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.amount - product.amountFree, 0)),
139
- 0
140
- ),
141
- maxOverlayHourly: isMaxAmountInProducts ? products.reduce(
142
- (acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.maxAmount - product.amountFree, 0)),
143
- 0
144
- ) : 0
143
+ total: !hideTotal ? products.reduce((acc, product) => acc + calculatePrice({
144
+ price: product.price,
145
+ amount: product.amount,
146
+ amountFree: product.amountFree,
147
+ timeUnit: product.noIteration ? "hours" : iteration.unit,
148
+ timeAmount: product.noIteration ? 1 : iteration.value,
149
+ discount: product.discount
150
+ }), 0) : 0,
151
+ maxTotal: isMaxAmountInProducts ? products.reduce((acc, product) => acc + calculatePrice({
152
+ price: product.price,
153
+ amount: product.maxAmount || product.amount,
154
+ // Not all products have maxAmount, so we need to check both
155
+ amountFree: product.amountFree,
156
+ timeUnit: product.noIteration ? "hours" : iteration.unit,
157
+ timeAmount: product.noIteration ? 1 : iteration.value,
158
+ discount: product.discount
159
+ }), 0) : 0,
160
+ hourly: products.reduce((acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.amount - product.amountFree, 0)), 0),
161
+ maxHourly: isMaxAmountInProducts ? products.reduce((acc, product) => acc && product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.maxAmount - product.amountFree, 0), 0) : 0,
162
+ overlayHourly: products.reduce((acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.amount - product.amountFree, 0)), 0),
163
+ maxOverlayHourly: isMaxAmountInProducts ? products.reduce((acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.maxAmount - product.amountFree, 0)), 0) : 0
145
164
  });
146
165
  }, [hideTotal, products, iteration, setTotalPrice]);
147
166
  useEffect(() => {
148
167
  if (hideTimeUnit && (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? "hours"))) {
149
- setIteration({ unit: defaultTimeUnit ?? "hours", value: 1 });
168
+ setIteration({
169
+ unit: defaultTimeUnit ?? "hours",
170
+ value: 1
171
+ });
150
172
  }
151
173
  }, [hideTimeUnit, iteration, defaultTimeUnit]);
152
174
  return /* @__PURE__ */ jsxs(Stack, { gap: 2, children: [
153
- !hideOverlay ? /* @__PURE__ */ jsx(
154
- OverlayComponent,
155
- {
156
- inView,
157
- totalPrice,
158
- disableOverlayLeft,
159
- disableOverlayRight,
160
- OverlayLeft,
161
- OverlayRight,
162
- isBeta,
163
- discount,
164
- unit: overlayUnit ?? "hours",
165
- overlayMargin,
166
- children
167
- }
168
- ) : null,
175
+ !hideOverlay ? /* @__PURE__ */ jsx(OverlayComponent, { inView, totalPrice, disableOverlayLeft, disableOverlayRight, OverlayLeft, OverlayRight, isBeta, discount, unit: overlayUnit ?? "hours", overlayMargin, children }) : null,
169
176
  description === false ? null : /* @__PURE__ */ jsx(DescriptionComponent, { description, locales }),
170
177
  alert ? /* @__PURE__ */ jsx(Alert, { sentiment: alertVariant, title: alertTitle, children: alert }) : null,
171
178
  /* @__PURE__ */ jsx(OverlayContextProvider, { value: providerValue, children: /* @__PURE__ */ jsxs("div", { children: [
172
- children ? /* @__PURE__ */ jsxs(
173
- StyledTable,
174
- {
175
- cellPadding: "0",
176
- cellSpacing: "0",
177
- ref,
178
- "data-testid": "summary",
179
- noTotal: hideTotal,
180
- children: [
181
- /* @__PURE__ */ jsxs("colgroup", { children: [
182
- /* @__PURE__ */ jsx("col", {}),
183
- /* @__PURE__ */ jsx(PriceCol, {})
184
- ] }),
185
- !hideTimeUnit ? /* @__PURE__ */ jsx("thead", { children: /* @__PURE__ */ jsxs("tr", { children: [
186
- /* @__PURE__ */ jsx("th", { children: /* @__PURE__ */ jsxs(Title, { children: [
187
- /* @__PURE__ */ jsx(
188
- StyledIcon,
189
- {
190
- name: "calculator",
191
- color: "primary",
192
- size: 20
193
- }
194
- ),
195
- locales["estimate.cost.label"]
196
- ] }) }),
197
- /* @__PURE__ */ jsx(StyledPriceCell, { children: /* @__PURE__ */ jsx(TimeCell, { children: /* @__PURE__ */ jsx(
198
- CustomUnitInput,
199
- {
200
- defaultTimeUnit,
201
- setIteration,
202
- iteration,
203
- timeUnits
204
- }
205
- ) }) })
206
- ] }) }) : null,
207
- /* @__PURE__ */ jsx("tbody", { children: Children.map(
208
- children,
209
- (child, index) => isValidElement(child) ? cloneElement(child, {
210
- isLastElement: index === Children.count(children) - 1,
211
- productsCallback,
212
- iteration,
213
- discount: discount && !child.props.discount ? discount : child.props.discount
214
- }) : child
215
- ) })
216
- ]
217
- }
218
- ) : null,
179
+ children ? /* @__PURE__ */ jsxs(StyledTable, { cellPadding: "0", cellSpacing: "0", ref, "data-testid": "summary", noTotal: hideTotal, children: [
180
+ /* @__PURE__ */ jsxs("colgroup", { children: [
181
+ /* @__PURE__ */ jsx("col", {}),
182
+ /* @__PURE__ */ jsx(PriceCol, {})
183
+ ] }),
184
+ !hideTimeUnit ? /* @__PURE__ */ jsx("thead", { children: /* @__PURE__ */ jsxs("tr", { children: [
185
+ /* @__PURE__ */ jsx("th", { children: /* @__PURE__ */ jsxs(Title, { children: [
186
+ /* @__PURE__ */ jsx(StyledIcon, { name: "calculator", color: "primary", size: 20 }),
187
+ locales["estimate.cost.label"]
188
+ ] }) }),
189
+ /* @__PURE__ */ jsx(StyledPriceCell, { children: /* @__PURE__ */ jsx(TimeCell, { children: /* @__PURE__ */ jsx(CustomUnitInput, { defaultTimeUnit, setIteration, iteration, timeUnits }) }) })
190
+ ] }) }) : null,
191
+ /* @__PURE__ */ jsx("tbody", { children: Children.map(children, (child, index) => isValidElement(child) ? cloneElement(child, {
192
+ isLastElement: index === Children.count(children) - 1,
193
+ productsCallback,
194
+ iteration,
195
+ discount: discount && !child.props.discount ? discount : child.props.discount
196
+ }) : child) })
197
+ ] }) : null,
219
198
  !hideTotal ? /* @__PURE__ */ jsxs(EmptyTable, { cellPadding: "0", cellSpacing: "0", children: [
220
199
  /* @__PURE__ */ jsxs("colgroup", { children: [
221
200
  /* @__PURE__ */ jsx("col", {}),
@@ -224,76 +203,37 @@ const EstimateCostContent = ({
224
203
  /* @__PURE__ */ jsx("tbody", { children: /* @__PURE__ */ jsxs("tr", { children: [
225
204
  /* @__PURE__ */ jsx(EmptyCell, { "aria-label": "control" }),
226
205
  /* @__PURE__ */ jsxs(TotalPriceCell, { hasBorder: false, children: [
227
- isBeta ? /* @__PURE__ */ jsx(
228
- BadgeBeta,
229
- {
230
- prominence: "strong",
231
- long: locales[`estimate.cost.beta.${discount > 0 ? "discount" : "free"}`].length > 25,
232
- sentiment: "warning",
233
- children: `${discount > 0 ? discount * 100 : ""}
234
- ${locales[`estimate.cost.beta.${discount > 0 ? "discount" : "free"}`]}`
235
- }
236
- ) : null,
237
- /* @__PURE__ */ jsx(
238
- StyledText,
239
- {
240
- as: "h3",
241
- variant: "heading",
242
- color: "primary",
243
- isBeta,
244
- children: /* @__PURE__ */ jsxs(
245
- LineThrough,
246
- {
247
- isActive: isBeta && (discount === 0 || discount >= 1),
248
- children: [
249
- formatNumber(totalPrice.total, {
250
- maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong[iteration.unit] : maximumFractionDigits[iteration.unit]
251
- }),
252
- totalPrice.maxTotal > 0 ? ` - ${formatNumber(totalPrice.maxTotal, {
253
- maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong[iteration.unit] : maximumFractionDigits[iteration.unit]
254
- })}` : null
255
- ]
256
- }
257
- )
258
- }
259
- ),
260
- hideHourlyPriceOnTotal && totalPrice.hourly > 0 && totalPrice.hourly !== totalPrice.total && totalPrice.total > 0 ? /* @__PURE__ */ jsx(RightAlignedText, { as: "p", variant: "body", children: /* @__PURE__ */ jsxs(
261
- LineThrough,
262
- {
263
- isActive: isBeta && (discount === 0 || discount >= 1),
264
- children: [
265
- formatNumber(totalPrice.hourly, {
266
- maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong.hours : maximumFractionDigits.hours
267
- }),
268
- totalPrice.maxHourly > 0 ? ` - ${formatNumber(totalPrice.maxHourly, {
269
- maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong.hours : maximumFractionDigits.hours
270
- })}` : null,
271
- "/",
272
- locales[`estimate.cost.units.hours.label`].toLowerCase()
273
- ]
274
- }
275
- ) }) : null
206
+ isBeta ? /* @__PURE__ */ jsx(BadgeBeta, { prominence: "strong", long: locales[`estimate.cost.beta.${discount > 0 ? "discount" : "free"}`].length > 25, sentiment: "warning", children: `${discount > 0 ? discount * 100 : ""}
207
+ ${locales[`estimate.cost.beta.${discount > 0 ? "discount" : "free"}`]}` }) : null,
208
+ /* @__PURE__ */ jsx(StyledText, { as: "h3", variant: "heading", color: "primary", isBeta, children: /* @__PURE__ */ jsxs(LineThrough, { isActive: isBeta && (discount === 0 || discount >= 1), children: [
209
+ formatNumber(totalPrice.total, {
210
+ maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong[iteration.unit] : maximumFractionDigits[iteration.unit]
211
+ }),
212
+ totalPrice.maxTotal > 0 ? ` - ${formatNumber(totalPrice.maxTotal, {
213
+ maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong[iteration.unit] : maximumFractionDigits[iteration.unit]
214
+ })}` : null
215
+ ] }) }),
216
+ hideHourlyPriceOnTotal && totalPrice.hourly > 0 && totalPrice.hourly !== totalPrice.total && totalPrice.total > 0 ? /* @__PURE__ */ jsx(RightAlignedText, { as: "p", variant: "body", children: /* @__PURE__ */ jsxs(LineThrough, { isActive: isBeta && (discount === 0 || discount >= 1), children: [
217
+ formatNumber(totalPrice.hourly, {
218
+ maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong.hours : maximumFractionDigits.hours
219
+ }),
220
+ totalPrice.maxHourly > 0 ? ` - ${formatNumber(totalPrice.maxHourly, {
221
+ maximumFractionDigits: isLongFractionDigits ? maximumFractionDigitsLong.hours : maximumFractionDigits.hours
222
+ })}` : null,
223
+ "/",
224
+ locales[`estimate.cost.units.hours.label`].toLowerCase()
225
+ ] }) }) : null
276
226
  ] })
277
227
  ] }) })
278
228
  ] }) : null,
279
229
  commitmentFees !== void 0 || monthlyFees !== void 0 ? /* @__PURE__ */ jsxs(Fragment, { children: [
280
230
  /* @__PURE__ */ jsx(FeesText, { as: "h3", variant: "headingSmall", children: locales[`estimate.cost.fees.${commitmentFees ? "oneTime" : "monthly"}.title`] }),
281
- /* @__PURE__ */ jsx(StyledFeesTable, { children: /* @__PURE__ */ jsx("tbody", { children: /* @__PURE__ */ jsx(
282
- Item,
283
- {
284
- label: commitmentFees ? locales["estimate.cost.fees.commitment"] : monthlyFeesLabel,
285
- noIteration: true,
286
- isLastElement: true,
287
- price: commitmentFees || monthlyFees,
288
- productsCallback: {
289
- add: () => {
290
- },
291
- remove: () => {
292
- }
293
- },
294
- children: commitmentFees ? commitmentFeesContent : monthlyFeesContent
231
+ /* @__PURE__ */ jsx(StyledFeesTable, { children: /* @__PURE__ */ jsx("tbody", { children: /* @__PURE__ */ jsx(Item, { label: commitmentFees ? locales["estimate.cost.fees.commitment"] : monthlyFeesLabel, noIteration: true, isLastElement: true, price: commitmentFees || monthlyFees, productsCallback: {
232
+ add: () => {
233
+ },
234
+ remove: () => {
295
235
  }
296
- ) }) })
236
+ }, children: commitmentFees ? commitmentFeesContent : monthlyFeesContent }) }) })
297
237
  ] }) : null
298
238
  ] }) })
299
239
  ] });