UNPKG

@ultraviolet/plus

Version:
51 lines 127 kB
"use client"; "use strict"; Object.defineProperty(exports, Symbol.toStringTag, { value: "Module" }); const jsxRuntime = require("@emotion/react/jsx-runtime"); const _styled = require("@emotion/styled/base"); const ui = require("@ultraviolet/ui"); const react = require("react"); const reactIntersectionObserver = require("react-intersection-observer"); const CustomUnitInput = require("./Components/CustomUnitInput.cjs"); const Item = require("./Components/Item.cjs"); const LineThrough = require("./Components/LineThrough.cjs"); const EstimateCostProvider = require("./EstimateCostProvider.cjs"); const OverlayComponent = require("./OverlayComponent.cjs"); const OverlayContext = require("./OverlayContext.cjs"); const componentStyle = require("./componentStyle.cjs"); const constants = require("./constants.cjs"); const helper = require("./helper.cjs"); const en = require("./locales/en.cjs"); const _interopDefaultCompat = (e) => e && typeof e === "object" && "default" in e ? e : { default: e }; const _styled__default = /* @__PURE__ */ _interopDefaultCompat(_styled); function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; } const FeesText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? { target: "excc3v74" } : { target: "excc3v74", label: "FeesText" })("margin-top:", ({ theme }) => 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":"AA6C6B","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ComponentProps, ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  memo,\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 = memo(\n  ({\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)\n\nconst TitleComponent = memo(\n  ({\n    locales,\n  }: {\n    locales: Required<ComponentProps<typeof EstimateCostContent>['locales']>\n  }) => (\n    <Title>\n      <StyledIcon name=\"calculator\" color=\"primary\" size={20} />\n      {locales?.['estimate.cost.label']}\n    </Title>\n  ),\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 < 0 ? 0 : 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 < 0 ? 0 : 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                      <TitleComponent locales={locales} />\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"]} */")); const StyledText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? { target: "excc3v73" } : { target: "excc3v73", label: "StyledText" })("text-align:right;", ({ isBeta, theme }) => 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":"AAmDE","file":"/home/runner/work/ultraviolet/ultraviolet/packages/plus/src/components/EstimateCost/EstimateCostContent.tsx","sourcesContent":["'use client'\n\nimport styled from '@emotion/styled'\nimport { Alert, Icon, Stack, Text } from '@ultraviolet/ui'\nimport type { ComponentProps, ReactNode } from 'react'\nimport {\n  Children,\n  cloneElement,\n  isValidElement,\n  memo,\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 = memo(\n  ({\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)\n\nconst TitleComponent = memo(\n  ({\n    locales,\n  }: {\n    locales: Required<ComponentProps<typeof EstimateCostContent>['locales']>\n  }) => (\n    <Title>\n      <StyledIcon name=\"calculator\" color=\"primary\" size={20} />\n      {locales?.['estimate.cost.label']}\n    </Title>\n  ),\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 < 0 ? 0 : 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 < 0 ? 0 : 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                      <TitleComponent locales={locales} />\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"]} */")); const RightAlignedText = /* @__PURE__ */ _styled__default.default(ui.Text, process.env.NODE_ENV === "production" ? { target: "excc3v72" } : { target: "excc3v72", label: "RightAlignedText" })(process.env.NODE_ENV === "production" ? { name: "2qga7i", styles: "text-align:right" } : { name: "2qga7i", styles: "text-align:right/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9ob21lL3J1bm5lci93b3JrL3VsdHJhdmlvbGV0L3VsdHJhdmlvbGV0L3BhY2thZ2VzL3BsdXMvc3JjL2NvbXBvbmVudHMvRXN0aW1hdGVDb3N0L0VzdGltYXRlQ29zdENvbnRlbnQudHN4Il0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQXlEcUMiLCJmaWxlIjoiL2hvbWUvcnVubmVyL3dvcmsvdWx0cmF2aW9sZXQvdWx0cmF2aW9sZXQvcGFja2FnZXMvcGx1cy9zcmMvY29tcG9uZW50cy9Fc3RpbWF0ZUNvc3QvRXN0aW1hdGVDb3N0Q29udGVudC50c3giLCJzb3VyY2VzQ29udGVudCI6WyIndXNlIGNsaWVudCdcblxuaW1wb3J0IHN0eWxlZCBmcm9tICdAZW1vdGlvbi9zdHlsZWQnXG5pbXBvcnQgeyBBbGVydCwgSWNvbiwgU3RhY2ssIFRleHQgfSBmcm9tICdAdWx0cmF2aW9sZXQvdWknXG5pbXBvcnQgdHlwZSB7IENvbXBvbmVudFByb3BzLCBSZWFjdE5vZGUgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7XG4gIENoaWxkcmVuLFxuICBjbG9uZUVsZW1lbnQsXG4gIGlzVmFsaWRFbGVtZW50LFxuICBtZW1vLFxuICB1c2VFZmZlY3QsXG4gIHVzZU1lbW8sXG4gIHVzZVN0YXRlLFxufSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHVzZUluVmlldyB9IGZyb20gJ3JlYWN0LWludGVyc2VjdGlvbi1vYnNlcnZlcidcbmltcG9ydCB7IEN1c3RvbVVuaXRJbnB1dCB9IGZyb20gJy4vQ29tcG9uZW50cy9DdXN0b21Vbml0SW5wdXQnXG5pbXBvcnQgeyBJdGVtIH0gZnJvbSAnLi9Db21wb25lbnRzL0l0ZW0nXG5pbXBvcnQgeyBMaW5lVGhyb3VnaCB9IGZyb20gJy4vQ29tcG9uZW50cy9MaW5lVGhyb3VnaCdcbmltcG9ydCB7IHVzZUVzdGltYXRlQ29zdCB9IGZyb20gJy4vRXN0aW1hdGVDb3N0UHJvdmlkZXInXG5pbXBvcnQgeyBPdmVybGF5Q29tcG9uZW50IH0gZnJvbSAnLi9PdmVybGF5Q29tcG9uZW50J1xuaW1wb3J0IHsgT3ZlcmxheUNvbnRleHRQcm92aWRlciB9IGZyb20gJy4vT3ZlcmxheUNvbnRleHQnXG5pbXBvcnQge1xuICBCYWRnZUJldGEsXG4gIENlbGwsXG4gIEVtcHR5Q2VsbCxcbiAgRW1wdHlUYWJsZSxcbiAgUHJpY2VDZWxsLFxuICBQcmljZUNvbCxcbiAgU3R5bGVkRmVlc1RhYmxlLFxuICBTdHlsZWRUYWJsZSxcbiAgVGltZUNlbGwsXG4gIFRpdGxlLFxuICBUb3RhbFByaWNlQ2VsbCxcbn0gZnJvbSAnLi9jb21wb25lbnRTdHlsZSdcbmltcG9ydCB7IG1heGltdW1GcmFjdGlvbkRpZ2l0cywgbWF4aW11bUZyYWN0aW9uRGlnaXRzTG9uZyB9IGZyb20gJy4vY29uc3RhbnRzJ1xuaW1wb3J0IHsgY2FsY3VsYXRlUHJpY2UgfSBmcm9tICcuL2hlbHBlcidcbmltcG9ydCBFc3RpbWF0ZUNvc3RMb2NhbGVzIGZyb20gJy4vbG9jYWxlcy9lbidcbmltcG9ydCB0eXBlIHtcbiAgQmFyZUVzdGltYXRlUHJvZHVjdCxcbiAgRXN0aW1hdGVDb3N0UHJvcHMsXG4gIEVzdGltYXRlUHJvZHVjdCxcbiAgSXRlcmF0aW9uLFxuICBVbml0cyxcbn0gZnJvbSAnLi90eXBlcydcblxuY29uc3QgRmVlc1RleHQgPSBzdHlsZWQoVGV4dClgXG4gIG1hcmdpbi10b3A6ICR7KHsgdGhlbWUgfSkgPT4gdGhlbWUuc3BhY2VbJzMnXX07XG5gXG5cbmNvbnN0IFN0eWxlZFRleHQgPSBzdHlsZWQoVGV4dCk8e1xuICBpc0JldGE/OiBib29sZWFuXG59PmBcbiAgdGV4dC1hbGlnbjogcmlnaHQ7XG4