@ultraviolet/plus 0.17.1 → 0.17.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.
@@ -27,7 +27,7 @@ const FeesText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.N
27
27
  label: "FeesText"
28
28
  })("margin-top:", ({
29
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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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"]} */"));
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: Record<keyof typeof EstimateCostLocales, string>\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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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
31
  const StyledText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? {
32
32
  target: "excc3v73"
33
33
  } : {
@@ -36,7 +36,7 @@ const StyledText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env
36
36
  })("text-align:right;", ({
37
37
  isBeta,
38
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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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"]} */"));
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: Record<keyof typeof EstimateCostLocales, string>\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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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
40
  const RightAlignedText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? {
41
41
  target: "excc3v72"
42
42
  } : {
@@ -48,7 +48,7 @@ const RightAlignedText = /* @__PURE__ */ _styled__default.default(ui.Text, proce
48
48
  } : {
49
49
  name: "2qga7i",
50
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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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"]} */",
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: Record<keyof typeof EstimateCostLocales, string>\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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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
52
  toString: _EMOTION_STRINGIFIED_CSS_ERROR__
53
53
  });
54
54
  const StyledIcon = /* @__PURE__ */ _styled__default.default(ui.Icon, process.env.NODE_ENV === "production" ? {
@@ -58,7 +58,7 @@ const StyledIcon = /* @__PURE__ */ _styled__default.default(ui.Icon, process.env
58
58
  label: "StyledIcon"
59
59
  })("margin-right:", ({
60
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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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"]} */"));
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: Record<keyof typeof EstimateCostLocales, string>\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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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
62
  const StyledPriceCell = /* @__PURE__ */ _styled__default.default(componentStyle.Cell.withComponent("th", process.env.NODE_ENV === "production" ? {
63
63
  target: "excc3v75"
64
64
  } : {
@@ -71,7 +71,7 @@ const StyledPriceCell = /* @__PURE__ */ _styled__default.default(componentStyle.
71
71
  label: "StyledPriceCell"
72
72
  })(({
73
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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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"]} */"));
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: Record<keyof typeof EstimateCostLocales, string>\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  onTotalPriceChange,\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 totalValue = useMemo(\n    () =>\n      formatNumber(totalPrice.total, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.total],\n  )\n\n  const totalMaxValue = useMemo(\n    () =>\n      formatNumber(totalPrice.maxTotal, {\n        maximumFractionDigits: isLongFractionDigits\n          ? maximumFractionDigitsLong[iteration.unit]\n          : maximumFractionDigits[iteration.unit],\n      }),\n    [formatNumber, isLongFractionDigits, iteration.unit, totalPrice.maxTotal],\n  )\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    onTotalPriceChange?.({\n      total: totalPrice.total,\n      totalMax: totalPrice.maxTotal > 0 ? totalPrice.maxTotal : undefined,\n    })\n  }, [\n    hideTotal,\n    products,\n    iteration,\n    setTotalPrice,\n    onTotalPriceChange,\n    totalPrice.total,\n    totalPrice.maxTotal,\n    totalValue,\n    totalMaxValue,\n  ])\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                        {totalValue}\n                        {totalPrice.maxTotal > 0 ? ` - ${totalMaxValue}` : 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
75
  const DEFAULT_UNIT_LIST = ["hours", "days", "months"];
76
76
  const DescriptionComponent = ({
77
77
  description,