@ultraviolet/plus 0.15.2 → 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 (63) hide show
  1. package/dist/components/CodeEditor/CodeEditor.cjs +39 -0
  2. package/dist/components/ContentCard/Skeleton.cjs +43 -0
  3. package/dist/components/ContentCard/index.cjs +154 -0
  4. package/dist/components/ContentCardGroup/Card.cjs +93 -0
  5. package/dist/components/ContentCardGroup/SkeletonCard.cjs +34 -0
  6. package/dist/components/ContentCardGroup/index.cjs +33 -0
  7. package/dist/components/Conversation/index.cjs +128 -0
  8. package/dist/components/CustomerSatisfaction/assets/1-5.svg.cjs +3 -0
  9. package/dist/components/CustomerSatisfaction/assets/1-5NB.svg.cjs +3 -0
  10. package/dist/components/CustomerSatisfaction/assets/2-5.svg.cjs +3 -0
  11. package/dist/components/CustomerSatisfaction/assets/2-5NB.svg.cjs +3 -0
  12. package/dist/components/CustomerSatisfaction/assets/3-5.svg.cjs +3 -0
  13. package/dist/components/CustomerSatisfaction/assets/3-5NB.svg.cjs +3 -0
  14. package/dist/components/CustomerSatisfaction/assets/4-5.svg.cjs +3 -0
  15. package/dist/components/CustomerSatisfaction/assets/4-5NB.svg.cjs +3 -0
  16. package/dist/components/CustomerSatisfaction/assets/5-5.svg.cjs +3 -0
  17. package/dist/components/CustomerSatisfaction/assets/5-5NB.svg.cjs +3 -0
  18. package/dist/components/CustomerSatisfaction/index.cjs +123 -0
  19. package/dist/components/EstimateCost/Components/CustomUnitInput.cjs +33 -0
  20. package/dist/components/EstimateCost/Components/Item.cjs +303 -0
  21. package/dist/components/EstimateCost/Components/LineThrough.cjs +18 -0
  22. package/dist/components/EstimateCost/Components/NumberInput.cjs +29 -0
  23. package/dist/components/EstimateCost/Components/Region.cjs +41 -0
  24. package/dist/components/EstimateCost/Components/Regular.cjs +35 -0
  25. package/dist/components/EstimateCost/Components/Strong.cjs +28 -0
  26. package/dist/components/EstimateCost/Components/Unit.cjs +50 -0
  27. package/dist/components/EstimateCost/Components/UnitInput.cjs +90 -0
  28. package/dist/components/EstimateCost/Components/Zone.cjs +40 -0
  29. package/dist/components/EstimateCost/EstimateCost.cjs +88 -0
  30. package/dist/components/EstimateCost/EstimateCostContent.cjs +245 -0
  31. package/dist/components/EstimateCost/EstimateCostProvider.cjs +36 -0
  32. package/dist/components/EstimateCost/OverlayComponent.cjs +122 -0
  33. package/dist/components/EstimateCost/OverlayContext.cjs +14 -0
  34. package/dist/components/EstimateCost/componentStyle.cjs +230 -0
  35. package/dist/components/EstimateCost/constants.cjs +30 -0
  36. package/dist/components/EstimateCost/helper.cjs +19 -0
  37. package/dist/components/EstimateCost/locales/en.cjs +23 -0
  38. package/dist/components/FAQ/index.cjs +36 -0
  39. package/dist/components/Navigation/Navigation.cjs +20 -0
  40. package/dist/components/Navigation/Navigation.d.ts +4 -3
  41. package/dist/components/Navigation/NavigationContent.cjs +216 -0
  42. package/dist/components/Navigation/NavigationProvider.cjs +111 -0
  43. package/dist/components/Navigation/NavigationProvider.d.ts +7 -18
  44. package/dist/components/Navigation/NavigationProvider.js +18 -16
  45. package/dist/components/Navigation/components/Group.cjs +57 -0
  46. package/dist/components/Navigation/components/Item.cjs +438 -0
  47. package/dist/components/Navigation/components/Item.d.ts +4 -3
  48. package/dist/components/Navigation/components/Item.js +34 -23
  49. package/dist/components/Navigation/components/PinnedItems.cjs +108 -0
  50. package/dist/components/Navigation/components/PinnedItems.d.ts +5 -1
  51. package/dist/components/Navigation/components/PinnedItems.js +11 -8
  52. package/dist/components/Navigation/components/Separator.cjs +17 -0
  53. package/dist/components/Navigation/constants.cjs +37 -0
  54. package/dist/components/Navigation/constants.d.ts +14 -0
  55. package/dist/components/Navigation/index.d.ts +1 -0
  56. package/dist/components/Navigation/locales/en.cjs +11 -0
  57. package/dist/components/SteppedListCard/Step.cjs +37 -0
  58. package/dist/components/SteppedListCard/SteppedListContainer.cjs +80 -0
  59. package/dist/components/SteppedListCard/SteppedListContent.cjs +67 -0
  60. package/dist/components/SteppedListCard/helper.cjs +37 -0
  61. package/dist/components/index.d.ts +1 -0
  62. package/dist/index.cjs +25 -0
  63. package/package.json +23 -7
@@ -0,0 +1,245 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const jsxRuntime = require("@emotion/react/jsx-runtime");
4
+ const _styled = require("@emotion/styled/base");
5
+ const ui = require("@ultraviolet/ui");
6
+ const react = require("react");
7
+ const reactIntersectionObserver = require("react-intersection-observer");
8
+ const CustomUnitInput = require("./Components/CustomUnitInput.cjs");
9
+ const Item = require("./Components/Item.cjs");
10
+ const LineThrough = require("./Components/LineThrough.cjs");
11
+ const EstimateCostProvider = require("./EstimateCostProvider.cjs");
12
+ const OverlayComponent = require("./OverlayComponent.cjs");
13
+ const OverlayContext = require("./OverlayContext.cjs");
14
+ const componentStyle = require("./componentStyle.cjs");
15
+ const constants = require("./constants.cjs");
16
+ const helper = require("./helper.cjs");
17
+ const en = require("./locales/en.cjs");
18
+ const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e };
19
+ const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled);
20
+ function _EMOTION_STRINGIFIED_CSS_ERROR__() {
21
+ 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).";
22
+ }
23
+ const FeesText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? {
24
+ target: "excc3v74"
25
+ } : {
26
+ target: "excc3v74",
27
+ label: "FeesText"
28
+ })("margin-top:", ({
29
+ theme
30
+ }) => 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"]} */"));
31
+ const StyledText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? {
32
+ target: "excc3v73"
33
+ } : {
34
+ target: "excc3v73",
35
+ label: "StyledText"
36
+ })("text-align:right;", ({
37
+ isBeta,
38
+ theme
39
+ }) => 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"]} */"));
40
+ const RightAlignedText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? {
41
+ target: "excc3v72"
42
+ } : {
43
+ target: "excc3v72",
44
+ label: "RightAlignedText"
45
+ })(process.env.NODE_ENV === "production" ? {
46
+ name: "2qga7i",
47
+ styles: "text-align:right"
48
+ } : {
49
+ name: "2qga7i",
50
+ styles: "text-align:right",
51
+ 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"]} */",
52
+ toString: _EMOTION_STRINGIFIED_CSS_ERROR__
53
+ });
54
+ const StyledIcon = /* @__PURE__ */ _styled__default.default(ui.Icon, process.env.NODE_ENV === "production" ? {
55
+ target: "excc3v71"
56
+ } : {
57
+ target: "excc3v71",
58
+ label: "StyledIcon"
59
+ })("margin-right:", ({
60
+ theme
61
+ }) => 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"]} */"));
62
+ const StyledPriceCell = /* @__PURE__ */ _styled__default.default(componentStyle.Cell.withComponent("th", process.env.NODE_ENV === "production" ? {
63
+ target: "excc3v75"
64
+ } : {
65
+ target: "excc3v75",
66
+ label: "StyledPriceCell"
67
+ }), process.env.NODE_ENV === "production" ? {
68
+ target: "excc3v70"
69
+ } : {
70
+ target: "excc3v70",
71
+ label: "StyledPriceCell"
72
+ })(({
73
+ theme
74
+ }) => componentStyle.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"]} */"));
75
+ const DEFAULT_UNIT_LIST = ["hours", "days", "months"];
76
+ const DescriptionComponent = ({
77
+ description,
78
+ locales
79
+ }) => description === void 0 || typeof description === "string" ? /* @__PURE__ */ jsxRuntime.jsx(ui.Text, { as: "span", variant: "body", children: description || locales["estimate.cost.description"] }) : description;
80
+ const EstimateCostContent = ({
81
+ description,
82
+ alert,
83
+ alertTitle,
84
+ alertVariant = "warning",
85
+ defaultTimeUnit = "hours",
86
+ timeUnits = DEFAULT_UNIT_LIST,
87
+ hideOverlay = false,
88
+ disableOverlayLeft = false,
89
+ disableOverlayRight = false,
90
+ hideTimeUnit = false,
91
+ hideTotal = false,
92
+ hideHourlyPriceOnTotal = false,
93
+ discount = 0,
94
+ OverlayRight,
95
+ OverlayLeft,
96
+ isBeta = false,
97
+ commitmentFees,
98
+ commitmentFeesContent,
99
+ monthlyFees,
100
+ monthlyFeesLabel,
101
+ monthlyFeesContent,
102
+ overlayUnit = "hours",
103
+ children = null,
104
+ locales = en,
105
+ overlayMargin
106
+ }) => {
107
+ const {
108
+ formatNumber
109
+ } = EstimateCostProvider.useEstimateCost();
110
+ const [ref, inView] = reactIntersectionObserver.useInView();
111
+ const [products, setProducts] = react.useState([]);
112
+ const [totalPrice, setTotalPrice] = react.useState({
113
+ overlayHourly: 0,
114
+ maxOverlayHourly: 0,
115
+ hourly: 0,
116
+ maxHourly: 0,
117
+ total: 0,
118
+ maxTotal: 0
119
+ });
120
+ const [iteration, setIteration] = react.useState({
121
+ value: 1,
122
+ unit: defaultTimeUnit ?? "hours"
123
+ });
124
+ const [isLongFractionDigits, setIsLongFractionDigits] = react.useState(false);
125
+ const providerValue = react.useMemo(() => ({
126
+ isOverlay: false
127
+ }), []);
128
+ const productsCallback = react.useMemo(() => ({
129
+ add: (newProduct) => {
130
+ setProducts((total) => {
131
+ if (total.find((product) => product.id === newProduct.id)) {
132
+ return total.map((product) => product.id === newProduct.id ? newProduct : product);
133
+ }
134
+ return [...total, newProduct];
135
+ });
136
+ },
137
+ remove: ({
138
+ id
139
+ }) => {
140
+ setProducts((total) => total.filter((product) => product.id !== id));
141
+ }
142
+ }), [setProducts]);
143
+ react.useEffect(() => {
144
+ const isMaxAmountInProducts = products.find((product) => product.maxAmount);
145
+ setIsLongFractionDigits(!!products.find((product) => product.longFractionDigits));
146
+ setTotalPrice({
147
+ total: !hideTotal ? products.reduce((acc, product) => acc + helper.calculatePrice({
148
+ price: product.price,
149
+ amount: product.amount,
150
+ amountFree: product.amountFree,
151
+ timeUnit: product.noIteration ? "hours" : iteration.unit,
152
+ timeAmount: product.noIteration ? 1 : iteration.value,
153
+ discount: product.discount
154
+ }), 0) : 0,
155
+ maxTotal: isMaxAmountInProducts ? products.reduce((acc, product) => acc + helper.calculatePrice({
156
+ price: product.price,
157
+ amount: product.maxAmount || product.amount,
158
+ // Not all products have maxAmount, so we need to check both
159
+ amountFree: product.amountFree,
160
+ timeUnit: product.noIteration ? "hours" : iteration.unit,
161
+ timeAmount: product.noIteration ? 1 : iteration.value,
162
+ discount: product.discount
163
+ }), 0) : 0,
164
+ hourly: products.reduce((acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.amount - product.amountFree, 0)), 0),
165
+ 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,
166
+ overlayHourly: products.reduce((acc, product) => acc + (product.noIteration ? 0 : (product.price - product.price * product.discount) * Math.max(product.amount - product.amountFree, 0)), 0),
167
+ 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
168
+ });
169
+ }, [hideTotal, products, iteration, setTotalPrice]);
170
+ react.useEffect(() => {
171
+ if (hideTimeUnit && (iteration.value > 1 || iteration.unit !== (defaultTimeUnit ?? "hours"))) {
172
+ setIteration({
173
+ unit: defaultTimeUnit ?? "hours",
174
+ value: 1
175
+ });
176
+ }
177
+ }, [hideTimeUnit, iteration, defaultTimeUnit]);
178
+ return /* @__PURE__ */ jsxRuntime.jsxs(ui.Stack, { gap: 2, children: [
179
+ !hideOverlay ? /* @__PURE__ */ jsxRuntime.jsx(OverlayComponent.OverlayComponent, { inView, totalPrice, disableOverlayLeft, disableOverlayRight, OverlayLeft, OverlayRight, isBeta, discount, unit: overlayUnit ?? "hours", overlayMargin, children }) : null,
180
+ description === false ? null : /* @__PURE__ */ jsxRuntime.jsx(DescriptionComponent, { description, locales }),
181
+ alert ? /* @__PURE__ */ jsxRuntime.jsx(ui.Alert, { sentiment: alertVariant, title: alertTitle, children: alert }) : null,
182
+ /* @__PURE__ */ jsxRuntime.jsx(OverlayContext.OverlayContextProvider, { value: providerValue, children: /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
183
+ children ? /* @__PURE__ */ jsxRuntime.jsxs(componentStyle.StyledTable, { cellPadding: "0", cellSpacing: "0", ref, "data-testid": "summary", noTotal: hideTotal, children: [
184
+ /* @__PURE__ */ jsxRuntime.jsxs("colgroup", { children: [
185
+ /* @__PURE__ */ jsxRuntime.jsx("col", {}),
186
+ /* @__PURE__ */ jsxRuntime.jsx(componentStyle.PriceCol, {})
187
+ ] }),
188
+ !hideTimeUnit ? /* @__PURE__ */ jsxRuntime.jsx("thead", { children: /* @__PURE__ */ jsxRuntime.jsxs("tr", { children: [
189
+ /* @__PURE__ */ jsxRuntime.jsx("th", { children: /* @__PURE__ */ jsxRuntime.jsxs(componentStyle.Title, { children: [
190
+ /* @__PURE__ */ jsxRuntime.jsx(StyledIcon, { name: "calculator", color: "primary", size: 20 }),
191
+ locales["estimate.cost.label"]
192
+ ] }) }),
193
+ /* @__PURE__ */ jsxRuntime.jsx(StyledPriceCell, { children: /* @__PURE__ */ jsxRuntime.jsx(componentStyle.TimeCell, { children: /* @__PURE__ */ jsxRuntime.jsx(CustomUnitInput.CustomUnitInput, { defaultTimeUnit, setIteration, iteration, timeUnits }) }) })
194
+ ] }) }) : null,
195
+ /* @__PURE__ */ jsxRuntime.jsx("tbody", { children: react.Children.map(children, (child, index) => react.isValidElement(child) ? react.cloneElement(child, {
196
+ isLastElement: index === react.Children.count(children) - 1,
197
+ productsCallback,
198
+ iteration,
199
+ discount: discount && !child.props.discount ? discount : child.props.discount
200
+ }) : child) })
201
+ ] }) : null,
202
+ !hideTotal ? /* @__PURE__ */ jsxRuntime.jsxs(componentStyle.EmptyTable, { cellPadding: "0", cellSpacing: "0", children: [
203
+ /* @__PURE__ */ jsxRuntime.jsxs("colgroup", { children: [
204
+ /* @__PURE__ */ jsxRuntime.jsx("col", {}),
205
+ /* @__PURE__ */ jsxRuntime.jsx(componentStyle.PriceCol, {})
206
+ ] }),
207
+ /* @__PURE__ */ jsxRuntime.jsx("tbody", { children: /* @__PURE__ */ jsxRuntime.jsxs("tr", { children: [
208
+ /* @__PURE__ */ jsxRuntime.jsx(componentStyle.EmptyCell, { "aria-label": "control" }),
209
+ /* @__PURE__ */ jsxRuntime.jsxs(componentStyle.TotalPriceCell, { hasBorder: false, children: [
210
+ isBeta ? /* @__PURE__ */ jsxRuntime.jsx(componentStyle.BadgeBeta, { prominence: "strong", long: locales[`estimate.cost.beta.${discount > 0 ? "discount" : "free"}`].length > 25, sentiment: "warning", children: `${discount > 0 ? discount * 100 : ""}
211
+ ${locales[`estimate.cost.beta.${discount > 0 ? "discount" : "free"}`]}` }) : null,
212
+ /* @__PURE__ */ jsxRuntime.jsx(StyledText, { as: "h3", variant: "heading", color: "primary", isBeta, children: /* @__PURE__ */ jsxRuntime.jsxs(LineThrough.LineThrough, { isActive: isBeta && (discount === 0 || discount >= 1), children: [
213
+ formatNumber(totalPrice.total, {
214
+ maximumFractionDigits: isLongFractionDigits ? constants.maximumFractionDigitsLong[iteration.unit] : constants.maximumFractionDigits[iteration.unit]
215
+ }),
216
+ totalPrice.maxTotal > 0 ? ` - ${formatNumber(totalPrice.maxTotal, {
217
+ maximumFractionDigits: isLongFractionDigits ? constants.maximumFractionDigitsLong[iteration.unit] : constants.maximumFractionDigits[iteration.unit]
218
+ })}` : null
219
+ ] }) }),
220
+ hideHourlyPriceOnTotal && totalPrice.hourly > 0 && totalPrice.hourly !== totalPrice.total && totalPrice.total > 0 ? /* @__PURE__ */ jsxRuntime.jsx(RightAlignedText, { as: "p", variant: "body", children: /* @__PURE__ */ jsxRuntime.jsxs(LineThrough.LineThrough, { isActive: isBeta && (discount === 0 || discount >= 1), children: [
221
+ formatNumber(totalPrice.hourly, {
222
+ maximumFractionDigits: isLongFractionDigits ? constants.maximumFractionDigitsLong.hours : constants.maximumFractionDigits.hours
223
+ }),
224
+ totalPrice.maxHourly > 0 ? ` - ${formatNumber(totalPrice.maxHourly, {
225
+ maximumFractionDigits: isLongFractionDigits ? constants.maximumFractionDigitsLong.hours : constants.maximumFractionDigits.hours
226
+ })}` : null,
227
+ "/",
228
+ locales[`estimate.cost.units.hours.label`].toLowerCase()
229
+ ] }) }) : null
230
+ ] })
231
+ ] }) })
232
+ ] }) : null,
233
+ commitmentFees !== void 0 || monthlyFees !== void 0 ? /* @__PURE__ */ jsxRuntime.jsxs(jsxRuntime.Fragment, { children: [
234
+ /* @__PURE__ */ jsxRuntime.jsx(FeesText, { as: "h3", variant: "headingSmall", children: locales[`estimate.cost.fees.${commitmentFees ? "oneTime" : "monthly"}.title`] }),
235
+ /* @__PURE__ */ jsxRuntime.jsx(componentStyle.StyledFeesTable, { children: /* @__PURE__ */ jsxRuntime.jsx("tbody", { children: /* @__PURE__ */ jsxRuntime.jsx(Item.Item, { label: commitmentFees ? locales["estimate.cost.fees.commitment"] : monthlyFeesLabel, noIteration: true, isLastElement: true, price: commitmentFees || monthlyFees, productsCallback: {
236
+ add: () => {
237
+ },
238
+ remove: () => {
239
+ }
240
+ }, children: commitmentFees ? commitmentFeesContent : monthlyFeesContent }) }) })
241
+ ] }) : null
242
+ ] }) })
243
+ ] });
244
+ };
245
+ exports.EstimateCostContent = EstimateCostContent;
@@ -0,0 +1,36 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" });
3
+ const jsxRuntime = require("@emotion/react/jsx-runtime");
4
+ const react = require("react");
5
+ const en = require("./locales/en.cjs");
6
+ const EstimateCostContext = react.createContext({
7
+ locales: en,
8
+ formatNumber: () => ""
9
+ });
10
+ const useEstimateCost = () => react.useContext(EstimateCostContext);
11
+ const EstimateCostProvider = ({
12
+ children,
13
+ locales,
14
+ currency,
15
+ numberLocales
16
+ }) => {
17
+ const newLocales = react.useMemo(() => locales ? {
18
+ ...en,
19
+ ...locales
20
+ } : en, [locales]);
21
+ const formatNumber = react.useCallback((number, options) => {
22
+ const numberFormat = new Intl.NumberFormat(numberLocales, {
23
+ style: "currency",
24
+ currency,
25
+ ...options
26
+ });
27
+ return numberFormat.format(number);
28
+ }, [currency, numberLocales]);
29
+ const value = react.useMemo(() => ({
30
+ locales: newLocales,
31
+ formatNumber
32
+ }), [formatNumber, newLocales]);
33
+ return /* @__PURE__ */ jsxRuntime.jsx(EstimateCostContext.Provider, { value, children });
34
+ };
35
+ exports.EstimateCostProvider = EstimateCostProvider;
36
+ exports.useEstimateCost = useEstimateCost;