{"version":3,"file":"index-CDA-Goi-.js","sources":["../../src/common/utils/Authorisation/AuthorisationUtils.js","../../src/common/themes/LightTheme.js","../../src/common/themes/overrides/WhitelabelThemeOverride.js","../../src/common/utils/Constants/Constants.js","../../src/common/utils/Paths/Paths.js","../../src/common/utils/BrowserHistory/BrowserHistory.js","../../src/common/utils/ApplicationInsights/ApplicationInsights.js","../../src/common/utils/CurrencyUtils/CurrencyUtils.js","../../src/common/utils/DataModel/DataModel.js","../../src/common/utils/DrawingUtils/DrawingUtils.js","../../src/common/helpers/formatUtilities.js","../../src/common/utils/GeneralUtils/GeneralUtils.js","../../src/common/utils/SheetUtils/SheetUtils.js","../../src/common/utils/UserFlow/UserFlow.js","../../src/authConfig.js","../../src/common/helpers/auth/getAccessToken.js","../../src/app/services/api.js","../../src/app/services/quickPartsApi.js","../../src/app/services/web-store/webStoreApi.js","../../src/app/services/user.js","../../src/app/slices/appSlice.js","../../src/app/services/contacts.js","../../src/app/slices/customersSlice.js","../../src/app/services/customers.js","../../src/app/slices/contactsSlice.js","../../src/app/slices/cuttingTechnologiesSlice.js","../../src/app/slices/materialsSlice.js","../../src/app/slices/miscItemsSlice.js","../../src/app/services/organisation.js","../../src/app/slices/organisationSlice.js","../../src/app/slices/quoteItemsSlice.js","../../src/app/slices/ratesSlice.js","../../src/app/slices/rateTablesSlice.js","../../src/app/slices/sheetsSlice.js","../../src/app/services/web-store/webStoreAuth.js","../../src/features/web-store/helpers/auth/cookie.js","../../src/app/services/web-store/webStoreCustomer.js","../../src/app/slices/web-store/webStoreAuthSlice.js","../../src/app/slices/web-store/webStoreQuoteItemsSlice.js","../../src/app/store.js","../../src/common/components/CountrySelectInput/CountrySelectInput.jsx","../../src/common/components/LocationSearchInput/LocationSearchInput.jsx","../../src/common/components/RegionSelectInput/RegionSelectInput.jsx","../../src/common/components/AddressInput/AddressInput.jsx","../../src/common/components/AlertDialog/AlertDialog.jsx","../../src/common/components/AppBar/AppBar.jsx","../../src/common/hooks/useCurrencyFormatter.js","../../src/common/hooks/useKeyboardShortcut.js","../../src/common/hooks/useNumberFormatter.js","../../src/common/hooks/useEffectOnlyOnUpdate.js","../../src/common/hooks/useOnAllImagesLoaded.js","../../src/app/services/cuttingTechnologies.js","../../src/app/services/materials.js","../../src/app/services/rateTables.js","../../src/app/services/sheets.js","../../src/common/hooks/useQuoteItemPropertyOptions.js","../../src/common/hooks/useToolBoxTreatments.js","../../src/app/services/organisationUsers.js","../../src/common/utils/GetSingletonChannel.js","../../src/common/hooks/useBroadcastChannel.js","../../src/common/hooks/useUserContextSwitcher.js","../../src/common/components/AppLogo/AppLogo.jsx","../../src/common/components/Button/Button.jsx","../../src/common/components/TbxTooltip/TbxTooltip.jsx","../../src/common/components/ButtonGroup/ButtonGroup.jsx","../../src/common/components/ButtonGroup/ButtonGroupOLD.jsx","../../src/common/components/DashedBorderContainer/DashedBorderContainer.jsx","../../src/common/components/DateTime/DateTime.jsx","../../src/common/icons/CrossIcon/CrossIcon.jsx","../../src/common/components/IconButton/IconButton.jsx","../../src/common/components/Dialog/Dialog.jsx","../../src/common/components/Divider/Divider.jsx","../../src/common/components/ErrorSnackbar/ErrorSnackbar.jsx","../../src/common/components/FileUploader/FileUploader.jsx","../../src/common/components/SelectOption/SelectOption.jsx","../../src/common/components/Input/Input.jsx","../../src/common/components/InputGroup/InputGroup.jsx","../../src/common/components/InputHelperText/InputHelperText.jsx","../../src/common/components/LinearProgress/LinearProgress.jsx","../../src/common/components/Link/Link.jsx","../../src/common/components/LoadingOverlay/LoadingOverlay.jsx","../../src/common/components/Menu/Menu.jsx","../../src/common/components/MenuItem/MenuItem.jsx","../../src/common/components/NotificationBar/NotificationBar.jsx","../../src/common/components/OrganisationLogo/OrganisationLogo.jsx","../../src/common/components/OrganisationSelect/OrganisationSelect.jsx","../../src/common/components/PageSpinner/PageSpinner.jsx","../../src/common/components/Popover/Popover.jsx","../../src/common/components/Price/Price.jsx","../../src/common/components/ProjectStatusIcon/ProjectStatusIcon.jsx","../../src/common/components/QuoteStatusIcon/QuoteStatusIcon.jsx","../../src/common/components/RichContentEditor/RichContentEditor.jsx","../../src/common/components/SearchInput/SearchInput.jsx","../../src/common/components/SearchInput/SearchTextField.jsx","../../src/common/components/SelectButton/SelectButton.jsx","../../src/common/components/SignOut/SignOut.jsx","../../src/common/components/Spacer/HorizontalSpacer.jsx","../../src/common/icons/DeleteIcon/DeleteIcon.jsx","../../src/common/components/Table/TableCell/TableCell.jsx","../../src/common/components/Table/TableActionsCell/TableActionsCell.jsx","../../src/common/components/CurrencyInput/CurrencyInput.jsx","../../src/common/components/TbxLocalizationProvider/TbxLocalizationProvider.jsx","../../src/common/components/Table/TableEditComponent/TableEditComponent.jsx","../../src/common/components/Table/TableRow/TableRow.jsx","../../src/common/components/Table/TableHead/TableHead.jsx","../../src/common/components/Table/TableToolbar/TableToolbar.jsx","../../src/common/components/Table/Table.jsx","../../src/common/components/Tabs/Tabs.jsx","../../src/common/components/TbxBulkEditModal/TbxBulkEditModal.jsx","../../src/common/components/TbxDrawer/TbxDrawer.jsx","../../src/common/components/TbxShowToggle/TbxShowToggle.jsx","../../src/common/components/ThumbnailImage/ThumbnailImage.jsx","../../src/common/components/Toolbar/Toolbar.jsx","../../src/common/components/UserAvatar/UserAvatar.jsx","../../src/common/components/MainAppBar/PublicAppBar.jsx","../../src/common/components/ErrorPages/ErrorPageContainer/ErrorPageContainer.jsx","../../src/common/components/ErrorPages/GenericErrorPage/GenericErrorPage.jsx","../../src/common/components/ErrorPages/NotFoundPage/NotFoundPage.jsx","../../src/common/components/TbxToolbar/TbxToolbar.jsx","../../src/common/components/TbxToolbar/TbxToolbarActions.jsx","../../src/common/components/TbxToolbar/TbxToolbarTitle.jsx","../../src/layout/ActivateSubscriptionBar/ActivateSubscriptionBar.jsx","../../src/layout/FreePlanBar/FreePlanBar.jsx","../../src/layout/TrialWithSubscriptionBar/TrialWithSubscriptionBar.jsx","../../src/common/components/TbxToolbar/TbxToolbarMessages.jsx","../../src/common/components/MainAppBar/CustomerCentralToolbar/CustomerCentralToolbar.jsx","../../src/common/components/TbxToolbar/TbxToolbarDivider.jsx","../../src/common/components/MainAppBar/MaterialsToolbar/MaterialsToolbar.jsx","../../src/common/components/MainAppBar/OrganisationManagementToolbar/OrganisationManagementToolbar.jsx","../../src/app/services/secondaryProcesses.js","../../src/common/components/UpgradePlanLink/UpgradePlanLink.jsx","../../src/common/components/MainAppBar/ProcessesToolbar/ProcessesToolbar.jsx","../../src/common/components/MainAppBar/QuotesDashboardToolbar/QuotesDashboardToolbar.jsx","../../src/common/components/MainAppBar/RateTablesToolbar/RateTablesToolbar.jsx","../../src/common/components/MainAppBar/AppBarPortal.jsx","../../src/common/components/MainAppBar/SubscriptionToolbar/SubscriptionToolbar.jsx","../../src/common/components/TbxToolbar/TbxToolbarSearch.jsx","../../src/features/customer-central/components/ArchiveCustomerDialog.jsx","../../src/features/customer-central/components/CustomersListItem.jsx","../../src/features/customer-central/components/CustomersList.jsx","../../src/features/materials/components/ArchiveMaterialDialog.jsx","../../src/features/materials/components/MaterialsListItem.jsx","../../src/features/materials/components/MaterialsList.jsx","../../src/features/rate-tables/components/ArchiveRateTableDialog.jsx","../../src/features/rate-tables/components/RateTablesListItem.jsx","../../src/features/rate-tables/components/RateTablesList.jsx","../../src/common/components/MainAppBar/UserflowHelpWidgetToggle.jsx","../../src/common/components/MainAppBar/UserflowTooltipsToggle.jsx","../../src/common/components/MainAppBar/MainAppBar.jsx","../../src/layout/Layout.jsx","../../src/app/services/taxRates.js","../../src/common/themes/overrides/StrikerThemeOverride.js","../../src/common/components/UserRequiredContainer/UserObserverContainer.jsx","../../src/common/components/UserRequiredContainer/UserRequiredContainer.jsx","../../src/features/account/views/Register.jsx","../../src/common/components/LanguageSelect/LegacyLanguageSelect.jsx","../../src/app/services/admin.js","../../src/features/administration/views/Administration.jsx","../../src/app/services/quotes.js","../../src/features/customer-central/components/AddCustomerForm.jsx","../../src/features/customer-central/components/CustomerHeader.jsx","../../src/features/customer-central/components/AddContactForm.jsx","../../src/features/customer-central/components/Tabs/CustomerContactsTab/ArchiveContactDialog/ArchiveContactDialog.jsx","../../src/features/customer-central/components/Tabs/CustomerContactsTab/CustomerContactsRow.jsx","../../src/features/customer-central/components/Tabs/CustomerContactsTab/CustomerContactsTable.jsx","../../src/features/customer-central/components/Tabs/CustomerPricingTab/CustomerPricingTab.jsx","../../src/features/customer-central/components/AddressEditForm.jsx","../../src/features/customer-central/components/AddressCard.jsx","../../src/features/customer-central/components/TaxRateCard.jsx","../../src/features/customer-central/components/Tabs/CustomerSummaryTab/CustomerSummaryTab.jsx","../../src/app/services/partLibrary.js","../../src/features/customer-central/components/Tabs/PartLibraryTab/ArchivePartLibraryDialog/ArchivePartLibraryDialog.jsx","../../src/features/customer-central/components/Tabs/PartLibraryTab/PartLibraryRow.jsx","../../src/features/customer-central/components/Tabs/PartLibraryTab/PartLibraryTable.jsx","../../src/features/customer-central/components/Tabs/PartLibraryTab/PartLibraryQuoteDetailsTable.jsx","../../src/features/customer-central/components/Tabs/PartLibraryTab/PartQuoteHistoryDialog.jsx","../../src/features/customer-central/components/Tabs/PartLibraryTab/PartLibraryTab.jsx","../../src/features/customer-central/components/Tabs/QuoteHistoryTab/ArchiveQuoteDialog.jsx","../../src/features/customer-central/components/Tabs/QuoteHistoryTab/QuoteHistoryRow.jsx","../../src/features/customer-central/components/Tabs/QuoteHistoryTab/QuoteHistoryTable.jsx","../../src/features/customer-central/components/Tabs/QuoteHistoryTab/QuoteHistoryTab.jsx","../../src/features/customer-central/components/Tabs/CustomerTabs.jsx","../../src/features/customer-central/views/CustomerCentral.jsx","../../src/app/services/integrations.js","../../src/features/quotes/components/common/QuotePaidStatusChip.jsx","../../src/common/components/MainAppBar/DocumentsAppBar.jsx","../../src/common/components/PDFExporter/PDFExporterHelpers.jsx","../../src/common/managers/NetworkManager/NetworkError/NetworkError.js","../../src/common/managers/NetworkManager/NetworkManager.js","../../src/common/services/PdfService.js","../../src/common/themes/overrides/PrintDocumentsOverrides.js","../../src/common/icons/AddIcon/AddIcon.jsx","../../src/common/icons/CustomerSuppliedMaterialIcon/CustomerSuppliedMaterialIcon.jsx","../../src/common/icons/LayersIcon/LayersIcon.jsx","../../src/common/icons/MiscItemIcon/MiscItemIcon.jsx","../../src/common/icons/PartDetailsIcon/PartDetailsIcon.jsx","../../src/common/icons/ProcessesIcon/ProcessesIcon.jsx","../../src/common/icons/RightArrowIcon/RightArrowIcon.jsx","../../src/features/documents/components/FoldingDetailsDisplay.jsx","../../src/features/documents/components/ItemRow.jsx","../../src/common/helpers/addressUtilities.js","../../src/features/documents/components/ProductionDocumentsCustomerInfo.jsx","../../src/features/documents/components/SecondaryProcessDetailsDisplay.jsx","../../src/features/documents/views/Quote/DeliveryDocket.jsx","../../src/features/documents/views/Quote/ProductionLabels.jsx","../../src/features/documents/components/OrderProcessCell.jsx","../../src/features/documents/components/WorkOrderMiscItemRow.jsx","../../src/features/documents/components/WorkOrderQuoteItemRow.jsx","../../src/features/documents/views/Quote/WorkOrder.jsx","../../src/features/documents/views/Quote/WorkOrderSummary.jsx","../../src/features/documents/views/PrivateDocuments.jsx","../../src/features/documents/components/AccountingDocumentsCustomerInfo.jsx","../../src/features/documents/components/QuoteOrderMiscItemRow.jsx","../../src/features/documents/components/QuoteOrderQuoteItemRow.jsx","../../src/common/icons/PaymentMethodsIcon/paymentMethodsIcon.jsx","../../src/features/documents/components/TotalTableRow.jsx","../../src/features/documents/views/Quote/OrderConfirmation.jsx","../../src/features/documents/views/Quote/ProformaInvoice.jsx","../../src/features/documents/views/Quote/QuoteOrder.jsx","../../src/features/documents/views/Quote/TaxInvoice.jsx","../../src/features/documents/views/PublicDocuments.jsx","../../src/features/documents/views/Documents.jsx","../../src/features/documents/views/RedirectToDocument.jsx","../../src/features/materials/components/AddMaterialForm.jsx","../../src/common/helpers/MaterialConsumptionModes.js","../../src/features/materials/components/AddSheet.jsx","../../src/features/materials/components/DuplicateMaterialDialog.jsx","../../src/common/components/TbxCardWithAction/TbxCardWithAction.jsx","../../src/features/materials/components/ChangeMaterialDensityDialog.jsx","../../src/common/components/SetupModeSelect/SetupModeSelect.jsx","../../src/common/components/SheetChangeModeSelect/SheetChangeModeSelect.jsx","../../src/features/processes/components/CuttingTechTypeSelect/CuttingTechTypeSelect.jsx","../../src/features/processes/components/AddCuttingTechnologyForm/AddCuttingTechnologyForm.jsx","../../src/features/rate-tables/components/AddRateTableForm.jsx","../../src/features/rate-tables/components/MaterialRateTableDialog.jsx","../../src/features/materials/components/MaterialDetails.jsx","../../src/features/materials/components/MaterialHeader.jsx","../../src/features/materials/components/ImportSheetsRecords.jsx","../../src/features/materials/components/ArchiveSheetDialog.jsx","../../src/features/materials/components/SheetsBulkEdit.jsx","../../src/app/services/rates.js","../../src/features/materials/components/SheetsTableRow.jsx","../../src/features/materials/components/SheetsTable.jsx","../../src/features/materials/views/Materials.jsx","../../src/common/components/TbxShadowScroll/index.jsx","../../src/features/payments/PaymentSuccess.jsx","../../src/common/components/RoleRequiredContainer/Unauthorized/Unauthorized.jsx","../../src/common/components/RoleRequiredContainer/RoleRequiredContainer.jsx","../../src/common/components/TbxFoldingCard/TbxFoldingCard.jsx","../../src/common/components/TbxTechCard/TbxTechCard.jsx","../../src/features/processes/components/EditFoldingProcessForm/EditFoldingProcessForm.jsx","../../src/features/processes/components/ProcessMeasureTypeSelect/ProcessMeasureTypeSelect.jsx","../../src/features/processes/components/ProcessSetupModeSelect/ProcessSetupModeSelect.jsx","../../src/features/processes/components/AddSecondaryProcessForm/AddSecondaryProcessForm.jsx","../../src/features/processes/components/SecondaryProcessesTable/ArchiveSecondaryProcessDialog/ArchiveSecondaryProcessDialog.jsx","../../src/features/processes/components/SecondaryProcessesTable/SecondaryProcessesRow.jsx","../../src/features/processes/components/SecondaryProcessesTable/SecondaryProcessOverlay.jsx","../../src/features/processes/components/SecondaryProcessesTable/SecondaryProcessesTable.jsx","../../src/features/processes/views/Process.jsx","../../src/app/services/miscItems.js","../../src/app/services/quoteItems.js","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItemDrawing.jsx","../../src/features/web-store/helpers/utilities.js","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItemPrices.jsx","../../src/features/quotes/components/common/SaveNotesDialog.jsx","../../src/features/quotes/components/common/ItemNotesModal.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItemSettings.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItemDetails.jsx","../../src/features/web-store/components/shared/TbxDialog/index.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItemName.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItemSummary.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscItemContent/MiscItem.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscellaneousItemOverlay.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscellaneousItemList.jsx","../../src/common/utils/quoteUtilities.js","../../src/features/quotes/components/MiscellaneousItemList/MiscItemListSort.jsx","../../src/features/quotes/components/MiscellaneousItemList/MiscItemListActions.jsx","../../src/features/quotes/components/common/QuoteIntegrationChip.jsx","../../src/features/quotes/components/common/QuoteSourceChip.jsx","../../src/features/quotes/components/common/QuoteStatusChip.jsx","../../src/features/quotes/components/common/QuoteTaxRateChip.jsx","../../src/features/quotes/components/QuoteHeader/QuoteCustomerContact.jsx","../../src/features/quotes/components/QuoteHeader/QuoteDetailsOptions.jsx","../../src/features/quotes/components/QuoteHeader/QuoteNotesModal.jsx","../../src/features/quotes/components/QuoteHeader/QuoteNotes.jsx","../../src/features/quotes/components/PriceSettingsDialog/PriceSettingsDialog.jsx","../../src/features/quotes/components/QuoteHeader/QuotePriceAdjustment.jsx","../../src/app/services/deliveryPriceDetails.js","../../src/features/quotes/components/ShippingBreakdownDialog/DeliveryContainersRow.jsx","../../src/features/quotes/components/ShippingBreakdownDialog/DeliveryContainersTable.jsx","../../src/features/quotes/components/ShippingBreakdownDialog/ShippingBreakdownDialog.jsx","../../src/features/quotes/components/QuoteHeader/QuoteShippingBreakdown.jsx","../../src/features/quotes/components/QuoteHeader/QuoteShippingOptions.jsx","../../src/features/quotes/components/QuoteHeader/QuoteHeader.jsx","../../src/common/themes/DarkTheme.js","../../src/features/quotes/components/QuoteItemList/ItemContent/SecondaryProcesses/SecondaryProcessesModal.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/DrawerItem.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/Folding.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DeleteLayerAlertDialog.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/LayerName.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/LayerType.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/Layers.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/PartDetails.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/PartProperties.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/SecondaryProcesses/QuantitySecProcess.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/SecondaryProcesses.jsx","../../src/features/quotes/components/DrawingDoctor/Components/DrawingDoctorDrawer/DrawingDoctorDrawer.jsx","../../src/features/quotes/components/DrawingDoctor/Components/ExtractPartDialog.jsx","../../src/common/components/LayerColorAdornment/LayerColorAdornment.jsx","../../src/common/components/PathTypeLegend/PathTypeLegend.jsx","../../src/common/components/ShortcutsLegend/ShortcutsLegend.jsx","../../src/features/quotes/components/DrawingDoctor/CadView.jsx","../../src/features/quotes/components/DrawingDoctor/Utilities/CustomError.js","../../src/features/quotes/components/DrawingDoctor/Entities/Layer.js","../../src/features/quotes/components/DrawingDoctor/Entities/Point.js","../../src/features/quotes/components/DrawingDoctor/Entities/Circle.js","../../src/features/quotes/components/DrawingDoctor/Entities/EntityConstants.js","../../src/features/quotes/components/DrawingDoctor/Entities/Line.js","../../src/features/quotes/components/DrawingDoctor/Entities/Arc.js","../../src/features/quotes/components/DrawingDoctor/Entities/Text.js","../../src/features/quotes/components/DrawingDoctor/Entities/Path.js","../../src/features/quotes/components/DrawingDoctor/Renderers/CircleRenderer.jsx","../../src/features/quotes/components/DrawingDoctor/Renderers/TextRenderer.jsx","../../src/features/quotes/components/DrawingDoctor/Renderers/PathRenderer.jsx","../../src/features/quotes/components/DrawingDoctor/Renderers/DrawingRenderer.jsx","../../src/features/quotes/components/DrawingDoctor/ZoomControls.jsx","../../src/features/quotes/components/DrawingDoctor/DrawingDoctor.jsx","../../src/features/quotes/components/DrawingDoctor/Entities/Drawing.js","../../src/features/quotes/components/ScaleDrawingModal/ScaleDrawingModal.jsx","../../src/features/quotes/components/DrawingDoctorModal/RenderDimensions.jsx","../../src/features/quotes/components/DrawingDoctorModal/DrawingDoctorModal.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/SecondaryProcesses/ItemSecondaryProcesses.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemDrawing.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemFolding.jsx","../../src/features/quotes/components/QuoteCalculatorNestDialog/QuoteCalculatorNestDialog.jsx","../../src/features/quotes/components/PriceDetailsDialog/PriceDetailsDialog.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemPrices.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemTimes.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemPricesAndTimes.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemSettings.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemDetails.jsx","../../src/common/icons/FixedPriceIcon/FixedPriceIcon.jsx","../../src/common/icons/LibraryIcon/LibraryIcon.jsx","../../src/features/quotes/components/common/QuoteItemStatus.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemName.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/ItemSummary.jsx","../../src/features/quotes/components/QuoteItemList/ItemContent/QuoteItem.jsx","../../src/features/quotes/components/QuoteItemList/QuoteItemOverlay.jsx","../../src/features/quotes/components/QuoteItemList/SeekQuoteItemOverlay.jsx","../../src/features/quotes/components/QuoteItemList/QuoteItemList.jsx","../../src/features/quotes/components/QuoteItemList/BulkEdit/FoldingBulkEdit.jsx","../../src/features/quotes/components/QuoteItemList/BulkEdit/SecondaryProcessesBulkEditModal.jsx","../../src/features/quotes/components/QuoteItemList/BulkEdit/SecondaryProcessesBulkEdit.jsx","../../src/features/quotes/components/QuoteItemList/BulkEdit/SettingsBulkEdit.jsx","../../src/features/quotes/components/QuoteItemList/BulkEdit/QuoteItemsBulkEdit.jsx","../../src/features/quotes/components/QuoteItemList/QuoteItemListFilter.jsx","../../src/features/quotes/components/QuoteItemList/QuoteItemListSort.jsx","../../src/features/quotes/components/QuoteItemList/QuoteItemListActions.jsx","../../src/features/quotes/components/QuoteReview/QuoteReviewMiscItemTableRow.jsx","../../src/features/quotes/components/QuoteReview/QuoteReviewMiscItemsTable.jsx","../../src/features/quotes/components/QuoteReview/QuoteReviewPartTableRow.jsx","../../src/features/quotes/components/QuoteReview/QuoteReviewPartsTable.jsx","../../src/features/quotes/components/QuoteReview/QuoteReview.jsx","../../src/features/quotes/components/QuoteSummary/MaterialRequirements.jsx","../../src/features/quotes/components/QuoteSummary/PriceTotals.jsx","../../src/features/quotes/components/QuoteSummary/Summary.jsx","../../src/features/quotes/components/QuoteSummary/QuoteSummary.jsx","../../src/common/services/QuoteService.js","../../src/features/quotes/components/DownloadDocuments/DownloadDocuments.jsx","../../src/features/quotes/components/EmailCustomerModal/EmailCustomerModal.jsx","../../src/features/quotes/components/QuoteToolbar/ActionsButtonGroup.jsx","../../src/features/quotes/components/Unfold/Unfold.jsx","../../src/features/quotes/components/common/Add3DPartsDialog.jsx","../../src/common/components/PartLibrary/PartLibraryRow.jsx","../../src/common/components/PartLibrary/PartLibraryTable.jsx","../../src/common/components/PartLibrary/PartLibrary.jsx","../../src/features/quotes/components/common/AddFromPartLibrary.jsx","../../src/features/quotes/components/Pdf2Quote/Pdf2Quote.jsx","../../src/features/quotes/components/common/AddPDFParts.jsx","../../src/features/quotes/components/QuickPart/PartsList/PartCategoryPanel/PartCategoryPanel.jsx","../../src/features/quotes/components/QuickPart/PartsList/PartsList.jsx","../../src/features/quotes/components/QuickPart/PartSpecificationPanel/DimensionList/DimensionList.jsx","../../src/features/quotes/components/QuickPart/PartSpecificationPanel/PartSpecificationPanel.jsx","../../src/features/quotes/components/QuickPart/PartViewer/AnimatedDimension/AnimatedDimension.jsx","../../src/features/quotes/components/QuickPart/PartViewer/PartViewer.jsx","../../src/features/quotes/components/QuickPart/QuickPart.jsx","../../src/features/quotes/components/common/AddQuickPartDialog.jsx","../../src/common/components/FileRequirementsList/index.jsx","../../src/common/components/QuoteItemDropzone/dropZoneIcon.jsx","../../src/common/components/QuoteItemDropzone/index.jsx","../../src/features/quotes/components/common/AddQuoteItemsDialog.jsx","../../src/features/quotes/components/QuoteToolbar/AddPartsButtonGroup.jsx","../../src/features/quotes/components/QuoteToolbar/QuoteToolbar.jsx","../../src/features/quotes/views/Quote.jsx","../../src/features/quotes/components/NoProjectsPanel/NoProjectsPanel.jsx","../../src/features/quotes/components/QuotesDashboard/ArchiveQuoteDialog.jsx","../../src/features/quotes/components/QuotesDashboard/QuotesDashboardRow.jsx","../../src/features/quotes/components/QuotesDashboard/QuotesDashboardTable.jsx","../../src/features/quotes/views/QuotesDashboard.jsx","../../src/features/quotes/views/RedirectToQuote.jsx","../../src/features/rate-tables/components/DuplicateRateTableDialog.jsx","../../src/features/rate-tables/components/AddRateForm.jsx","../../src/features/rate-tables/components/ImportRates.jsx","../../src/features/rate-tables/components/ArchiveRateDialog.jsx","../../src/features/rate-tables/components/RatesBulkEdit.jsx","../../src/features/rate-tables/components/RatesTableRow.jsx","../../src/features/rate-tables/components/RatesTable.jsx","../../src/features/rate-tables/components/RateTableDetails.jsx","../../src/features/rate-tables/components/RateTableHeader.jsx","../../src/features/rate-tables/views/RateTables.jsx","../../src/common/components/LanguageSelect/LanguageSelect.jsx","../../src/common/components/MainAppBar/SettingsToolbar/SettingsToolbar.jsx","../../src/features/settings/components/AccountSettings/AccountSettings.jsx","../../src/common/services/ParagonService.js","../../src/hooks/useGoogleApi.js","../../src/features/settings/components/OrganisationSettings/EditGoogleDriveStatusesDialog.jsx","../../src/features/settings/components/OrganisationSettings/GoogleDriveIntegrationWithAuthCode.jsx","../../src/features/settings/components/OrganisationSettings/ParagonConnectIntegration.jsx","../../src/features/settings/components/OrganisationSettings/Integrations.jsx","../../src/features/settings/components/OrganisationSettings/Payments.jsx","../../src/common/components/AutoCadVersionSelect/AutoCadVersionSelect.jsx","../../src/common/components/DrawingFileTypeSelect/DrawingFileTypeSelect.jsx","../../src/features/settings/components/OrganisationSettings/AddTaxRateDialog.jsx","../../src/features/settings/components/OrganisationSettings/TaxRates.jsx","../../src/features/settings/components/OrganisationSettings/Settings.jsx","../../src/features/settings/components/OrganisationSettings/AddUserDialog.jsx","../../src/features/settings/components/OrganisationSettings/ChangeUserRole.jsx","../../src/features/settings/components/OrganisationSettings/TransferOwnershipDialog.jsx","../../src/features/settings/components/OrganisationSettings/UserManagement.jsx","../../src/features/settings/components/OrganisationSettings/OrganisationSettings.jsx","../../src/app/services/deliveryContainers.js","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryContainers/AddDeliveryContainerDialog.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryContainers/DeliveryContainerRow.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryContainers/DeliveryContainers.jsx","../../src/app/services/deliveryZones.js","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryPrices/AssignContainerDialog.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryPrices/DeliveryZoneContainerRow.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryPrices/DeliveryPrices.jsx","../../src/common/components/FormComponents/FormLocationPicker.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryZones/AddDeliveryZoneDialog.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryZones/DeliveryZoneOverlay.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryZones/DeliveryZonesRow.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/DeliveryZones/DeliveryZones.jsx","../../src/features/settings/components/ShippingSettings/SmartShippingCalculator/SmartShippingCalculator.jsx","../../src/common/components/FormComponents/FormInputAddress.jsx","../../src/common/components/FormComponents/FormInputText.jsx","../../src/features/settings/components/ShippingSettings/InternalQuoteSettings.jsx","../../src/features/settings/components/ShippingSettings/WebStoreSettings.jsx","../../src/features/settings/components/ShippingSettings/ShippingSettings.jsx","../../src/common/components/ColourPicker/ColourPicker.jsx","../../src/features/settings/components/StoreSettings/StoreSettings.jsx","../../src/features/settings/components/SubscriptionSettings/components/RedirectToPricingRequest.jsx","../../src/features/settings/components/SubscriptionSettings/views/NoActiveSubscription.jsx","../../src/features/settings/components/SubscriptionSettings/components/FreePlanCard.jsx","../../src/features/settings/components/SubscriptionSettings/components/NoActiveSubscriptionCard.jsx","../../src/features/settings/components/SubscriptionSettings/views/SubscriptionSettings.jsx","../../src/common/components/ScrollToAnchor/ScrollToAnchor.jsx","../../src/features/settings/views/Settings.jsx","../../src/common/services/PromotionalCodeService.js","../../src/features/user-welcome/UserWelcome.jsx","../../src/features/web-store/components/shared/ScrollToTop/index.jsx","../../src/app/services/web-store/webStoreQuote.js","../../src/app/services/web-store/webStoreQuoteItems.js","../../src/features/web-store/components/shared/FileRequirementsList/index.jsx","../../src/features/web-store/components/shared/QuoteItemDropzone/dropZoneIcon.jsx","../../src/features/web-store/components/shared/QuoteItemDropzone/index.jsx","../../src/features/web-store/pages/home/Home.jsx","../../src/app/services/web-store/webStoreMaterial.js","../../src/features/web-store/components/Login/Login.jsx","../../src/features/web-store/components/Quote/AddQuoteItemsDialog.jsx","../../src/features/web-store/components/shared/AppLogo/index.jsx","../../src/features/web-store/components/shared/AppBar/AppBar.jsx","../../src/features/web-store/components/shared/TbxLoadingOverlay/index.jsx","../../src/features/web-store/pages/theme/storeThemeOverride.js","../../src/features/web-store/pages/layout/Layout.jsx","../../src/features/web-store/pages/login/Login.jsx","../../src/features/web-store/pages/notFound/NotFound.jsx","../../src/common/components/FormComponents/FormInputCheckbox.jsx","../../src/common/components/FormComponents/FormInputRadio.jsx","../../src/features/web-store/components/shared/TbxTooltip/index.jsx","../../src/features/web-store/components/Order/OrderDetails.jsx","../../src/features/web-store/components/Order/OrderSummary.jsx","../../src/features/web-store/pages/order/Order.jsx","../../src/features/web-store/components/shared/AppBar/AppBarPortal.jsx","../../src/features/web-store/components/Order/OrderedActions.jsx","../../src/features/web-store/pages/orderConfirmation/OrderConfirmation.jsx","../../src/features/web-store/components/Quote/QuoteActions.jsx","../../src/features/web-store/helpers/quoteUtilities.js","../../src/features/web-store/hooks/useQuoteItemPropertyOptions.js","../../src/features/web-store/components/DrawingCleaner/Components/DrawingCleanerFooter.jsx","../../src/features/web-store/components/DrawingCleaner/Components/ExtractPartDialog.jsx","../../src/features/web-store/components/DrawingCleaner/Utilities/CustomError.js","../../src/features/web-store/components/DrawingCleaner/Entities/EntityConstants.js","../../src/features/web-store/components/DrawingCleaner/Entities/Layer.js","../../src/features/web-store/components/DrawingCleaner/Entities/Line.js","../../src/features/web-store/components/DrawingCleaner/Entities/Arc.js","../../src/features/web-store/components/DrawingCleaner/Entities/Point.js","../../src/features/web-store/components/DrawingCleaner/Entities/Circle.js","../../src/features/web-store/components/DrawingCleaner/Entities/Text.js","../../src/features/web-store/components/DrawingCleaner/Entities/Path.js","../../src/features/web-store/components/DrawingCleaner/Entities/Drawing.js","../../src/features/web-store/components/DrawingCleaner/CadView.jsx","../../src/features/web-store/components/DrawingCleaner/Renderers/CircleRenderer.jsx","../../src/features/web-store/components/DrawingCleaner/Renderers/TextRenderer.jsx","../../src/features/web-store/components/DrawingCleaner/Renderers/PathRenderer.jsx","../../src/features/web-store/components/DrawingCleaner/Renderers/DrawingRenderer.jsx","../../src/features/web-store/components/DrawingCleaner/ZoomControls.jsx","../../src/features/web-store/components/DrawingCleaner/DrawingCleaner.jsx","../../src/features/web-store/components/Quote/QuoteItemStatus.jsx","../../src/features/web-store/components/shared/DrawingCleanerModal/index.jsx","../../src/features/web-store/components/Quote/QuoteItem/ItemDetails.jsx","../../src/features/web-store/components/Quote/QuoteItem/ItemSummary.jsx","../../src/features/web-store/components/Quote/QuoteItem/QuoteItem.jsx","../../src/features/web-store/components/Quote/QuoteItemList.jsx","../../src/features/web-store/components/Quote/QuoteItemOverlay.jsx","../../src/features/web-store/components/shared/TbxBulkEditModal/index.jsx","../../src/features/web-store/components/Quote/QuoteItemsBulkEdit.jsx","../../src/features/web-store/components/Quote/QuoteSummary.jsx","../../src/features/web-store/pages/quote/Quote.jsx","../../src/features/web-store/index.jsx","../../src/i18n.js","../../src/App.jsx","../../src/registerServiceWorker.js","../../src/index.jsx"],"sourcesContent":["export const UserRole = {\n SiteAdmin: 'SiteAdmin',\n SiteSupport: 'SiteSupport',\n}\n\n/**\n * Returns true if the user is a site admin\n * @param {AccountInfo} accountInfo from @azure/msal-browser PublicClientApplication.getActiveAccount()\n * @see {@link https://azuread.github.io/microsoft-authentication-library-for-js/ref/classes/_azure_msal_browser.publicclientapplication.html#getactiveaccount @azure/msal-browser documentation for PublicClientApplication.getActiveAccount()}\n * @returns {boolean} True if account has UserRole == SiteAdmin\n */\nexport function isSiteAdmin(accountInfo) {\n return accountInfo?.idTokenClaims?.extension_UserRole === UserRole.SiteAdmin\n}\n\n/**\n * Returns true if the user is a site admin\n * @param {AccountInfo} accountInfo from @azure/msal-browser PublicClientApplication.getActiveAccount()\n * @see {@link https://azuread.github.io/microsoft-authentication-library-for-js/ref/classes/_azure_msal_browser.publicclientapplication.html#getactiveaccount @azure/msal-browser documentation for PublicClientApplication.getActiveAccount()}\n * @returns {boolean} True if account has UserRole == SiteAdmin || UserRole == SiteSupport\n */\nexport function isSiteAdminOrSupport(accountInfo) {\n return (\n accountInfo?.idTokenClaims?.extension_UserRole === UserRole.SiteAdmin ||\n accountInfo?.idTokenClaims?.extension_UserRole === UserRole.SiteSupport\n )\n}\n","import { createTheme, darken } from '@mui/material/styles'\n\nexport const theme = createTheme({\n palette: {\n mode: 'light',\n primary: {\n main: '#F78F1E',\n },\n secondary: {\n main: '#0078A4',\n },\n action: {\n active: '#9FA2B4',\n hover: 'rgba(52, 73, 94, 0.04)',\n selected: 'rgba(52, 73, 94, 0.08)',\n disabled: 'rgba(52, 73, 94, 0.26)',\n disabledBackground: 'rgba(52, 73, 94, 0.12)',\n focus: 'rgba(52, 73, 94, 0.12)',\n },\n background: {\n default: '#FAFAFA',\n paper: '#FFFFFF',\n },\n divider: 'rgba(52, 73, 94, 0.12)',\n text: {\n primary: '#34495E',\n secondary: '#5E7387',\n disabled: 'rgba(52, 73, 94, 0.38);',\n },\n },\n typography: {\n fontFamily: \"'Roboto', 'Asap', sans-serif\",\n inter: \"'Inter', sans-serif\",\n strong1: {\n fontWeight: 700,\n fontSize: '1rem',\n lineHeight: 1.5,\n letterSpacing: '0.00938em',\n },\n strong2: {\n fontWeight: 700,\n fontSize: '0.875rem',\n lineHeight: 1.43,\n letterSpacing: '0.01071em',\n },\n small: {\n fontWeight: 400,\n fontSize: '0.75rem',\n lineHeight: 1.43,\n letterSpacing: '0.0125em',\n },\n inputTextSmall: {\n fontWeight: 400,\n fontSize: '0.875rem',\n lineHeight: 1.5,\n letterSpacing: '0.0125em',\n },\n },\n components: {\n MuiTextField: {\n defaultProps: {\n variant: 'standard',\n },\n },\n MuiFormControl: {\n defaultProps: {\n variant: 'standard',\n },\n },\n MuiButton: {\n defaultProps: {\n disableElevation: true,\n },\n styleOverrides: {\n containedPrimary: {\n color: '#FFFFFF',\n },\n },\n },\n MuiFab: {\n styleOverrides: {\n root: {\n boxShadow: 'none',\n },\n },\n },\n MuiCardActionArea: {\n styleOverrides: {\n focusHighlight: {\n backgroundColor: darken('#E9ECEF', 0.5),\n },\n },\n },\n MuiTableCell: {\n styleOverrides: {\n root: {\n padding: '8px',\n },\n },\n },\n MuiTypography: {\n defaultProps: {\n variantMapping: {\n strong1: 'p',\n strong2: 'p',\n small: 'p',\n inputTextSmall: 'p',\n },\n },\n },\n },\n})\n","import { createTheme } from '@mui/material/styles'\nimport { deepmerge } from '@mui/utils'\n\nexport const WhitelabelThemeOverride = (outerTheme) =>\n createTheme(\n deepmerge(outerTheme, {\n palette: {\n ...outerTheme.palette,\n primary: {\n main: '#808080',\n },\n secondary: {\n main: '#A9A9A9',\n },\n },\n components: {\n MuiButton: {\n styleOverrides: {\n root: {\n '&:hover': { backgroundColor: 'rgb(89, 89, 89)' },\n },\n },\n },\n MuiButtonGroup: {\n styleOverrides: {\n root: {\n '.MuiButtonGroup-grouped:not(:last-of-type)': { borderColor: 'rgb(89, 89, 89)' },\n },\n },\n },\n },\n })\n )\n","import { theme as LightTheme } from '../../themes/LightTheme'\nimport { WhitelabelThemeOverride } from '../../themes/overrides/WhitelabelThemeOverride'\n\nexport const EMPTY_GUID = '00000000-0000-0000-0000-000000000000'\n\nexport const EMAIL_PATTERN = /^[A-Z0-9._%+-]+@[A-Z0-9.-]+\\.[A-Z]{2,}$/i\n\nexport const URL_PATTERN = /^(https?:\\/\\/)((([a-z0-9-]+\\.)+[a-z]{2,})|localhost)(:[0-9]{1,5})?(\\/\\S*)?$/i\n\nexport const HEX_PATTERN = /^#[0-9A-F]{6}$/i\n\nexport const RESELLERS = {\n libellula: {\n name: 'libellula',\n title: 'Libellula.QUOTATION',\n theme: WhitelabelThemeOverride,\n logo: 'libellula-logo.png',\n organisationUnits: 'metric',\n selectedLanguage: 'it-IT',\n organisationLocale: 'it-IT',\n organisationCurrencyCode: 'EUR',\n hasWebStoreAccess: false,\n },\n striker: {\n name: 'striker',\n title: 'Striker Quote',\n theme: WhitelabelThemeOverride,\n logo: 'striker-logo.png',\n organisationUnits: 'metric',\n selectedLanguage: 'en-US',\n organisationLocale: 'en-US',\n organisationCurrencyCode: 'USD',\n hasWebStoreAccess: false,\n },\n}\n\nexport const QUOTE_SOURCE_TYPES = {\n Internal: 'Internal',\n WebStore: 'Web Store',\n}\n\nexport const SHIPPING_OPTIONS = [\n {\n value: 'Pickup',\n label: 'Pickup',\n },\n {\n value: 'Delivery',\n label: 'Delivery',\n },\n]\n\nexport const PUBLIC_DOCUMENTS = {\n Quote: 'quote',\n OrderConfirmation: 'order-confirmation',\n ProformaInvoice: 'proforma',\n TaxInvoice: 'tax-invoice',\n}\n\nexport const PRIVATE_DOCUMENTS = {\n WorkOrderSummary: 'work-order-summary',\n WorkOrder: 'work-order',\n DeliveryDocket: 'delivery-docket',\n ProductionLabels: 'production-labels',\n}\n\nexport const PRODUCT_DETAILS = {\n name: 'toolbox',\n title: 'ToolBox',\n theme: LightTheme,\n logo: 'app-logo.svg',\n organisationUnits: 'imperial',\n selectedLanguage: 'en-US',\n organisationLocale: 'en-US',\n organisationCurrencyCode: 'USD',\n hasWebStoreAccess: true,\n}\n\nexport const WEB_STORE_QUOTE_STATUS = {\n NotCalculated: 'NotCalculated',\n Calculated: 'Calculated',\n Ordered: 'Ordered',\n Invoiced: 'Invoiced',\n Dispatched: 'Dispatched',\n Lost: 'Lost',\n Cancelled: 'Cancelled',\n PendingOrderConfirmation: 'PendingOrderConfirmation',\n Rejected: 'Rejected',\n Voided: 'Voided',\n Issued: 'Issued',\n}\n\nexport const QUOTE_STATUS = {\n NotCalculated: 'NotCalculated',\n Calculated: 'Calculated',\n Draft: 'Draft',\n Issued: 'Issued',\n Ordered: 'Ordered',\n Invoiced: 'Invoiced',\n Lost: 'Lost',\n Cancelled: 'Cancelled',\n PendingOrderConfirmation: 'PendingOrderConfirmation',\n Rejected: 'Rejected',\n Voided: 'Voided',\n Dispatched: 'Dispatched',\n}\n\nexport const QUOTE_STATUS_ORDER = [\n {\n index: 0,\n status: 'Editing',\n },\n {\n index: 1,\n status: 'Draft',\n },\n {\n index: 2,\n status: 'Issued',\n },\n {\n index: 3,\n status: 'Pending Confirmation',\n },\n {\n index: 4,\n status: 'Ordered',\n },\n {\n index: 5,\n status: 'Invoiced',\n },\n {\n index: 6,\n status: 'Lost',\n },\n {\n index: 7,\n status: 'Rejected',\n },\n {\n index: 8,\n status: 'Cancelled',\n },\n {\n index: 9,\n status: 'Voided',\n },\n]\n\nexport const QUOTE_PAGE_STATUS_LABEL = {\n NotCalculated: 'Editing',\n Calculated: 'Editing',\n Draft: 'Draft',\n Issued: 'Issued',\n PendingOrderConfirmation: 'Pending Confirmation',\n Ordered: 'Ordered',\n Invoiced: 'Invoiced',\n Lost: 'Lost',\n Cancelled: 'Cancelled',\n Voided: 'Voided',\n Rejected: 'Rejected',\n Dispatched: 'Dispatched',\n}\n\nexport const QUOTE_PAYMENT_STATUS = {\n Unpaid: 'Unpaid',\n ManualPaid: 'ManualPaid',\n GatewayPaid: 'GatewayPaid',\n}\n\nexport const QUOTE_PAYMENTS_STATUS_LABEL = {\n Unpaid: 'Unpaid',\n ManualPaid: 'Paid',\n GatewayPaid: 'Paid via Stripe',\n}\n\nexport const GET_STATUS_LABEL = (status) => {\n return QUOTE_PAGE_STATUS_LABEL[status]\n}\n\nexport const GET_PAYMENT_STATUS_LABEL = (status) => {\n if (status) {\n return QUOTE_PAYMENTS_STATUS_LABEL[status]\n }\n return QUOTE_PAYMENTS_STATUS_LABEL.Unpaid\n}\n\nexport const GET_INTEGRATION_STATUS_LABEL = (integrationExportSucceeded, integrationExportFailureMessage) => {\n if (integrationExportSucceeded) {\n return 'Invoice Exported'\n }\n if (integrationExportFailureMessage) {\n return 'Invoice Export Error'\n }\n return 'Not Exported'\n}\n\nexport const QUOTE_ACTIONS = {\n reviewQuote: {\n id: 'reviewQuote',\n label: 'Review quote',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Draft,\n tagAttrs: { 'data-testid': 'review-quote-button' },\n },\n calculateQuote: {\n id: 'calculateQuote',\n label: 'Calculate',\n handler: 'handleCalculateQuote',\n tagAttrs: { 'data-testid': 'calculate-quote-button' },\n },\n editQuote: {\n id: 'editQuote',\n label: 'Edit quote',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Calculated,\n tagAttrs: { 'data-testid': 'edit-quote-button' },\n },\n editPendingOrder: {\n id: 'editPendingOrder',\n label: 'Edit pending order',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Calculated,\n tagAttrs: { 'data-testid': 'edit-pending-order-button' },\n },\n markAsIssued: {\n id: 'markAsIssued',\n label: 'Mark as issued',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Issued,\n tagAttrs: { 'data-testid': 'mark-as-issued-button' },\n },\n markAsOrdered: {\n id: 'markAsOrdered',\n label: 'Mark as ordered',\n handler: 'handleMarkAsOrdered',\n tagAttrs: { 'data-testid': 'mark-as-ordered-button' },\n },\n markAsLost: {\n id: 'markAsLost',\n label: 'Mark as lost',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Lost,\n tagAttrs: { 'data-testid': 'mark-as-lost-button' },\n },\n markAsInvoiced: {\n id: 'markAsInvoiced',\n label: 'Mark as invoiced',\n handler: 'handleMarkAsInvoiced',\n tagAttrs: { 'data-testid': 'mark-as-invoiced-button' },\n },\n markAsCancelled: {\n id: 'markAsCancelled',\n label: 'Mark as cancelled',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Cancelled,\n tagAttrs: { 'data-testid': 'mark-as-cancelled-button' },\n },\n markAsVoided: {\n id: 'markAsVoided',\n label: 'Mark as voided',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Voided,\n tagAttrs: { 'data-testid': 'mark-as-voided-button' },\n },\n markAsConfirmed: {\n id: 'markAsConfirmed',\n label: 'Confirm pending order',\n handler: 'handleMarkAsOrdered',\n tagAttrs: { 'data-testid': 'mark-as-confirmed-button' },\n },\n markAsRejected: {\n id: 'markAsRejected',\n label: 'Reject pending order',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.Rejected,\n tagAttrs: { 'data-testid': 'mark-as-rejected-button' },\n },\n markAsPending: {\n id: 'markAsPending',\n label: 'Revert to pending order',\n handler: 'handleQuoteStatusChange',\n param: QUOTE_STATUS.PendingOrderConfirmation,\n tagAttrs: { 'data-testid': 'mark-as-pending-button' },\n },\n markAsPaid: {\n id: 'markAsPaid',\n label: 'Mark as paid',\n handler: 'handleQuotePaymentStatusChange',\n param: QUOTE_PAYMENT_STATUS.ManualPaid,\n tagAttrs: { 'data-testid': 'mark-as-paid-button' },\n hidden: (showPayments) => !showPayments,\n },\n markAsPaidAndOrdered: {\n id: 'markAsPaidAndOrdered',\n label: 'Mark as paid and ordered',\n handler: 'handleMarkAsPaidAndOrdered',\n param: QUOTE_PAYMENT_STATUS.ManualPaid,\n tagAttrs: { 'data-testid': 'mark-as-paid-and-ordered-button' },\n hidden: (showPayments) => !showPayments,\n },\n markAsUnpaid: {\n id: 'markAsUnpaid',\n label: 'Mark as unpaid',\n handler: 'handleQuotePaymentStatusChange',\n param: QUOTE_PAYMENT_STATUS.Unpaid,\n tagAttrs: { 'data-testid': 'mark-as-unpaid-button' },\n hidden: (showPayments) => !showPayments,\n },\n copyQuoteLink: {\n id: 'copyQuoteLink',\n label: 'Copy quote link',\n handler: 'handleCopyLink',\n param: PUBLIC_DOCUMENTS.Quote,\n tagAttrs: { 'data-testid': 'copy-quote-link-button' },\n },\n copyOrderLink: {\n id: 'copyOrderLink',\n label: 'Copy order confirmation link',\n handler: 'handleCopyLink',\n param: PUBLIC_DOCUMENTS.OrderConfirmation,\n tagAttrs: { 'data-testid': 'copy-order-link-button' },\n },\n copyProformaLink: {\n id: 'copyProformaLink',\n label: 'Copy proforma invoice link',\n handler: 'handleCopyLink',\n param: PUBLIC_DOCUMENTS.ProformaInvoice,\n tagAttrs: { 'data-testid': 'copy-proforma-link-button' },\n },\n copyInvoiceLink: {\n id: 'copyInvoiceLink',\n label: 'Copy invoice link',\n handler: 'handleCopyLink',\n param: PUBLIC_DOCUMENTS.TaxInvoice,\n tagAttrs: { 'data-testid': 'copy-invoice-link-button' },\n },\n emailQuote: {\n id: 'emailQuote',\n label: 'Email quote',\n handler: 'handleSendEmail',\n param: PUBLIC_DOCUMENTS.Quote,\n tagAttrs: { 'data-testid': 'email-quote-button' },\n },\n emailOrder: {\n id: 'emailOrder',\n label: 'Email order confirmation',\n handler: 'handleSendEmail',\n param: PUBLIC_DOCUMENTS.OrderConfirmation,\n tagAttrs: { 'data-testid': 'email-order-button' },\n },\n emailProformaInvoice: {\n id: 'emailProformaInvoice',\n label: 'Email proforma invoice',\n handler: 'handleSendEmail',\n param: PUBLIC_DOCUMENTS.ProformaInvoice,\n tagAttrs: { 'data-testid': 'email-proforma-invoice-button' },\n },\n emailTaxInvoice: {\n id: 'emailTaxInvoice',\n label: 'Email tax invoice',\n handler: 'handleSendEmail',\n param: PUBLIC_DOCUMENTS.TaxInvoice,\n tagAttrs: { 'data-testid': 'email-tax-invoice-button' },\n },\n emailCustomer: {\n id: 'emailCustomer',\n label: 'Email customer',\n handler: 'handleSendEmail',\n param: 'sendCustomer',\n tagAttrs: { 'data-testid': 'email-customer-button' },\n },\n\n sendToPartLibrary: {\n id: 'sendToPartLibrary',\n label: 'Send to part library',\n handler: 'handleSendToPartLibrary',\n tagAttrs: { 'data-testid': 'send-to-part-library-button' },\n },\n duplicateQuote: {\n id: 'duplicateQuote',\n label: 'Duplicate quote',\n handler: 'handleDuplicateQuote',\n tagAttrs: { 'data-testid': 'duplicate-quote-button' },\n },\n resendInvoiceData: {\n id: 'resendInvoiceData',\n label: 'Resend invoice data',\n handler: 'handleResendInvoiceData',\n tagAttrs: { 'data-testid': 'resend-invoice-data-button' },\n hidden: (showRetryIntegration) => !showRetryIntegration,\n },\n sendToGoogleDrive: {\n id: 'sendToGoogleDrive',\n label: 'Send to Google Drive',\n handler: 'handleSendToGoogleDrive',\n tagAttrs: { 'data-testid': 'send-to-google-drive-button' },\n hidden: (showGoogleDriveIntegration) => !showGoogleDriveIntegration,\n },\n}\n\nexport const QUOTE_STATE_TO_ACTION_MAP = (status, paymentStatus) => {\n let actions = null\n switch (status) {\n case QUOTE_STATUS.NotCalculated:\n actions = [\n QUOTE_ACTIONS.calculateQuote,\n QUOTE_ACTIONS.emailQuote,\n QUOTE_ACTIONS.markAsIssued,\n QUOTE_ACTIONS.copyQuoteLink,\n QUOTE_ACTIONS.sendToPartLibrary,\n QUOTE_ACTIONS.duplicateQuote,\n ]\n break\n case QUOTE_STATUS.Calculated:\n actions = [\n QUOTE_ACTIONS.reviewQuote,\n QUOTE_ACTIONS.emailQuote,\n QUOTE_ACTIONS.markAsIssued,\n QUOTE_ACTIONS.copyQuoteLink,\n QUOTE_ACTIONS.sendToPartLibrary,\n QUOTE_ACTIONS.duplicateQuote,\n ]\n break\n case QUOTE_STATUS.Draft:\n actions = [\n QUOTE_ACTIONS.markAsIssued,\n QUOTE_ACTIONS.editQuote,\n QUOTE_ACTIONS.emailQuote,\n QUOTE_ACTIONS.copyQuoteLink,\n QUOTE_ACTIONS.sendToPartLibrary,\n QUOTE_ACTIONS.duplicateQuote,\n QUOTE_ACTIONS.sendToGoogleDrive,\n ]\n break\n case QUOTE_STATUS.Issued:\n actions = [\n QUOTE_ACTIONS.markAsOrdered,\n ...(paymentStatus === QUOTE_PAYMENT_STATUS.Unpaid\n ? [QUOTE_ACTIONS.markAsPaidAndOrdered]\n : [QUOTE_ACTIONS.markAsUnpaid]),\n QUOTE_ACTIONS.editQuote,\n QUOTE_ACTIONS.emailQuote,\n QUOTE_ACTIONS.copyQuoteLink,\n QUOTE_ACTIONS.markAsLost,\n QUOTE_ACTIONS.sendToPartLibrary,\n QUOTE_ACTIONS.duplicateQuote,\n QUOTE_ACTIONS.sendToGoogleDrive,\n ]\n break\n case QUOTE_STATUS.PendingOrderConfirmation:\n actions = [\n QUOTE_ACTIONS.emailCustomer,\n QUOTE_ACTIONS.markAsConfirmed,\n ...(paymentStatus === QUOTE_PAYMENT_STATUS.Unpaid\n ? [QUOTE_ACTIONS.markAsPaidAndOrdered]\n : [QUOTE_ACTIONS.markAsUnpaid]),\n QUOTE_ACTIONS.editPendingOrder,\n\n QUOTE_ACTIONS.sendToGoogleDrive,\n QUOTE_ACTIONS.markAsRejected,\n ]\n break\n case QUOTE_STATUS.Ordered:\n actions = [\n QUOTE_ACTIONS.markAsInvoiced,\n ...(paymentStatus === QUOTE_PAYMENT_STATUS.Unpaid\n ? [QUOTE_ACTIONS.markAsPaid]\n : [QUOTE_ACTIONS.markAsUnpaid]),\n QUOTE_ACTIONS.emailOrder,\n QUOTE_ACTIONS.copyOrderLink,\n QUOTE_ACTIONS.emailProformaInvoice,\n QUOTE_ACTIONS.copyProformaLink,\n\n QUOTE_ACTIONS.markAsCancelled,\n QUOTE_ACTIONS.duplicateQuote,\n QUOTE_ACTIONS.sendToGoogleDrive,\n ]\n break\n case QUOTE_STATUS.Invoiced:\n actions = [\n QUOTE_ACTIONS.emailTaxInvoice,\n QUOTE_ACTIONS.copyInvoiceLink,\n ...(paymentStatus === QUOTE_PAYMENT_STATUS.Unpaid\n ? [QUOTE_ACTIONS.markAsPaid]\n : [QUOTE_ACTIONS.markAsUnpaid]),\n QUOTE_ACTIONS.markAsVoided,\n QUOTE_ACTIONS.duplicateQuote,\n QUOTE_ACTIONS.sendToGoogleDrive,\n QUOTE_ACTIONS.resendInvoiceData,\n ]\n break\n case QUOTE_STATUS.Lost:\n actions = [\n QUOTE_ACTIONS.editQuote,\n QUOTE_ACTIONS.emailCustomer,\n ...(paymentStatus === QUOTE_PAYMENT_STATUS.Unpaid\n ? [QUOTE_ACTIONS.markAsPaidAndOrdered]\n : [QUOTE_ACTIONS.markAsUnpaid]),\n QUOTE_ACTIONS.duplicateQuote,\n ]\n break\n case QUOTE_STATUS.Cancelled:\n actions = [QUOTE_ACTIONS.duplicateQuote]\n break\n case QUOTE_STATUS.Voided:\n actions = [QUOTE_ACTIONS.duplicateQuote]\n break\n case QUOTE_STATUS.Rejected:\n actions = [QUOTE_ACTIONS.emailCustomer, QUOTE_ACTIONS.markAsPending]\n break\n case QUOTE_STATUS.Dispatched:\n actions = [QUOTE_ACTIONS.markAsLost, QUOTE_ACTIONS.markAsCancelled]\n break\n default:\n actions = null\n }\n\n return actions\n}\n\nexport const QUOTE_ADD_PARTS_OPTIONS = [\n {\n id: 'add2dParts',\n label: 'Add 2D Parts',\n handler: 'handleAdd2dParts',\n tagAttrs: { 'data-testid': 'add-2d-parts-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n {\n id: 'add3dParts',\n label: 'Add 3D Parts',\n handler: 'handleAdd3dParts',\n tagAttrs: { 'data-testid': 'add-3d-parts-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n {\n id: 'addRotaryParts',\n label: 'Add Rotary Parts',\n handler: 'handleAddRotaryParts',\n tagAttrs: { 'data-testid': 'add-rotary-parts-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n {\n id: 'addFromQuickParts',\n label: 'Add from Quick Part',\n handler: 'handleAddFromQuickParts',\n tagAttrs: { 'data-testid': 'add-from-quickpart-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n {\n id: 'addPdfParts',\n label: 'Add parts from PDF',\n handler: 'handleAddPdfParts',\n tagAttrs: { 'data-testid': 'add-pdf-part-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n {\n id: 'addFromPartsLibrary',\n label: 'Add from Part Library',\n handler: 'handleAddFromPartsLibrary',\n tagAttrs: { 'data-testid': 'add-from-part-library-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n {\n id: 'addMiscellaneousItem',\n label: 'Add Miscellaneous Item',\n handler: 'handleAddMiscellaneousItem',\n tagAttrs: { 'data-testid': 'add-miscellaneous-item-button' },\n source: QUOTE_SOURCE_TYPES.Internal,\n },\n]\n\nexport const WEBSTORE_QUOTE_ADD_PARTS_OPTIONS = [\n {\n id: 'add2dParts',\n label: 'Add 2D Parts',\n handler: 'handleAdd2dParts',\n tagAttrs: { 'data-testid': 'add-2d-parts-button' },\n source: QUOTE_SOURCE_TYPES.WebStore,\n },\n {\n id: 'addFromQuickParts',\n label: 'Add from Quick Part',\n handler: 'handleAddFromQuickParts',\n tagAttrs: { 'data-testid': 'add-from-quickpart-button' },\n source: QUOTE_SOURCE_TYPES.WebStore,\n },\n // {\n // id: 'add3dParts',\n // label: 'Add 3D Parts',\n // handler: 'handleAdd3dParts',\n // tagAttrs: { 'data-testid': 'add-3d-parts-button' },\n // },\n // {\n // id: 'addRotaryParts',\n // label: 'Add Rotary Parts',\n // handler: 'handleAddRotaryParts',\n // tagAttrs: { 'data-testid': 'add-rotary-parts-button' },\n // },\n // {\n // id: 'addPdfParts',\n // label: 'Add parts from PDF',\n // handler: 'handleAddPdfParts',\n // tagAttrs: { 'data-testid': 'add-pdf-part-button' },\n // },\n // {\n // id: 'addFromPartsLibrary',\n // label: 'Add from Part Library',\n // handler: 'handleAddFromPartsLibrary',\n // tagAttrs: { 'data-testid': 'add-from-part-library-button' },\n // },\n]\n\nconst DOWNLOAD_DOCUMENTS_OPTIONS_KEYS_A = [\n QUOTE_STATUS.NotCalculated,\n QUOTE_STATUS.NotCalculated,\n QUOTE_STATUS.Draft,\n QUOTE_STATUS.Issued,\n QUOTE_STATUS.Lost,\n QUOTE_STATUS.PendingOrderConfirmation,\n QUOTE_STATUS.Rejected,\n]\n\nconst DOWNLOAD_DOCUMENTS_OPTIONS_KEYS_B = [QUOTE_STATUS.Ordered, QUOTE_STATUS.Cancelled, QUOTE_STATUS.Voided]\n\nexport const QUOTE_DOWNLOAD_DOCUMENTS_OPTIONS = {\n ...Object.fromEntries(\n DOWNLOAD_DOCUMENTS_OPTIONS_KEYS_A.map((key) => [\n key,\n [\n {\n id: PUBLIC_DOCUMENTS.Quote,\n label: 'Quote',\n disabled: false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-quote-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.OrderConfirmation,\n label: 'Order confirmation',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-order-confirmation-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.ProformaInvoice,\n label: 'Proforma invoice',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-proforma-invoice-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.TaxInvoice,\n label: 'Tax invoice',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-tax-invoice-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.WorkOrderSummary,\n label: 'Work order summary',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-work-order-summary-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.WorkOrder,\n label: 'Work order',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-work-order-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.DeliveryDocket,\n label: 'Delivery docket',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-delivery-docket-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.ProductionLabels,\n label: 'Production labels',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-production-labels-button' },\n },\n ],\n ])\n ),\n ...Object.fromEntries(\n DOWNLOAD_DOCUMENTS_OPTIONS_KEYS_B.map((key) => [\n key,\n [\n {\n id: PUBLIC_DOCUMENTS.Quote,\n label: 'Quote',\n disabled: false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-quote-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.OrderConfirmation,\n label: 'Order confirmation',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-order-confirmation-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.ProformaInvoice,\n label: 'Proforma invoice',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-proforma-invoice-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.TaxInvoice,\n label: 'Tax invoice',\n disabled: true,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-tax-invoice-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.WorkOrderSummary,\n label: 'Work order summary',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-work-order-summary-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.WorkOrder,\n label: 'Work order',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-work-order-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.DeliveryDocket,\n label: 'Delivery docket',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-delivery-docket-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.ProductionLabels,\n label: 'Production labels',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-production-labels-button' },\n },\n ],\n ])\n ),\n [QUOTE_STATUS.Invoiced]: [\n {\n id: PUBLIC_DOCUMENTS.Quote,\n label: 'Quote',\n disabled: false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-quote-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.OrderConfirmation,\n label: 'Order confirmation',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-order-confirmation-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.ProformaInvoice,\n label: 'Proforma invoice',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-proforma-invoice-button' },\n },\n {\n id: PUBLIC_DOCUMENTS.TaxInvoice,\n label: 'Tax invoice',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'public',\n tagAttrs: { 'data-testid': 'download-tax-invoice-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.WorkOrderSummary,\n label: 'Work order summary',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-work-order-summary-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.WorkOrder,\n label: 'Work order',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-work-order-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.DeliveryDocket,\n label: 'Delivery docket',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-delivery-docket-button' },\n },\n {\n id: PRIVATE_DOCUMENTS.ProductionLabels,\n label: 'Production labels',\n disabled: false,\n hidden: import.meta.env.VITE_RESELLER ?? false,\n type: 'private',\n tagAttrs: { 'data-testid': 'download-production-labels-button' },\n },\n ],\n}\n\nexport const QUOTE_USER_TYPE = {\n Customer: 'Customer',\n Guest: 'Guest',\n Internal: 'Internal',\n}\n\nexport const QUADERNO_JURISDICTIONS = [\n {\n country: 'BE',\n name: 'Belgica',\n },\n {\n country: 'BG',\n name: 'Bulgaria',\n },\n {\n country: 'CY',\n name: 'Cyprus',\n },\n {\n country: 'CZ',\n name: 'Czech Republic',\n },\n {\n country: 'DE',\n name: 'Germany',\n },\n {\n country: 'DK',\n name: 'Denmark',\n },\n {\n country: 'EE',\n name: 'Estonia',\n },\n {\n country: 'EL',\n name: 'Greece',\n },\n {\n country: 'ES',\n name: 'Spain',\n },\n {\n country: 'FI',\n name: 'Finland',\n },\n {\n country: 'FR',\n name: 'France',\n },\n {\n country: 'HR',\n name: 'Croatia',\n },\n {\n country: 'HU',\n name: 'Hungary',\n },\n {\n country: 'IE',\n name: 'Ireland',\n },\n {\n country: 'IT',\n name: 'Italy',\n },\n {\n country: 'LR',\n name: 'Lithuania',\n },\n {\n country: 'LU',\n name: 'Luxembourg',\n },\n {\n country: 'LV',\n name: 'Latvia',\n },\n {\n country: 'MT',\n name: 'Malta',\n },\n {\n country: 'NL',\n name: 'The Netherlands',\n },\n {\n country: 'PL',\n name: 'Poland',\n },\n {\n country: 'PT',\n name: 'Portugal',\n },\n {\n country: 'RO',\n name: 'Romania',\n },\n {\n country: 'SE',\n name: 'Sweden',\n },\n {\n country: 'SI',\n name: 'Slovenia',\n },\n {\n country: 'SK',\n name: 'Slovakia',\n },\n {\n country: 'XI',\n name: 'Northern Ireland',\n },\n {\n country: 'AU',\n name: 'Australia',\n },\n {\n country: 'GB',\n name: 'United Kingdom of Great Bretain and Northern Ireland',\n },\n {\n country: 'CH',\n name: 'Switzerland',\n },\n {\n country: 'NZ',\n name: 'New Zealand',\n },\n {\n country: 'CA',\n name: 'Canada',\n },\n]\n\nexport const EXPORTABLE_CONTENT = 'exportable-content'\nexport const FOOTER_BRANDING = 'exportable-footer-branding'\nexport const FOOTER_SELECTOR = 'exportable-footer'\nexport const IS_EXPORTING_SELECTOR = 'exportable-is-exporting'\n\nexport const CUTTING_TECHNOLOGY_POWERS = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]\n\nexport const PARTS_PER_PAGE_VARIABLE = 'parts-per-page'\nexport const PARTS_PER_PAGE_DEFAULT_VALUE = 10\nexport const PARTS_PER_PAGE_OPTIONS = [10, 25, 50]\n\nexport const ITEMS_PER_PAGE_VARIABLE = 'items-per-page'\nexport const ITEMS_PER_PAGE_DEFAULT_VALUE = 5\nexport const ITEMS_PER_PAGE_OPTIONS = [5, 10, 25, 50]\n\nexport const USERS_PER_PAGE_VARIABLE = 'users-per-page'\nexport const USERS_PER_PAGE_DEFAULT_VALUE = 5\nexport const USERS_PER_PAGE_OPTIONS = [5, 10, 15]\n\nexport const TAX_RATES_PER_PAGE_VARIABLE = 'tax-rates-per-page'\nexport const TAX_RATES_PER_PAGE_DEFAULT_VALUE = 5\nexport const TAX_RATES_PER_PAGE_OPTIONS = [5, 10, 15]\n\nexport const INVOICES_PER_PAGE_VARIABLE = 'invoices-per-page'\nexport const INVOICES_PER_PAGE_DEFAULT_VALUE = 5\nexport const INVOICES_PER_PAGE_OPTIONS = [5, 10, 15]\n\nexport const ZONES_PER_PAGE_VARIABLE = 'zones-per-page'\nexport const ZONES_PER_PAGE_DEFAULT_VALUE = 5\nexport const ZONES_PER_PAGE_OPTIONS = [5, 10, 15]\n\nexport const CONTAINERS_PER_PAGE_VARIABLE = 'containers-per-page'\nexport const CONTAINERS_PER_PAGE_DEFAULT_VALUE = 5\nexport const CONTAINERS_PER_PAGE_OPTIONS = [5, 10, 15]\n\nexport const ZONE_CONTAINERS_PER_PAGE_VARIABLE = 'zone-containers-per-page'\nexport const ZONE_CONTAINERS_PER_PAGE_DEFAULT_VALUE = 5\nexport const ZONE_CONTAINERS_PER_PAGE_OPTIONS = [5, 10, 15]\n\nexport const DEFAULT_QUOTE_FOOTER = `Thank you for your enquiry. This estimate is valid for 30 days.\nAll items are subject to the availability and price of raw materials at the time of order.\nAny revisions to this quotation may require a recalculation of price.`\n\nexport const DEFAULT_CONTACT_DETAILS = `Business name\nTax ID 1234567890\n123 Sample Lane\nCity, State, Zip Code\n+1 (123) 456-7890\nsales@yourbusiness.com`\n\nexport const DEFAULT_PAYMENT_DETAILS = `Bank name: Big Bank\nRouting number: 123456789\nAccount number 123456789000\nSWIFT code: WWWFFF`\n\nexport const PROFORMA_INVOICE_NOTES = `This proforma invoice must be paid in full before work can commence.\nPlease note this is not a tax invoice. A full tax invoice will follow delivery of goods.`\n\nexport const LIBELLULA_PRICING_REDIRECT_URL = 'mailto:contact@libellula.eu'\nexport const TOOLBOX_PRICING_REDIRECT_URL = 'https://tempustools.com/contact-us/pricing-request/?stripe_customer_id='\n\nexport const COUNTRY_LANGUAGE_MAP = {\n AU: { locale: 'en-GB', language: 'English' },\n CA: { locale: 'en-US', language: 'English' },\n CL: { locale: 'es-ES', language: 'Spanish' },\n FR: { locale: 'fr-FR', language: 'French' },\n DE: { locale: 'de-DE', language: 'German' },\n IT: { locale: 'it-IT', language: 'Italian' },\n NZ: { locale: 'en-GB', language: 'English' },\n GB: { locale: 'en-GB', language: 'English' },\n US: { locale: 'en-US', language: 'English' },\n BR: { locale: 'pt-PT', language: 'Portuguese' },\n PT: { locale: 'pt-PT', language: 'Portuguese' },\n}\n\nexport const DELIVERY_PRICING_METHODS = {\n SeparateCharge: {\n value: 'SeparateCharge',\n label: 'Charge the customer for delivery separately',\n shortLabel: 'Charge for delivery separately',\n documentLabel: 'Charged separately',\n helperText: 'Contact the customer after they have placed an order to arrange and pay for delivery separately.',\n sources: [QUOTE_SOURCE_TYPES.WebStore],\n },\n PayUponDelivery: {\n value: 'PayUponDelivery',\n label: 'Customer pays a delivery provider directly upon delivery',\n shortLabel: 'Customer pays a delivery provider',\n documentLabel: 'Payable to delivery provider',\n helperText:\n \"Arrange for the delivery provider to invoice the customer directly for payment. Alternatively, the customer might authorise the delivery provider to charge the customer's delivery account number, if they have one.\",\n sources: [QUOTE_SOURCE_TYPES.Internal, QUOTE_SOURCE_TYPES.WebStore],\n },\n ManualCharge: {\n value: 'ManualCharge',\n label: 'Charge for delivery manually',\n shortLabel: 'Charge manually',\n documentLabel: 'Included',\n helperText:\n 'Determine your own price to charge for delivery, if any. Add that charge as a miscellaneous item, or build the delivery charge into the price of the parts.',\n sources: [QUOTE_SOURCE_TYPES.Internal],\n },\n ChargeToOrder: {\n value: 'ChargeToOrder',\n label: 'Use the smart shipping calculator',\n shortLabel: 'Smart shipping calculator',\n documentLabel: 'Included',\n helperText:\n 'Use the smart shipping calculator to determine the delivery charge based on the delivery address, the weight and size of the parts, and the delivery method.',\n sources: [QUOTE_SOURCE_TYPES.Internal, QUOTE_SOURCE_TYPES.WebStore],\n },\n}\n\nexport const DELIVERY_PRICING_OPTIONS = {\n Flat: {\n value: 'Flat',\n label: () => 'Flat price',\n helperText: () => 'Charge a flat price based on the container size and weight, and the delivery address zone.',\n },\n ByDistance: {\n value: 'ByDistance',\n label: (isImperial) => `Flat price + price per ${isImperial ? 'mile' : 'kilometre'}`,\n helperText: (isImperial) =>\n `Charge a price per ${isImperial ? 'mile' : 'kilometre'} between your address and the delivery address, based on the container size and weight, and the delivery address zone.`,\n },\n ByWeight: {\n value: 'ByWeight',\n label: (isImperial) => `Flat price + price per ${isImperial ? 'pound' : 'kilogram'}`,\n helperText: (isImperial) =>\n `Charge a price per ${isImperial ? 'pound' : 'kilogram'} based on the net weight of parts, container size, and the delivery address zone.`,\n },\n}\n","export const ACCOUNT_PATHNAME = '/account'\nexport const ACTIVATE_BILLING_DETAILS_PATHNAME = '/activate/billing-details'\nexport const ACTIVATE_PATHNAME = '/activate'\nexport const ACTIVATE_PAYMENT_DETAILS_PATHNAME = '/activate/payment-details'\nexport const ADMINISTRATION = '/administration'\nexport const BILLING_HISTORY_PATHNAME = '/billing-history'\nexport const CUSTOMER_CENTRAL_PATHNAME = '/customer-central'\nexport const CUSTOMER_FORMATTED_QUOTE_PATHNAME = '/quote'\nexport const CUSTOMERS_PATHNAME = '/customers'\nexport const DASHBOARD_PATHNAME = '/dashboard'\nexport const DOCUMENT_PATHNAME = '/quotes/:quoteId/documents/:documentType/:action?'\nexport const ERROR_PATHNAME = '/sorry'\nexport const MATERIALS_MANAGEMENT = '/materials'\nexport const NO_ACTIVE_SUBSCRIPTION_PATHNAME = '/no-subscription'\nexport const NOT_FOUND = '/not-found'\nexport const PAYMENTS_PATHNAME = '/payments'\nexport const PAYMENTS_SUCCESS_PATHNAME = 'success-message'\nexport const PROCESS_CONFIGURATION = '/process-configuration'\nexport const QUOTE_PATHNAME = '/quotes/:quoteId'\nexport const RATE_TABLES = '/rate-tables'\nexport const REGISTER_PATHNAME = '/register'\nexport const SETTINGS = '/settings'\nexport const SHARED_PATHNAME = '/shared'\nexport const SHARED_V2_PATHNAME = '/shared/v2'\nexport const SIGNOUT_PATHNAME = '/signout'\nexport const WEB_STORE = '/store/:organisationId/*'\nexport const WEB_STORE_SETTINGS = '/settings/store'\nexport const WELCOME_PATHNAME = '/welcome'\n","import { createBrowserHistory } from 'history'\n\nconst baseUrl = import.meta.env.VITE_PUBLIC_URL\nconst browserHistory = createBrowserHistory({ basename: baseUrl })\n\nexport { browserHistory }\n","// based on https://docs.microsoft.com/en-us/azure/azure-monitor/app/javascript-react-plugin\nimport { ReactPlugin } from '@microsoft/applicationinsights-react-js'\nimport { ApplicationInsights } from '@microsoft/applicationinsights-web'\n\nimport { browserHistory } from '../BrowserHistory/BrowserHistory'\n\nconst reactPlugin = new ReactPlugin()\n\nlet appInsights\n\nif (import.meta.env.VITE_APPINSIGHTS_CONNECTIONSTRING) {\n appInsights = new ApplicationInsights({\n config: {\n namePrefix: import.meta.env.VITE_NAME,\n connectionString: import.meta.env.VITE_APPINSIGHTS_CONNECTIONSTRING,\n extensions: [reactPlugin],\n extensionConfig: {\n [reactPlugin.identifier]: { history: browserHistory },\n },\n },\n })\n appInsights.loadAppInsights()\n}\n\nexport { appInsights, reactPlugin }\n","const DEFAULT_CURRENCY = 'USD'\nconst DEFAULT_LOCALE = 'en-US'\n\nexport function getCurrencyFormatter(currencyCode = DEFAULT_CURRENCY, locale = DEFAULT_LOCALE, options) {\n const currencyOptions = {\n style: 'currency',\n currency: currencyCode,\n ...options,\n }\n const formatter = new Intl.NumberFormat(locale, currencyOptions)\n\n return formatter\n}\n\nexport function getFormattedPrice(amount, currencyCode = DEFAULT_CURRENCY, locale = DEFAULT_LOCALE, options) {\n const formatter = getCurrencyFormatter(currencyCode, locale, options)\n const formattedText = formatter.format(amount || 0)\n return formattedText\n}\n\nexport function getCurrencySymbol(currencyCode = DEFAULT_CURRENCY, locale = DEFAULT_LOCALE) {\n return (0)\n .toLocaleString(locale, {\n style: 'currency',\n currency: currencyCode,\n minimumFractionDigits: 0,\n maximumFractionDigits: 0,\n })\n .replace(/\\d/g, '')\n .trim()\n}\n\nexport function getSampleFormattedCurrencyString(locale, currencyCode) {\n const formatter = new Intl.NumberFormat(locale, {\n style: 'currency',\n currency: currencyCode,\n })\n return formatter.format(2500)\n}\n\nexport function getCurrencyFormat(currencyCode, locale) {\n const formatter = new Intl.NumberFormat(locale, {\n style: 'currency',\n currency: currencyCode,\n })\n\n const currencyFormat = {}\n formatter.formatToParts(10000.01).forEach((item) => {\n currencyFormat[item.type] = item.value\n })\n\n currencyFormat['decimal'] = currencyFormat['group'] === '.' ? ',' : '.'\n currencyFormat['decimalPlaces'] = currencyFormat['fraction'] ? currencyFormat['fraction'].length : 0\n currencyFormat['currencySymbolPosition'] = getCurrencySymbolPosition(currencyCode, locale)\n return currencyFormat\n}\n\nconst getCurrencySymbolPosition = function (currency, locale) {\n const formatter = new Intl.NumberFormat(locale, {\n style: 'currency',\n currency: currency,\n currencyDisplay: 'code',\n })\n const parts = formatter.formatToParts(99)\n const currencyIndex = parts.findIndex((part) => part.value === currency)\n\n if (currencyIndex === 0) {\n return 'before'\n } else {\n return 'after'\n }\n}\n","export function getUserDtoFromMsalAccountInfo({\n idTokenClaims: { family_name: lastName, given_name: firstName } = {},\n localAccountId: userId,\n name,\n username: emailAddress,\n} = {}) {\n return {\n userId,\n emailAddress,\n name,\n firstName,\n lastName,\n }\n}\n","/**\n * A compareFunction for use in a @type {Path[]}'s @see Array.prototype.sort() operation.\n * Sorts paths by descending depth, then by descending area.\n * This makes outermost, largest paths render first so they don't obscure the smaller inner paths.\n *\n * @param {Path} a The first Path for comparison\n * @param {Path} b The second Path for comparison\n * @returns If pathSort(a, b) returns a value > than 0, sort b before a. If pathSort(a, b) returns a value <= 0, leave a and b in the same order\n */\nexport function pathSort(a, b) {\n return (a?.depth ?? 0) - (b?.depth ?? 0) || (b?.width ?? 0) * (b?.height ?? 0) - (a?.width ?? 0) * (a?.height ?? 0)\n}\n","const DEFAULT_CURRENCY = 'USD'\nconst DEFAULT_CURRENCY_DISPLAY = 'narrowSymbol'\nconst DEFAULT_LOCALE = 'en-US'\nconst DEFAULT_DECIMAL_PLACES = 2\n\nexport function inchesToMm(value) {\n return value * 25.4\n}\n\nexport function mmToInches(value) {\n return value / 25.4\n}\n\nexport function percentageToFraction(percentage, decimalPlaces = 4) {\n return percentage != null ? parseFloat((percentage / 100).toFixed(decimalPlaces)) : undefined\n}\n\nexport function fractionToPercentage(fraction, decimalPlaces = 4) {\n return fraction != null ? parseFloat((fraction * 100).toFixed(decimalPlaces)) : undefined\n}\n\nexport function formatNumber(value, locale = DEFAULT_LOCALE, useImperialUnits = null) {\n let decimalPlaces = DEFAULT_DECIMAL_PLACES\n\n if (useImperialUnits != null) {\n decimalPlaces = useImperialUnits ? 4 : 2\n }\n\n const formatter = new Intl.NumberFormat(locale, {\n style: 'decimal',\n unitDisplay: 'long',\n signDisplay: 'auto',\n useGrouping: true,\n minimumFractionDigits: decimalPlaces,\n maximumFractionDigits: decimalPlaces,\n })\n return formatter.format(value)\n}\n\nexport function formatCurrency(\n value,\n currency = DEFAULT_CURRENCY,\n locale = DEFAULT_LOCALE,\n decimalPlaces = DEFAULT_DECIMAL_PLACES,\n currencyDisplay = DEFAULT_CURRENCY_DISPLAY\n) {\n const formatter = new Intl.NumberFormat(locale, {\n style: 'currency',\n currency: currency,\n currencyDisplay: currencyDisplay,\n unitDisplay: 'long',\n signDisplay: 'auto',\n useGrouping: true,\n minimumFractionDigits: decimalPlaces,\n maximumFractionDigits: decimalPlaces,\n })\n return formatter.format(value)\n}\n","import { formatNumber } from '@/common/helpers/formatUtilities'\n\nimport { PRODUCT_DETAILS, RESELLERS } from '../Constants/Constants'\n\n// Get current date in string format\nexport function currentDateTimeString() {\n const now = new Date()\n\n return (\n now.getFullYear() +\n String(now.getMonth() + 1).padStart(2, '0') +\n String(now.getDate()).padStart(2, '0') +\n now.getHours() +\n now.getMinutes()\n )\n}\n\nexport function stringIsEmptyOrWhitespace(str) {\n return typeof str === 'undefined' || str === null || str.trim() === ''\n}\n\nexport function getCookieByName(cname) {\n const name = cname + '='\n const decodedCookie = decodeURIComponent(document.cookie)\n const ca = decodedCookie.split(';')\n for (let i = 0; i < ca.length; i++) {\n let c = ca[i]\n while (c.charAt(0) === ' ') {\n c = c.substring(1)\n }\n if (c.indexOf(name) === 0) {\n return c.substring(name.length, c.length)\n }\n }\n return null\n}\n\nexport function inchesToMm(value) {\n return value * 25.4\n}\n\nexport function mmToInches(value) {\n return value / 25.4\n}\n\nexport const canQuoteItemFitOnSheet = (quoteItem, organisationDrawingUnits) =>\n quoteItem &&\n quoteItem.minimumBoundBoxWidth &&\n canPartFitOnSheet(\n {\n drawing: {\n width: quoteItem.minimumBoundBoxWidth,\n height: quoteItem.minimumBoundBoxHeight,\n },\n isImperial: quoteItem.isImperial,\n material: {\n sheetSize: {\n width: quoteItem.sheet?.sheetWidth,\n height: quoteItem.sheet?.sheetHeight,\n },\n },\n },\n organisationDrawingUnits,\n quoteItem.profile\n )\n\nexport function canPartFitOnSheet(\n {\n drawing: { height: partHeight = 0, width: partWidth = 0 },\n isImperial: partIsImperial,\n material: {\n sheetSize = {\n width: 0,\n height: 0,\n },\n },\n },\n organisationDrawingUnits,\n isRotary\n) {\n if (isRotary) {\n return true\n }\n const convertedSheetSize = convertSheetSize(\n sheetSize,\n partIsImperial,\n organisationDrawingUnits?.toLowerCase() === 'imperial'\n )\n\n const largestPartSide = Math.max(partWidth, partHeight)\n const smallestPartSide = Math.min(partWidth, partHeight)\n const largestSheetSide = Math.max(convertedSheetSize?.width, convertedSheetSize?.height)\n const smallestSheetSide = Math.min(convertedSheetSize?.width, convertedSheetSize?.height)\n\n const canFitOnSheetNoRotation = largestPartSide <= largestSheetSide && smallestPartSide <= smallestSheetSide\n let canFitOnSheet = canFitOnSheetNoRotation\n if (!canFitOnSheetNoRotation) {\n canFitOnSheet = partCanFitWithRotation(largestSheetSide, smallestSheetSide, largestPartSide, smallestPartSide)\n }\n\n return canFitOnSheet\n}\n\nexport function partCanFitWithRotation(a, b, p, q) {\n // https://stackoverflow.com/questions/13784274/detect-if-one-rect-can-be-put-into-another-rect\n // based off this paper https://www.jstor.org/stable/2691523\n // formula to calculate if a smaller rectangle can fit in a larger one for a larger rectangle with sides q and p, And a smaller rectangle a and b\n\n const canFitOnSheet =\n q <= b && (p <= a || b * (p * p + q * q) >= 2 * p * q * a + (p * p - q * q) * Math.sqrt(p * p + q * q - a * a))\n return canFitOnSheet\n}\n\nexport function convertSheetSize({ height, width }, partIsImperial, sheetIsImperial) {\n const resultSheetSize = { width, height }\n // if both sheet and part are using same unit of measure then no conversion needed\n if (partIsImperial !== sheetIsImperial) {\n if (partIsImperial) {\n // part is imperial, convert sheet to imperial\n resultSheetSize.width = mmToInches(width)\n resultSheetSize.height = mmToInches(height)\n } else {\n // part is metric, convert sheet to metric\n resultSheetSize.width = inchesToMm(width)\n resultSheetSize.height = inchesToMm(height)\n }\n }\n\n return resultSheetSize\n}\n\nexport function rowHasErrors(rowData, organisationDrawingUnits) {\n const materialError = !rowData.material\n const quantityError = rowData.quantity <= 0\n const sizeError = rowData.material ? !canPartFitOnSheet(rowData, organisationDrawingUnits) : true\n // TODO Use of rowData.calculationErrors should be changed\n // and calculationErrors should come through from the issue list instead\n return materialError || quantityError || sizeError || rowData.calculationErrors?.length > 0\n}\n\nexport function quoteItemHasErrors(quoteItem, organisationDrawingUnits, issueSeverityDictionary) {\n const drawingError = hasSevereIssues(quoteItem.drawingMetadata?.issues, issueSeverityDictionary)\n const materialError =\n !quoteItem.cuttingTechnologyId || !quoteItem.materialId || !quoteItem.sheetId || !quoteItem.thickness\n const quantityError = quoteItem.quantity <= 0\n const sizeError = !materialError ? !canQuoteItemFitOnSheet(quoteItem, organisationDrawingUnits) : true\n // TODO Use of quoteItem.calculationErrors should be changed\n // and calculationErrors should come through from the issue list instead\n return materialError || quantityError || sizeError\n}\n\nconst issueSeverity = {\n issue: 0,\n warning: 1,\n}\n\nexport function hasSevereIssues(partIssues, issueSeverityDictionary) {\n return (\n partIssues &&\n issueSeverityDictionary &&\n partIssues.some((i) => issueSeverityDictionary[i.issueType] === issueSeverity.issue)\n )\n}\n\nexport function sortedParts(parts, organisationDrawingUnits, issueSeverityDictionary) {\n function sortPartsByStatus(part1, part2) {\n const part1Status = hasSevereIssues(part1.drawing?.issues, issueSeverityDictionary)\n ? 0\n : part1.drawing?.issues.length > 0\n ? 1\n : rowHasErrors(part1, organisationDrawingUnits)\n ? 2\n : 3\n\n const part2Status = hasSevereIssues(part2.drawing?.issues, issueSeverityDictionary)\n ? 0\n : part2.drawing?.issues.length > 0\n ? 1\n : rowHasErrors(part2, organisationDrawingUnits)\n ? 2\n : 3\n\n return part1Status < part2Status ? -1 : part1Status === part2Status ? 0 : 1\n }\n\n return parts.sort(sortPartsByStatus)\n}\n\nexport function hasSevereValidationIssues(qi) {\n return (\n Array.isArray(qi.quoteItemMetadata?.quoteItemValidationIssues) &&\n qi.quoteItemMetadata?.quoteItemValidationIssues.length > 0 &&\n qi.quoteItemMetadata?.quoteItemValidationIssues?.some((i) => i.severity === 'Error')\n )\n}\n\nexport const getMessageForValidationIssue = (issue) => {\n switch (issue.type) {\n case 'ParametersInvalid':\n return 'Cutting time cannot be calculated'\n case 'CuttingTechnologyProfileConstraints':\n return 'Material length exceeds cutting technology size constraints'\n case 'CuttingTechnologyConstraints':\n return 'Material size exceeds cutting technology size constraints'\n case 'SheetConstraints':\n return 'Part cannot fit on material'\n case 'NumberOfPartsOrRuntimeInvalid':\n return 'Part nesting or cutting time cannot be calculated'\n case 'SheetWithWebConstraints':\n return 'Part with web cannot fit on material'\n case 'SheetWithWebAndChuckAllowanceConstraints':\n return 'Part with chuck allowance cannot fit on material'\n case 'NoCuttingTechnologySelected':\n return 'Cutting Technology is required'\n case 'NoSheetSelected':\n return 'Sheet is required'\n case 'SheetExpired':\n return 'Selected sheet has expired'\n case 'NoMatchingRate':\n return 'No rate entry found for sheet selection. Check rate table.'\n case 'HasSmallHoles':\n return 'Part contains profiles too small to process'\n case 'NoMaterialSelected':\n return 'Material is required'\n case 'NoThicknessSelected':\n return 'Thickness is required'\n default:\n return issue.message\n }\n}\n\nexport function sheetCanBeCut(sheet, cuttingTech, rateTable, rates) {\n const errorOutput = { errorMessage: '' }\n const canBeCut =\n sheetHasRateTable(rateTable, errorOutput) &&\n sheetCanBeUsedWithCuttingTech(sheet, cuttingTech, errorOutput) &&\n sheetHasRateTableThickness(sheet, rateTable, rates, errorOutput)\n return { canBeCut: canBeCut, errorMessage: errorOutput.errorMessage }\n}\n\nconst sheetCanBeUsedWithCuttingTech = (sheet, cuttingTech, errorOutput) => {\n let length = sheet.sheetWidth\n let width = sheet.sheetHeight\n if (width > length) {\n const tmp = length\n length = width\n width = tmp\n }\n\n const sheetNotLargerThanMax =\n length <= (cuttingTech?.maximumSheetLength ? cuttingTech?.maximumSheetLength : length) &&\n width <= (cuttingTech?.maximumSheetWidth ? cuttingTech?.maximumSheetWidth : width)\n if (!sheetNotLargerThanMax) {\n errorOutput.errorMessage = 'Sheet size is larger than the cutting technology’s maximum settings'\n }\n return sheetNotLargerThanMax\n}\n\nconst sheetHasRateTable = (rateTable, errorOutput) => {\n if (!rateTable) {\n errorOutput.errorMessage = 'No rate table assigned'\n return false\n } else if (rateTable.isDeleted) {\n errorOutput.errorMessage =\n 'The assigned rate table has been archived. Unarchive the rate table or assign a new one.'\n return false\n }\n return true\n}\n\nconst sheetHasRateTableThickness = (sheet, rateTable, rates, errorOutput) => {\n if (rateTable) {\n const sortedRates = Object.values(rates).filter(\n (rate) => rate.rateTableId === rateTable.rateTableId && !rate.isDeleted\n )\n sortedRates.sort((a, b) => a.thickness - b.thickness)\n const rateTableHasSheetThickness =\n sheet.thickness >= sortedRates[0]?.thickness &&\n sheet.thickness <= sortedRates[sortedRates.length - 1]?.thickness\n if (!rateTableHasSheetThickness) {\n errorOutput.errorMessage = 'Sheet thickness not present in rate table'\n }\n return rateTableHasSheetThickness\n }\n return false\n}\n\nexport function sortQuoteItems({ _quote, issueSeverityDictionary, organisationDrawingUnits, quoteItems }) {\n function sortQuoteItemsByStatus(qi1, qi2) {\n const qi1Status = hasSevereIssues(qi1.drawingMetadata?.issues, issueSeverityDictionary)\n ? 0\n : hasSevereValidationIssues(qi1)\n ? 1\n : qi1.drawingMetadata?.issues?.length > 0\n ? 2\n : quoteItemHasErrors(qi1, organisationDrawingUnits)\n ? 3\n : 4\n\n const qi2Status = hasSevereIssues(qi2.drawingMetadata?.issues, issueSeverityDictionary)\n ? 0\n : hasSevereValidationIssues(qi2)\n ? 1\n : qi2.drawingMetadata?.issues?.length > 0\n ? 2\n : quoteItemHasErrors(qi2, organisationDrawingUnits)\n ? 3\n : 4\n\n return qi1Status < qi2Status ? -1 : qi1Status === qi2Status ? 0 : 1\n }\n\n return quoteItems.sort(sortQuoteItemsByStatus)\n}\n\nconst capitalLetters = /([A-Z])/g\nconst leadingKebabSeparator = /^-/\nconst leadingSpaceAndLowerCaseLetter = /^ [a-z]/\n\nexport function pascalToKebabCase(text) {\n return text.replace(capitalLetters, (x) => '-' + x.toLowerCase()).replace(leadingKebabSeparator, '')\n}\n\nexport function pascalToSentenceCase(text) {\n return text\n .replace(capitalLetters, (x) => ' ' + x.toLowerCase())\n .replace(leadingSpaceAndLowerCaseLetter, (substring) => substring.substring(1).toUpperCase())\n}\n\nexport function blobToBase64(blob) {\n const reader = new FileReader()\n reader.readAsDataURL(blob)\n return new Promise((resolve) => {\n reader.onloadend = () => {\n resolve(reader.result)\n }\n })\n}\n\nconst emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]{2,4}$/\n\nexport function validateEmail(email) {\n return emailRegex.test(email)\n}\n\nexport function percentageToFraction(percentage, numberOfDecimalPlaces) {\n return percentage != null ? parseFloat((percentage / 100).toFixed(numberOfDecimalPlaces ?? 4)) : undefined\n}\n\nexport function fractionToPercentage(fraction, numberOfDecimalPlaces) {\n return fraction != null && !isNaN(fraction)\n ? parseFloat((fraction * 100).toFixed(numberOfDecimalPlaces ?? 4))\n : undefined\n}\n\nexport function getLocaleToUse(organisation) {\n return organisation && organisation.locale\n}\n\nexport function getLanguageToUse(user, organisation) {\n return (user && user.language) || (organisation && organisation.language)\n}\n\nexport function browserFileDownload(payload, filename, filetype = '') {\n const url = window.URL.createObjectURL(new Blob([payload], { type: filetype }))\n const link = document.createElement('a')\n link.href = url\n link.setAttribute('download', filename)\n document.body.appendChild(link)\n link.click()\n document.body.removeChild(link)\n window.URL.revokeObjectURL(url)\n return\n}\n\n/**\n * Since we support only a limited number of locales and use en-US localization for the rest,\n * consult a map given by this function.\n * @param {string} localeId\n * @returns\n */\nexport const getSupportedLocale = (localeId) => {\n const localeMap = {\n 'en-AU': 'en-AU',\n 'gsw-FR': 'fr',\n 'br-FR': 'fr',\n 'ca-FR': 'fr',\n 'co-FR': 'fr',\n 'fr-FR': 'fr',\n 'ia-FR': 'fr',\n 'oc-FR': 'fr',\n 'en-DE': 'de',\n 'de-DE': 'de',\n 'nds-DE': 'de',\n 'dsb-DE': 'de',\n 'ksh-DE': 'de',\n 'hsb-DE': 'de',\n 'ca-IT': 'it',\n 'fur-IT': 'it',\n 'de-IT': 'it',\n 'it-IT': 'it',\n 'en-NZ': 'en-AU',\n 'mi-NZ': 'en-AU',\n 'kw-GB': 'en-GB',\n 'en-GB': 'en-GB',\n 'gd-GB': 'en-GB',\n 'cy-GB': 'en-GB',\n 'es-ES': 'es',\n 'pl-PL': 'pl',\n 'pt-PT': 'pt',\n 'pt-BR': 'pt',\n es: 'es',\n fr: 'fr',\n de: 'de',\n it: 'it',\n pt: 'pt',\n pl: 'pl',\n }\n\n const mappedLocale = localeMap[localeId]\n\n if (!mappedLocale) {\n return 'en-US'\n } else {\n return mappedLocale\n }\n}\n\nexport function getProductDetails(resellerName = import.meta.env.VITE_RESELLER) {\n if (resellerName) {\n return RESELLERS[resellerName]\n } else {\n return PRODUCT_DETAILS\n }\n}\n\nexport function organisationIsImperial(organisation) {\n return (\n organisation?.defaultDrawingUnits?.localeCompare('imperial', undefined, {\n sensitivity: 'accent',\n }) === 0\n )\n}\n\nexport function organisationLengthPrecision(organisation) {\n return organisationIsImperial(organisation) ? 4 : 2\n}\n\nexport function organisationDimensionTolerance(organisation) {\n return organisationIsImperial(organisation) ? 0.0001 : 0.01\n}\n\nexport function setPageTitleAndFavicon(theme) {\n const resellerName = import.meta.env.VITE_RESELLER\n const faviconElement = document.getElementById('favicon')\n const productDetails = getProductDetails(resellerName)\n if (resellerName) {\n document.title = productDetails.title\n faviconElement.href = `${productDetails.name}/favicon.ico`\n return productDetails.theme(theme)\n } else {\n document.title = productDetails.title\n faviconElement.href = 'favicon.ico'\n return theme\n }\n}\n\nexport function numbersWithinPercentageTolerance(number1, number2, tolerance) {\n const percentageDifference = 100 * (1 - Math.min(number1, number2) / Math.max(number1, number2))\n return percentageDifference <= tolerance\n}\n\nexport const MAX_DRAWING_UPLOAD_SIZE_BYTES = 4194304\n\nconst quoteItemsDimensions = (quoteItems, useImperialUnits) =>\n quoteItems.map((quoteItem) => {\n const { isImperial, minimumBoundBoxHeight, minimumBoundBoxWidth } = quoteItem\n const long = Math.max(minimumBoundBoxWidth, minimumBoundBoxHeight)\n const short = Math.min(minimumBoundBoxWidth, minimumBoundBoxHeight)\n return {\n long,\n short,\n isImperial,\n longInOrgUnits:\n useImperialUnits === isImperial\n ? long\n : useImperialUnits && !isImperial\n ? mmToInches(long)\n : inchesToMm(long),\n shortInOrgUnits:\n useImperialUnits === isImperial\n ? short\n : useImperialUnits && !isImperial\n ? mmToInches(short)\n : inchesToMm(short),\n }\n })\n\nexport const getLargestDimensions = (quoteItems, organisation) => {\n if (!quoteItems?.length || !organisation) return { long: '', short: '' }\n const orgIsImperial = organisationIsImperial(organisation)\n const dimensions = quoteItemsDimensions(quoteItems, orgIsImperial)\n\n const largestDimension = (key) =>\n dimensions.reduce((max, item) => (item[key] > max[key] ? item : max), dimensions[0])\n\n const largestLong = largestDimension('longInOrgUnits')\n const largestShort = largestDimension('shortInOrgUnits')\n\n const formatDimension = (dimension) =>\n `${formatNumber(dimension, organisation.locale, orgIsImperial)} ${orgIsImperial ? 'in' : 'mm'}`\n\n return {\n long: formatDimension(largestLong.longInOrgUnits),\n short: formatDimension(largestShort.shortInOrgUnits),\n }\n}\n\nexport const secondsToHHMMSS = (seconds) => {\n return new Date(seconds * 1000).toISOString().slice(11, 19)\n}\n\nexport const secondsToDHMS = (seconds) => {\n if (seconds === 0) {\n return String.fromCharCode(8212)\n }\n\n const days = Math.floor(seconds / (3600 * 24))\n const hours = Math.floor((seconds % (3600 * 24)) / 3600)\n const minutes = Math.floor((seconds % 3600) / 60)\n const secs = Math.floor(seconds % 60)\n\n if (days === 0) {\n return `${hours}h ${minutes}m ${secs}s`\n }\n\n return `${days}d ${hours}h ${minutes}m ${secs}s`\n}\n\nexport function getNumberFormat(locale) {\n const formatter = new Intl.NumberFormat(locale)\n\n const numberFormat = {}\n formatter.formatToParts(10000.01).forEach((item) => {\n numberFormat[item.type] = item.value\n })\n\n numberFormat['decimal'] = numberFormat['group'] === '.' ? ',' : '.'\n numberFormat['decimalPlaces'] = numberFormat['fraction'] ? numberFormat['fraction'].length : 0\n\n return numberFormat\n}\n\nexport function loadWebStoreFavicon(src) {\n const favicon = document.getElementById('favicon')\n favicon.href = src\n}\n","import {\n inchesToMm,\n mmToInches,\n numbersWithinPercentageTolerance,\n organisationDimensionTolerance,\n organisationLengthPrecision,\n} from '../GeneralUtils/GeneralUtils'\n\nconst sheetVolume = (thickness, width, height, useImperialUnits) => {\n let result = 0.0\n\n if (!useImperialUnits) {\n thickness = thickness / 1000\n width = width / 1000\n height = height / 1000\n } else {\n thickness = thickness / 12\n width = width / 12\n height = height / 12\n }\n\n result = height * width * thickness\n\n return result\n}\n\nconst rectangleVolume = (thickness, materialLength, width, height, useImperialUnits) => {\n let result = 0.0\n\n if (!useImperialUnits) {\n thickness = thickness / 1000\n materialLength = materialLength / 1000\n width = width / 1000\n height = height / 1000\n } else {\n thickness = thickness / 12\n materialLength = materialLength / 12\n width = width / 12\n height = height / 12\n }\n\n result = materialLength * (width * height - (width - 2 * thickness) * (height - 2 * thickness))\n\n return result\n}\n\nconst circleVolume = (thickness, materialLength, diameter, useImperialUnits) => {\n let result = 0.0\n\n if (!useImperialUnits) {\n thickness = thickness / 1000\n materialLength = materialLength / 1000\n diameter = diameter / 1000\n } else {\n thickness = thickness / 12\n materialLength = materialLength / 12\n diameter = diameter / 12\n }\n\n result = materialLength * Math.PI * (Math.pow(diameter / 2, 2) - Math.pow((diameter - 2 * thickness) / 2, 2))\n\n return result\n}\n\nconst ovalVolume = (thickness, materialLength, width, height, useImperialUnits) => {\n let result = 0.0\n if (!useImperialUnits) {\n thickness = thickness / 1000\n materialLength = materialLength / 1000\n width = width / 1000\n height = height / 1000\n } else {\n thickness = thickness / 12\n materialLength = materialLength / 12\n width = width / 12\n height = height / 12\n }\n\n result =\n materialLength *\n ((width - height) * 2 * thickness +\n Math.PI * (Math.pow(height / 2, 2) - Math.pow((height - 2 * thickness) / 2, 2)))\n\n return result\n}\n\nexport const sheetWeight = ({ density, height, thickness, useImperialUnits, width }) => {\n return density * sheetVolume(thickness, width, height, useImperialUnits)\n}\n\nexport const rectangleWeight = ({ density, height, materialLength, thickness, useImperialUnits, width }) => {\n return density * rectangleVolume(thickness, materialLength, width, height, useImperialUnits)\n}\n\nexport const circleWeight = ({ density, diameter, materialLength, thickness, useImperialUnits }) => {\n return density * circleVolume(thickness, materialLength, diameter, useImperialUnits)\n}\n\nexport const ovalWeight = ({ density, height, materialLength, thickness, useImperialUnits, width }) => {\n return density * ovalVolume(thickness, materialLength, width, height, useImperialUnits)\n}\n\nconst materialWeight = (profile, density, thickness, materialLength, width, height, useImperialUnits, diameter) => {\n let mWeight = 0.0\n switch (profile) {\n case 'Rectangle':\n case 'Square':\n mWeight = rectangleWeight({\n density,\n thickness,\n materialLength,\n width,\n height,\n useImperialUnits,\n })\n break\n case 'Circle':\n mWeight = circleWeight({\n density,\n thickness,\n materialLength,\n diameter,\n useImperialUnits,\n })\n break\n case 'FlatSidedOval':\n mWeight = ovalWeight({\n density,\n thickness,\n materialLength,\n width,\n height,\n useImperialUnits,\n })\n break\n default:\n mWeight = sheetWeight({\n density,\n thickness,\n width,\n height,\n useImperialUnits,\n })\n break\n }\n return mWeight\n}\n\nexport const calculateRatePrice = (\n val,\n profile,\n { density, diameter, height, materialLength, thickness, useImperialUnits, width }\n) => {\n let result = 0.0\n\n if (profile === 'Square') {\n height = width\n }\n\n const mWeight = materialWeight(\n profile,\n density,\n thickness,\n materialLength,\n width,\n height,\n useImperialUnits,\n diameter\n )\n\n if (mWeight > 0) {\n result = val / mWeight\n result = parseFloat(result).toFixed(2)\n }\n return result\n}\n\nexport const calculateCostPrice = (\n val,\n profile,\n { density, diameter, height, materialLength, thickness, useImperialUnits, width }\n) => {\n let result = 0.0\n\n if (profile === 'Square') {\n height = width\n }\n\n const mWeight = materialWeight(\n profile,\n density,\n thickness,\n materialLength,\n width,\n height,\n useImperialUnits,\n diameter\n )\n\n if (mWeight > 0) {\n result = val * mWeight\n result = parseFloat(result).toFixed(2)\n }\n return result\n}\n\nexport const getPartDimensions = (part) => {\n if (!part) return null\n return {\n profile: part.profile,\n displayBoundBoxHeight: part.displayBoundBoxHeight,\n displayBoundBoxWidth: part.displayBoundBoxWidth,\n rotaryProfileLength: part.rotaryProfileLength,\n rotaryProfileWidth: part.rotaryProfileWidth,\n diameter: part.diameter,\n isImperial: part.isImperial,\n }\n}\n\nexport const getPartDimensionsDisplay = (part, dimensionFormat) => {\n if (!part || !part.profile || typeof dimensionFormat !== 'function') return null\n switch (part.profile.toLowerCase()) {\n case 'square':\n return part.rotaryProfileWidth\n case 'circle':\n return part.diameter\n default:\n return `${dimensionFormat(part.rotaryProfileWidth)} x ${dimensionFormat(part.rotaryProfileLength)}`\n }\n}\n\nexport const getRecordDimensionsDisplay = (sheet, dimensionFormat) => {\n if (!sheet || !sheet.profile || typeof dimensionFormat !== 'function') return null\n switch (sheet.profile.toLowerCase()) {\n case 'square':\n return sheet.sheetWidth\n case 'circle':\n return sheet.diameter\n default:\n return `${dimensionFormat(sheet.sheetWidth)} x ${dimensionFormat(sheet.sheetHeight)}`\n }\n}\n\nexport const isProfileSizeFit = (sheet, partDimensions, organisation) => {\n if (\n !sheet ||\n !partDimensions ||\n (sheet.profile && partDimensions.profile && sheet.profile !== partDimensions.profile)\n ) {\n return false\n }\n\n if (!sheet.profile) {\n //we check later if a flat part can fit on a sheet, currently we do not have the means\n //of letting a user know that their part wont fit on a sheet on the part details panel\n //this if statement can be removed when we handle this case better later\n return true\n }\n\n const orgUsesImperial = organisation.defaultDrawingUnits.toLowerCase() === 'imperial'\n const orgAndPartUnitsAreEqual = partDimensions.isImperial === orgUsesImperial\n\n const convertedPartDimensions = {\n rotaryProfileWidth:\n !partDimensions.rotaryProfileWidth || orgAndPartUnitsAreEqual\n ? partDimensions.rotaryProfileWidth\n : orgUsesImperial\n ? mmToInches(partDimensions.rotaryProfileWidth)\n : inchesToMm(partDimensions.rotaryProfileWidth),\n rotaryProfileLength:\n !partDimensions.rotaryProfileLength || orgAndPartUnitsAreEqual\n ? partDimensions.rotaryProfileLength\n : orgUsesImperial\n ? mmToInches(partDimensions.rotaryProfileLength)\n : inchesToMm(partDimensions.rotaryProfileLength),\n diameter:\n !partDimensions.diameter || orgAndPartUnitsAreEqual\n ? partDimensions.diameter\n : orgUsesImperial\n ? mmToInches(partDimensions.diameter)\n : inchesToMm(partDimensions.diameter),\n displayBoundBoxHeight:\n !partDimensions.displayBoundBoxHeight || orgAndPartUnitsAreEqual\n ? partDimensions.displayBoundBoxHeight\n : orgUsesImperial\n ? mmToInches(partDimensions.displayBoundBoxHeight)\n : inchesToMm(partDimensions.displayBoundBoxHeight),\n displayBoundBoxWidth:\n !partDimensions.displayBoundBoxWidth || orgAndPartUnitsAreEqual\n ? partDimensions.displayBoundBoxWidth\n : orgUsesImperial\n ? mmToInches(partDimensions.displayBoundBoxWidth)\n : inchesToMm(partDimensions.displayBoundBoxWidth),\n }\n\n const width = partDimensions.profile\n ? convertedPartDimensions.rotaryProfileWidth\n : convertedPartDimensions.displayBoundBoxWidth\n const height = partDimensions.profile\n ? convertedPartDimensions.rotaryProfileLength\n : convertedPartDimensions.displayBoundBoxHeight\n\n switch (sheet.profile?.toLowerCase()) {\n case 'square':\n return dimensionsSameWithTolerance(\n sheet.sheetWidth,\n convertedPartDimensions.rotaryProfileWidth,\n organisation,\n true\n )\n case 'circle':\n return dimensionsSameWithTolerance(sheet.diameter, convertedPartDimensions.diameter, organisation, true)\n case 'flatsidedoval':\n return (\n dimensionsSameWithTolerance(sheet.sheetWidth, width, organisation, true) &&\n dimensionsSameWithTolerance(sheet.sheetHeight, height, organisation, true)\n )\n case 'rectangle':\n return (\n (dimensionsSameWithTolerance(sheet.sheetWidth, width, organisation, true) &&\n dimensionsSameWithTolerance(sheet.sheetHeight, height, organisation, true)) ||\n (dimensionsSameWithTolerance(sheet.sheetHeight, width, organisation, true) &&\n dimensionsSameWithTolerance(sheet.sheetWidth, height, organisation, true))\n )\n default:\n return (\n partFitsWithTolerance(sheet.sheetHeight, width, organisation) &&\n partFitsWithTolerance(sheet.sheetWidth, height, organisation)\n )\n }\n}\n\nexport const areProfileSizeEqual = (size1, size2) => {\n if (!size1 || !size1.profile || !size2 || !size2.profile) return true\n return (\n size1.rotaryProfileWidth === size2.rotaryProfileWidth &&\n size1.rotaryProfileLength === size2.rotaryProfileLength &&\n size1.diameter === size2.diameter &&\n size1.displayBoundBoxWidth === size2.displayBoundBoxWidth &&\n size1.displayBoundBoxHeight === size2.displayBoundBoxHeight\n )\n}\n\nexport const thicknessSameWithTolerance = (\n sheet,\n partThickness,\n partDimensions,\n organisation,\n usePercentage = false\n) => {\n const orgUsesImperial = organisation.defaultDrawingUnits.toLowerCase() === 'imperial'\n const orgAndPartUnitsAreEqual = partDimensions.isImperial === orgUsesImperial\n const convertedPartThickness = orgAndPartUnitsAreEqual\n ? partThickness\n : orgUsesImperial\n ? mmToInches(partThickness)\n : inchesToMm(partThickness)\n\n return dimensionsSameWithTolerance(sheet.thickness, convertedPartThickness, organisation, usePercentage)\n}\n\nconst PROFILE_SIZE_TOLERANCE_PERCENTAGE = 2\n\nconst dimensionsSameWithTolerance = (sheetDimension, partDimension, organisation, usePercentage = false) => {\n const precision = organisationLengthPrecision(organisation)\n if (usePercentage) {\n return numbersWithinPercentageTolerance(\n partDimension?.toFixed(precision),\n sheetDimension?.toFixed(precision),\n PROFILE_SIZE_TOLERANCE_PERCENTAGE\n )\n } else {\n return (\n Math.abs(partDimension?.toFixed(precision) - sheetDimension?.toFixed(precision)) <=\n organisationDimensionTolerance(organisation)\n )\n }\n}\n\nconst partFitsWithTolerance = (sheetDimension, partDimension, organisation) => {\n const precision = organisationLengthPrecision(organisation)\n return (\n partDimension?.toFixed(precision) - sheetDimension?.toFixed(precision) <=\n organisationDimensionTolerance(organisation)\n )\n}\n","import userflow from 'userflow.js'\n\nexport const initUserFlow = ({ language, localeToUse, organisation, planMetadata, reseller, user }) => {\n if (user) {\n userflow.init(import.meta.env.VITE_USERFLOW_TOKEN)\n userflow.identify(user.userId, {\n name: user.firstName + (user.lastName ? ' ' : '') + user.lastName,\n email: user.emailAddress,\n locale_code: language, // UserFlow docs state that locale_code must set to the user's preferred language\n locality: localeToUse, // Send locality, which has a special meaning in our app\n language: language,\n organisation_id: organisation ? organisation.organisationId : undefined,\n organisation_units: organisation ? organisation.defaultDrawingUnits : undefined,\n reseller,\n plan_metadata: planMetadata ? JSON.stringify(planMetadata) : undefined,\n is_legacy_materials: organisation ? organisation.isLegacyMaterials : undefined,\n promotional_codes: user.promotionalCodes ? JSON.stringify(user.promotionalCodes) : undefined,\n device_width: window.screen.width,\n device_height: window.screen.height,\n user_agent: window.navigator.userAgent,\n stripe_id: organisation ? organisation.paymentGatewayCustomerId : undefined,\n help_tips_toggle: user.showUserflowTooltips,\n laser_k_w: organisation ? organisation.cuttingTechnologyPower : undefined,\n quoting_experience: 2023,\n help_widget: user.showHelpWidget ? 'show' : 'hide'\n })\n }\n}\n\nexport const showTooltipsDisplay = (userId, showUserflowTooltips) => {\n if (userId) {\n userflow.init(import.meta.env.VITE_USERFLOW_TOKEN)\n userflow.identify(userId,\n { help_tips_toggle: showUserflowTooltips }\n )\n }\n}\n\nexport const showHelpWidget = (userId, showHelpWidget) => {\n if (userId) {\n userflow.init(import.meta.env.VITE_USERFLOW_TOKEN)\n userflow.identify(userId,\n { help_widget: showHelpWidget ? 'show' : 'hide' }\n )\n }\n}\n\nexport const showResourceCentre = (showResourceCentre) => {\n userflow.setResourceCenterLauncherHidden(!showResourceCentre)\n}\n","import { EventType, LogLevel, PublicClientApplication } from '@azure/msal-browser'\n\nimport { Paths } from './common/utils'\n\n// Browser check variables\n// If you support IE, our recommendation is that you sign-in using Redirect APIs\n// If you as a developer are testing using Edge InPrivate mode, please add \"isEdge\" to the if check\nconst ua = window.navigator.userAgent\nconst msie = ua.indexOf('MSIE ')\nconst msie11 = ua.indexOf('Trident/')\nconst msedge = ua.indexOf('Edge/')\nconst firefox = ua.indexOf('Firefox')\nconst isIE = msie > 0 || msie11 > 0\nconst isEdge = msedge > 0\nconst isFirefox = firefox > 0 // Only needed if you need to support the redirect flow in Firefox incognito\n\nconst tenantSubdomain = import.meta.env.VITE_AUTH_TENANT_SUBDOMAIN\nconst signInPolicy = import.meta.env.VITE_AUTH_SIGN_IN_POLICY\nconst signUpPolicy = import.meta.env.VITE_AUTH_SIGN_UP_POLICY\nconst xeroPolicy = import.meta.env.VITE_AUTH_XERO_POLICY\nconst resetPasswordPolicy = import.meta.env.VITE_AUTH_RESET_PASSWORD_POLICY\nconst editProfilePolicy = import.meta.env.VITE_AUTH_EDIT_PROFILE_POLICY\nconst clientId = import.meta.env.VITE_AUTH_CLIENT_ID\nconst cacheLocation = import.meta.env.VITE_AUTH_CACHE_LOCATION\n\n// With the changes below, we are allowing redirects to whatever subdomain has loaded the webstore app as well as\n// allowing loading the app from the main domain\n\nconst redirectUri = `https://${window.location.host}` // we are no longer using import.meta.env.VITE_AUTH_REDIRECT_URI here\nconst registerRedirectUri = `${redirectUri}/register` // not using import.meta.env.VITE_AUTH_REGISTER_REDIRECT_URI anymore\nconst postLogoutRedirectUri = `${redirectUri}/signout` // not using import.meta.env.VITE_POSTLOGOUT_REDIRECT_URI either\n\nconst tenant = `${tenantSubdomain}.onmicrosoft.com`\nconst b2cInstance = `https://${tenantSubdomain}.b2clogin.com/tfp/`\nconst scopes = [`https://${tenant}/api/quotemate.api.read`, `https://${tenant}/api/quotemate.api.write`]\nconst signInAuthority = `${b2cInstance}${tenant}/${signInPolicy}`\nconst signUpAuthority = `${b2cInstance}${tenant}/${signUpPolicy}`\nconst xeroAuthority = `${b2cInstance}${tenant}/${xeroPolicy}`\nconst resetPasswordAuthority = `${b2cInstance}${tenant}/${resetPasswordPolicy}`\nconst editProfileAuthority = `${b2cInstance}${tenant}/${editProfilePolicy}`\nconst logLevel = import.meta.env.VITE_ENV === 'development' ? LogLevel.Warning : LogLevel.Error\n\nconst params = new URLSearchParams(window.location.search)\nconst hasXero = params.has('xero')\nconst authority = hasXero ? xeroAuthority : signInAuthority\n\nexport const msalConfig = {\n auth: {\n clientId: clientId,\n authority: authority,\n knownAuthorities: [signInAuthority, xeroAuthority],\n validateAuthority: false,\n redirectUri: redirectUri,\n postLogoutRedirectUri: postLogoutRedirectUri,\n scopes: scopes,\n },\n cache: {\n cacheLocation,\n storeAuthStateInCookie: isIE || isEdge || isFirefox,\n },\n system: {\n loggerOptions: {\n logLevel: logLevel,\n loggerCallback: (level, message, containsPii) => {\n if (containsPii) {\n return\n }\n switch (level) {\n case LogLevel.Error:\n console.error(message)\n return\n case LogLevel.Info:\n console.info(message)\n return\n case LogLevel.Verbose:\n console.debug(message)\n return\n case LogLevel.Warning:\n console.warn(message)\n return\n default:\n return\n }\n },\n },\n },\n}\n\n// Add here scopes for id token to be used at MS Identity Platform endpoints.\nexport const loginRequest = {\n scopes,\n authority: authority,\n}\n\nexport const resetPasswordRequest = {\n scopes,\n authority: resetPasswordAuthority,\n}\n\nexport const editProfileRequest = {\n scopes,\n authority: editProfileAuthority,\n}\n\nexport const signUpRequest = {\n scopes,\n authority: signUpAuthority,\n redirectUri: registerRedirectUri,\n}\n\nexport const tokenRequest = {\n scopes,\n}\n\nexport const msalInstance = new PublicClientApplication(msalConfig)\n\nmsalInstance.enableAccountStorageEvents()\n\n// Account selection logic is app dependent. Adjust as needed for different use cases.\nconst accounts = msalInstance.getAllAccounts()\nif (accounts && accounts.length > 0) {\n msalInstance.setActiveAccount(accounts[0])\n}\n\nmsalInstance.addEventCallback((message) => {\n if (message?.eventType === EventType.LOGIN_SUCCESS && message?.payload?.account) {\n const account = message.payload.account\n msalInstance.setActiveAccount(account)\n } else if (message?.eventType === EventType.ACCOUNT_REMOVED) {\n msalInstance.setActiveAccount(null)\n window.location.href = Paths.SIGNOUT_PATHNAME\n } else if (message.eventType === EventType.ACTIVE_ACCOUNT_CHANGED) {\n const activeAccount = msalInstance.getActiveAccount()\n if (activeAccount) {\n msalInstance.setActiveAccount(activeAccount)\n }\n window.location.href = Paths.DASHBOARD_PATHNAME\n }\n})\n","import { InteractionRequiredAuthError } from '@azure/msal-browser'\n\nimport { msalInstance, tokenRequest } from '@/authConfig'\n\nconst getAccessToken = async () => {\n try {\n const tokenResponse = await msalInstance.acquireTokenSilent(tokenRequest)\n const token = tokenResponse?.accessToken ?? null\n const tokenExpireOn = tokenResponse?.expiresOn ?? null\n\n return {\n token,\n tokenExpireOn,\n }\n } catch (error) {\n console.error('Error acquiring token:', error.message)\n if (error instanceof InteractionRequiredAuthError) {\n try {\n const tokenResponse = await msalInstance.acquireTokenPopup(tokenRequest)\n const token = tokenResponse?.accessToken ?? null\n const tokenExpireOn = tokenResponse?.expiresOn ?? null\n\n return {\n token,\n tokenExpireOn,\n }\n } catch (innerError) {\n console.error('Failed to acquire token via popup:', innerError.message)\n return {\n token: null,\n tokenExpireOn: null,\n }\n }\n } else {\n console.error('Failed to acquire token:', error.message)\n return {\n token: null,\n tokenExpireOn: null,\n }\n }\n }\n}\n\nexport { getAccessToken }\n","import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'\n\nimport { getAccessToken } from '@/common/helpers/auth/getAccessToken'\n\n// Create our baseQuery instance\nconst baseQuery = fetchBaseQuery({\n baseUrl: `${import.meta.env.VITE_API_SERVER_URL}`,\n prepareHeaders: async (headers) => {\n const { token } = await getAccessToken()\n\n if (token) {\n headers.set('authorization', `Bearer ${token}`)\n }\n return headers\n },\n})\n\n/**\n * Create a base API to inject endpoints into elsewhere.\n * Components using this API should import from the injected site,\n * in order to get the appropriate types,\n * and to ensure that the file injecting the endpoints is loaded\n */\nexport const api = createApi({\n /**\n * `reducerPath` is optional and will not be required by most users.\n * This is useful if you have multiple API definitions,\n * e.g. where each has a different domain, with no interaction between endpoints.\n * Otherwise, a single API definition should be used in order to support tag invalidation,\n * among other features\n */\n reducerPath: 'api',\n /**\n * A bare bones base query would just be `baseQuery: fetchBaseQuery({ baseUrl: '/' })`\n */\n baseQuery,\n /**\n * Tag types must be defined in the original API definition\n * for any tags that would be provided by injected endpoints\n */\n tagTypes: [],\n /**\n * This api has endpoints injected in adjacent files,\n * which is why no endpoints are shown below.\n * If you want all endpoints defined in the same file, they could be included here instead\n */\n endpoints: () => ({}),\n})\n","import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'\n\nimport { getAccessToken } from '@/common/helpers/auth/getAccessToken'\n\nconst baseQuery = fetchBaseQuery({\n baseUrl: `${import.meta.env.VITE_API_SERVER_URL}`,\n prepareHeaders: async (headers) => {\n const { token } = await getAccessToken()\n\n if (token) {\n headers.set('authorization', `Bearer ${token}`)\n }\n return headers\n },\n})\n\nexport const quickPartsApi = createApi({\n reducerPath: 'quickPartsApi',\n baseQuery,\n tagTypes: ['QuickParts'],\n endpoints: (builder) => ({\n getQuickParts: builder.query({\n query: (params) => ({\n url: '/quickparts',\n params,\n }),\n providesTags: ['QuickParts'],\n }),\n\n getQuickPart: builder.query({\n query: ({ id, params }) => ({\n url: `/quickparts/${id}`,\n params,\n }),\n providesTags: ['QuickParts'],\n }),\n\n getQuickPartsDrawingDto: builder.query({\n query: ({ name, part }) => ({\n url: '/quickparts/drawing',\n method: 'POST',\n body: { part, name },\n }),\n }),\n\n getQuickPartsSvg: builder.query({\n query: (part) => ({\n url: '/quickparts/svg',\n method: 'POST',\n body: part,\n responseHandler: (response) => {\n const svg = response.text()\n return svg\n },\n }),\n }),\n }),\n})\n\nexport const {\n useGetQuickPartQuery,\n useGetQuickPartsQuery,\n useGetQuickPartsSvgQuery,\n useLazyGetQuickPartsDrawingDtoQuery,\n} = quickPartsApi\n","import { createApi, fetchBaseQuery } from '@reduxjs/toolkit/query/react'\n\n// Create our baseQuery instance\nconst baseQuery = fetchBaseQuery({\n baseUrl: `${import.meta.env.VITE_API_SERVER_URL}/store`,\n prepareHeaders: (headers, { getState }) => {\n // By default, if we have a token in the store, let's use that for authenticated requests\n const token = getState().webStoreAuth.token\n if (token) {\n headers.set('authorization', `Bearer ${token}`)\n }\n return headers\n },\n})\n\n/**\n * Create a base API to inject endpoints into elsewhere.\n * Components using this API should import from the injected site,\n * in order to get the appropriate types,\n * and to ensure that the file injecting the endpoints is loaded\n */\nexport const webStoreApi = createApi({\n /**\n * `reducerPath` is optional and will not be required by most users.\n * This is useful if you have multiple API definitions,\n * e.g. where each has a different domain, with no interaction between endpoints.\n * Otherwise, a single API definition should be used in order to support tag invalidation,\n * among other features\n */\n reducerPath: 'webStoreApi',\n /**\n * A bare bones base query would just be `baseQuery: fetchBaseQuery({ baseUrl: '/' })`\n */\n baseQuery,\n /**\n * Tag types must be defined in the original API definition\n * for any tags that would be provided by injected endpoints\n */\n tagTypes: [],\n /**\n * This api has endpoints injected in adjacent files,\n * which is why no endpoints are shown below.\n * If you want all endpoints defined in the same file, they could be included here instead\n */\n endpoints: () => ({}),\n})\n","import { api } from './api'\n\nexport const userApi = api\n .enhanceEndpoints({\n addTagTypes: ['User'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getUser: builder.query({\n query: () => '/users',\n providesTags: ['User'],\n }),\n\n getOrCreateUserContext: builder.mutation({\n query: ({ user }) => ({\n url: '/users/context/',\n method: 'POST',\n body: user,\n }),\n }),\n\n getUserOrganisations: builder.query({\n query: ({ userId }) => `/users/${userId}/organisations`,\n transformResponse: (response) => {\n response.sort((a, b) => {\n return a.name.localeCompare(b.name)\n })\n\n return response\n },\n }),\n\n createUser: builder.mutation({\n query: ({ user }) => ({\n url: '/users',\n method: 'POST',\n body: user,\n }),\n }),\n\n updateUser: builder.mutation({\n query: ({ user, userId }) => ({\n url: `/users/${userId}`,\n method: 'POST',\n body: user,\n }),\n invalidatesTags: ['User'],\n }),\n\n createOrganisation: builder.mutation({\n query: ({ options, userId }) => ({\n url: `/users/${userId}/organisation`,\n method: 'POST',\n body: options,\n }),\n }),\n\n startTrial: builder.mutation({\n query: ({ userId }) => ({\n url: `/users/${userId}/startTrial`,\n method: 'POST',\n }),\n }),\n }),\n })\n\nexport const {\n useCreateOrganisationMutation,\n useCreateUserMutation,\n useGetOrCreateUserContextMutation,\n useGetUserOrganisationsQuery,\n useGetUserQuery,\n useLazyGetUserOrganisationsQuery,\n useStartTrialMutation,\n useUpdateUserMutation,\n} = userApi\n","import { createSlice } from '@reduxjs/toolkit'\n\nimport { userApi } from '../services/user'\n\nconst initialState = {\n userRole: 'user',\n currentUser: undefined,\n currentUserOrganisations: [],\n appIsLoading: true,\n appBarState: {\n hidden: false,\n menuHidden: false,\n myAccountHidden: false,\n title: '',\n titlePosition: 'center',\n toolbarItems: [],\n justifyItems: 'space-around',\n showSubscriptionBar: false,\n showTrialWithSubscriptionBar: false,\n showUpdatePaymentBar: false,\n freePlan: false,\n },\n activationState: {\n selectedPrice: {},\n billingAddress: {},\n taxIdentifier: '',\n },\n language: 'en-US',\n locale: 'en-US',\n languageList: {\n 'en-US': {\n nativeName: 'English',\n region: 'US',\n },\n },\n displayQuotePricing: true,\n subscription: {},\n showDrawerMenu: false,\n showNestedDrawerMenu: null,\n}\n\nexport const appSlice = createSlice({\n name: 'appSlice',\n initialState,\n reducers: {\n setUserRole: (state, action) => {\n state.userRole = action.payload\n },\n\n setCurrentUser: (state, action) => {\n state.currentUser = { ...state.currentUser, ...action.payload }\n },\n\n setAppIsLoading: (state, action) => {\n state.appIsLoading = action.payload\n },\n\n setAppBarState: (state, action) => {\n state.appBarState = { ...state.appBarState, ...action.payload }\n },\n\n resetAppBarState: (state) => {\n state.appBarState = initialState.appBarState\n },\n\n setActivationState: (state, action) => {\n state.activationState = { ...state.activationState, ...action.payload }\n },\n\n setLanguage: (state, action) => {\n state.language = action.payload\n },\n\n setLocale: (state, action) => {\n state.locale = action.payload\n },\n\n setLanguageList: (state, action) => {\n state.languageList = action.payload\n },\n\n setDisplayQuotePricing: (state, action) => {\n state.displayQuotePricing = action.payload\n },\n\n setSubscription: (state, action) => {\n state.subscription = action.payload\n },\n\n setShowDrawerMenu: (state, action) => {\n state.showDrawerMenu = action.payload\n },\n\n setShowNestedDrawerMenu: (state, action) => {\n state.showNestedDrawerMenu = action.payload\n },\n setCurrentUserOrganisations: (state, action) => {\n state.currentUserOrganisations = action.payload\n },\n },\n extraReducers: (builder) => {\n builder\n .addMatcher(userApi.endpoints.getOrCreateUserContext.matchFulfilled, (state, { payload }) => {\n state.currentUser = { ...state.currentUser, ...payload.user }\n })\n .addMatcher(userApi.endpoints.updateUser.matchFulfilled, (state, { payload }) => {\n state.currentUser = { ...state.currentUser, ...payload.user }\n })\n .addMatcher(userApi.endpoints.getUserOrganisations.matchFulfilled, (state, { payload }) => {\n state.currentUserOrganisations = Array.isArray(payload) ? payload : []\n })\n },\n})\n\nexport const selectUserRole = (state) => state.appSlice.userRole\nexport const selectCurrentUser = (state) => state.appSlice.currentUser\nexport const selectCurrentUserId = (state) => state.appSlice.currentUser?.userId\nexport const selectCurrentUserOrganisations = (state) => state.appSlice.currentUserOrganisations\nexport const selectAppIsLoading = (state) => state.appSlice.appIsLoading\nexport const selectAppBarState = (state) => state.appSlice.appBarState\nexport const selectActivationState = (state) => state.appSlice.activationState\nexport const selectLanguage = (state) => state.appSlice.language\nexport const selectLocale = (state) => state.appSlice.locale\nexport const selectLanguageList = (state) => state.appSlice.languageList\nexport const selectDisplayQuotePricing = (state) => state.appSlice.displayQuotePricing\nexport const selectSubscription = (state) => state.appSlice.subscription\nexport const selectShowDrawerMenu = (state) => state.appSlice.showDrawerMenu\nexport const selectShowNestedDrawerMenu = (state) => state.appSlice.showNestedDrawerMenu\nexport const selectIsSubscriptionSet = (state) => {\n const subscriptionExists = state?.appSlice?.subscription\n return Boolean(subscriptionExists && Object.keys(subscriptionExists).length)\n}\nexport const selectIsCurrentUserAdmin = (state) => state.appSlice.userRole === 'administrator'\nexport const selectIsCurrentUserOwner = (state) => state.appSlice.userRole === 'owner'\n\nexport const {\n resetAppBarState,\n setActivationState,\n setAppBarState,\n setAppIsLoading,\n setCurrentUser,\n setCurrentUserOrganisations,\n setDisplayQuotePricing,\n setLanguage,\n setLanguageList,\n setLocale,\n setShowDrawerMenu,\n setShowNestedDrawerMenu,\n setSubscription,\n setUserRole,\n} = appSlice.actions\n\nexport default appSlice.reducer\n","import { api } from './api'\n\nexport const contactsApi = api\n .enhanceEndpoints({\n addTagTypes: ['Contacts'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getContacts: builder.query({\n query: ({ customerId, includeDeleted, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts`,\n params: { includeDeleted },\n }),\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'Contacts', id: 'LIST' },\n ...result.map(({ contactId }) => ({ type: 'Contacts', id: contactId })),\n ]\n : [{ type: 'Contacts', id: 'LIST' }],\n transformResponse: (response) => {\n // sort contacts by name\n return response?.sort((a, b) => a.name.toLowerCase().localeCompare(b.name.toLowerCase()))\n },\n }),\n\n getContact: builder.query({\n query: ({ contactId, customerId, organisationId }) =>\n `/organisations/${organisationId}/customers/${customerId}/contacts/${contactId}`,\n providesTags: (_result, _error, arg) => [{ type: 'Contacts', id: arg.contactId }],\n }),\n\n createContact: builder.mutation({\n query: ({ contact, customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts`,\n method: 'POST',\n body: contact,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Customers', id: arg.customerId },\n { type: 'Contacts', id: 'LIST' },\n ],\n }),\n\n updateContact: builder.mutation({\n query: ({ contact, contactId, customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts/${contactId}`,\n method: 'PUT',\n body: contact,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Customers', id: arg.customerId },\n { type: 'Contacts', id: arg.contactId },\n ],\n }),\n\n deleteContacts: builder.mutation({\n query: ({ contactIds, customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts`,\n method: 'DELETE',\n body: contactIds,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Contacts', id: 'LIST' },\n arg.contactIds.map((contactId) => ({ type: 'Contacts', id: contactId })),\n ],\n }),\n\n archiveContact: builder.mutation({\n query: ({ contactId, customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts/${contactId}/archive`,\n method: 'PUT',\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'Contacts', id: arg.contactId }],\n }),\n\n unarchiveContact: builder.mutation({\n query: ({ contactId, customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts/${contactId}/unarchive`,\n method: 'PUT',\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'Contacts', id: arg.contactId }],\n }),\n\n setAsDefaultContact: builder.mutation({\n query: ({ contactId, customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/contacts/${contactId}/default`,\n method: 'PUT',\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'Contacts', id: arg.contactId }],\n }),\n }),\n })\n\nexport const {\n useArchiveContactMutation,\n useCreateContactMutation,\n useDeleteContactsMutation,\n useGetContactQuery,\n useGetContactsQuery,\n useSetAsDefaultContactMutation,\n useUnarchiveContactMutation,\n useUpdateContactMutation,\n} = contactsApi\n","import { createSlice } from '@reduxjs/toolkit'\n\nimport { customersApi } from '../services/customers'\n\nconst initialState = {\n customers: [],\n selectedCustomer: null,\n selectedCustomerId: null,\n showAddCustomerForm: false,\n}\n\nexport const customersSlice = createSlice({\n name: 'customersSlice',\n initialState,\n reducers: {\n setCustomers: (state, action) => {\n state.customers = action.payload\n },\n setSelectedCustomer: (state, action) => {\n state.selectedCustomer = action.payload\n },\n setSelectedCustomerId: (state, action) => {\n state.selectedCustomerId = action.payload\n },\n setShowAddCustomerForm: (state, action) => {\n state.showAddCustomerForm = action.payload\n },\n },\n extraReducers: (builder) => {\n builder.addMatcher(customersApi.endpoints.getCustomers.matchFulfilled, (state, { payload }) => {\n state.customers = payload\n })\n },\n})\n\nexport const selectCustomers = (state) => state.customersSlice.customers\nexport const selectSelectedCustomer = (state) => state.customersSlice.selectedCustomer\nexport const selectSelectedCustomerId = (state) => state.customersSlice.selectedCustomerId\nexport const selectShowAddCustomerForm = (state) => state.customersSlice.showAddCustomerForm\n\nexport const { setCustomers, setSelectedCustomer, setSelectedCustomerId, setShowAddCustomerForm } =\n customersSlice.actions\n\nexport default customersSlice.reducer\n","import { setSelectedCustomer, setSelectedCustomerId } from '../slices/customersSlice'\n\nimport { api } from './api'\n\nexport const customersApi = api\n .enhanceEndpoints({\n addTagTypes: ['Customers'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getCustomers: builder.query({\n query: ({ organisationId, params }) => ({\n url: `/organisations/${organisationId}/customers`,\n params,\n }),\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'Customers', id: 'LIST' },\n ...result.map(({ customerId }) => ({ type: 'Customers', id: customerId })),\n ]\n : [{ type: 'Customers', id: 'LIST' }],\n transformResponse: (response) => {\n return response?.sort((a, b) =>\n a?.companyName?.toLowerCase()?.localeCompare(b?.companyName?.toLowerCase())\n )\n },\n }),\n\n getCustomer: builder.query({\n query: ({ customerId, organisationId }) => `/organisations/${organisationId}/customers/${customerId}`,\n providesTags: (result, _error, arg) => [{ type: 'Customers', id: arg.customerId }],\n }),\n\n createCustomer: builder.mutation({\n query: ({ customer, organisationId }) => ({\n url: `/organisations/${organisationId}/customers`,\n method: 'POST',\n body: customer,\n }),\n invalidatesTags: [{ type: 'Customers', id: 'LIST' }],\n }),\n\n updateCustomer: builder.mutation({\n query: ({ customer, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customer.customerId}`,\n method: 'PUT',\n body: customer,\n }),\n async onQueryStarted({ customer, organisationId }, { dispatch, queryFulfilled }) {\n dispatch(\n customersApi.util.updateQueryData(\n 'getCustomers',\n { customerId: customer.customerId, organisationId },\n (draft) => {\n const index = draft.findIndex((c) => c.customerId === customer.customerId)\n draft[index] = customer\n }\n )\n )\n\n try {\n await queryFulfilled\n dispatch(setSelectedCustomer(customer))\n dispatch(setSelectedCustomerId(customer.customerId))\n dispatch(customersApi.util.invalidateTags([{ type: 'Customers', id: 'LIST' }]))\n } catch {\n //patchResult.undo()\n dispatch(\n customersApi.util.invalidateTags([\n { type: 'Customers', id: customer.customerId },\n { type: 'Customers', id: 'LIST' },\n ])\n )\n }\n },\n }),\n\n archiveCustomer: builder.mutation({\n query: ({ customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (result, error, arg) => [{ type: 'Customers', id: arg.customerId }],\n }),\n\n unarchiveCustomer: builder.mutation({\n query: ({ customerId, organisationId }) => ({\n url: `/organisations/${organisationId}/customers/${customerId}/unarchive`,\n method: 'PUT',\n }),\n invalidatesTags: (result, error, arg) => [{ type: 'Customers', id: arg.customerId }],\n }),\n }),\n })\n\nexport const {\n useArchiveCustomerMutation,\n useCreateCustomerMutation,\n useGetCustomerQuery,\n useGetCustomersQuery,\n useLazyGetCustomerQuery,\n useUnarchiveCustomerMutation,\n useUpdateCustomerMutation,\n} = customersApi\n","import { createSlice } from '@reduxjs/toolkit'\n\nimport { contactsApi } from '../services/contacts'\nimport { customersApi } from '../services/customers'\n\nconst initialState = {\n allContacts: [],\n customerContacts: [],\n selectedContact: undefined,\n}\n\nexport const contactsSlice = createSlice({\n name: 'contactsSlice',\n initialState,\n reducers: {\n setAllContacts: (state, action) => {\n state.allContacts = action.payload\n },\n setCustomerContacts: (state, action) => {\n state.customerContacts = action.payload\n },\n setSelectedContact: (state, action) => {\n state.selectedContact = action.payload\n },\n },\n extraReducers: (builder) => {\n builder\n .addMatcher(customersApi.endpoints.getCustomers.matchFulfilled, (state, { payload }) => {\n state.allContacts = payload.reduce((acc, customer) => {\n return [...acc, ...customer.contacts]\n }, [])\n })\n\n .addMatcher(contactsApi.endpoints.getContacts.matchFulfilled, (state, { payload }) => {\n state.customerContacts = payload\n })\n },\n})\n\nexport const selectAllContacts = (state) => state.contactsSlice.allContacts\nexport const selectCustomerContacts = (state) => state.contactsSlice.customerContacts\nexport const selectSelectedContact = (state) => state.contactsSlice.selectedContact\n\nexport const { setAllContacts, setCustomerContacts, setSelectedContact } = contactsSlice.actions\n\nexport default contactsSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedCuttingTechId: null,\n}\n\nexport const cuttingTechnologiesSlice = createSlice({\n name: 'cuttingTechnologiesSlice',\n initialState,\n reducers: {\n setSelectedCuttingTechId: (state, action) => {\n state.selectedCuttingTechId = action.payload\n },\n },\n})\n\nexport const selectSelectedCuttingTechId = (state) => state.cuttingTechnologiesSlice.selectedCuttingTechId\n\nexport const { setSelectedCuttingTechId } = cuttingTechnologiesSlice.actions\n\nexport default cuttingTechnologiesSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedMaterialId: null,\n showAddMaterialForm: false,\n}\n\nexport const materialsSlice = createSlice({\n name: 'materialsSlice',\n initialState,\n reducers: {\n setSelectedMaterialId: (state, action) => {\n state.selectedMaterialId = action.payload\n },\n setShowAddMaterialForm: (state, action) => {\n state.showAddMaterialForm = action.payload\n },\n },\n})\n\nexport const selectSelectedMaterialId = (state) => state.materialsSlice.selectedMaterialId\nexport const selectShowAddMaterialForm = (state) => state.materialsSlice.showAddMaterialForm\n\nexport const { setSelectedMaterialId, setShowAddMaterialForm } = materialsSlice.actions\n\nexport default materialsSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedMiscItemsIds: [],\n expandedMiscItemsIds: [],\n miscFilterOptions: 'showAllMiscItems',\n}\n\nexport const miscItemsSlice = createSlice({\n name: 'miscItemsSlice',\n initialState,\n reducers: {\n setSelectedMiscItemsIds: (state, action) => {\n state.selectedMiscItemsIds = action.payload\n },\n addSelectedMiscItem: (state, action) => {\n state.selectedMiscItemsIds.push(action.payload)\n },\n removeSelectedMiscItem: (state, action) => {\n state.selectedMiscItemsIds = state.selectedMiscItemsIds.filter((item) => item !== action.payload)\n },\n clearSelectedMiscItems: (state) => {\n state.selectedMiscItemsIds = []\n },\n setExpandedMiscItemsIds: (state, action) => {\n state.expandedMiscItemsIds = action.payload\n },\n addExpandedMiscItem: (state, action) => {\n state.expandedMiscItemsIds.push(action.payload)\n },\n removeExpandedMiscItem: (state, action) => {\n state.expandedMiscItemsIds = state.expandedMiscItemsIds.filter((item) => item !== action.payload)\n },\n clearExpandedMiscItems: (state) => {\n state.expandedMiscItemsIds = []\n },\n setMiscFilterOptions: (state, action) => {\n state.miscFilterOptions = action.payload\n },\n },\n})\n\nexport const selectSelectedMiscItems = (state) => state.miscItemsSlice.selectedMiscItemsIds\nexport const selectExpandedMiscItems = (state) => state.miscItemsSlice.expandedMiscItemsIds\nexport const selectMiscFilterOptions = (state) => state.miscItemsSlice.miscFilterOptions\n\nexport const {\n addExpandedMiscItem,\n addSelectedMiscItem,\n clearExpandedMiscItems,\n clearSelectedMiscItems,\n removeExpandedMiscItem,\n removeSelectedMiscItem,\n setExpandedMiscItemsIds,\n setSelectedMiscItemsIds,\n} = miscItemsSlice.actions\n\nexport default miscItemsSlice.reducer\n","/* eslint-disable no-unused-vars */\nimport { api } from './api'\n\nexport const organisationApi = api\n .enhanceEndpoints({\n addTagTypes: ['Organisation', 'SharedOrganisation'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getOrganisation: builder.query({\n query: ({ organisationId }) => `/organisations/${organisationId}`,\n providesTags: ['Organisation'],\n }),\n\n getSharedOrganisation: builder.query({\n query: ({ organisationId }) => `/organisations/shared/${organisationId}`,\n providesTags: ['SharedOrganisation'],\n }),\n\n updateOrganisation: builder.mutation({\n query: ({ organisationDto }) => {\n const { $type, ...requestModel } = organisationDto\n return {\n url: `/organisations/${organisationDto.organisationId}`,\n method: 'POST',\n body: requestModel,\n }\n },\n invalidatesTags: ['Organisation'],\n }),\n\n getStoreSettings: builder.query({\n query: ({ organisationId }) => `/organisations/${organisationId}/store`,\n }),\n\n transferOwnership: builder.mutation({\n query: ({ organisationId, userId }) => ({\n url: `/organisations/${organisationId}/transfer-ownership`,\n method: 'POST',\n body: { userId },\n }),\n invalidatesTags: ['Organisation'],\n }),\n\n uploadOrganisationLogo: builder.mutation({\n query: ({ logo, organisationId }) => {\n const formData = new FormData()\n formData.append('file', logo)\n return {\n url: `/organisations/${organisationId}/logo`,\n method: 'POST',\n body: formData,\n }\n },\n invalidatesTags: ['Organisation'],\n }),\n\n changeDrawingUnits: builder.mutation({\n query: ({ drawingUnits, organisationId }) => ({\n url: `/organisations/${organisationId}/change-drawing-units?drawingUnits=${drawingUnits}`,\n method: 'POST',\n }),\n invalidatesTags: ['Organisation'],\n }),\n\n toggleLegacyMaterials: builder.mutation({\n query: ({ organisationId }) => ({\n url: `/organisations/${organisationId}/toggle-legacy-materials`,\n method: 'POST',\n }),\n invalidatesTags: ['Organisation'],\n }),\n\n downloadOrganisationAsJson: builder.mutation({\n query: ({ organisationId }) => ({\n url: `/organisations/${organisationId}?isFullDownload=true`,\n method: 'GET',\n }),\n transformResponse: (response, _meta, arg) => {\n const blob = new Blob([JSON.stringify(response, null, 2)])\n const url = window.URL.createObjectURL(blob)\n const link = document.createElement('a')\n link.href = url\n link.setAttribute('download', `${arg.organisationId}_${response.name}.json`)\n document.body.appendChild(link)\n link.click()\n document.body.removeChild(link)\n window.URL.revokeObjectURL(url)\n return response\n },\n }),\n\n exportOrganisationAsJson: builder.mutation({\n query: ({ organisationId }) => ({\n url: `/organisations/${organisationId}/export`,\n method: 'GET',\n }),\n transformResponse: (response, _meta, arg) => {\n const blob = new Blob([JSON.stringify(response, null, 2)])\n const url = window.URL.createObjectURL(blob)\n const link = document.createElement('a')\n link.href = url\n link.setAttribute('download', `${arg.organisationId}_${response.name}.json`)\n document.body.appendChild(link)\n link.click()\n document.body.removeChild(link)\n window.URL.revokeObjectURL(url)\n return response\n },\n }),\n\n importOrganisation: builder.mutation({\n query: ({ file, organisationId }) => {\n const formData = new FormData()\n formData.append('file', file)\n return {\n url: `/organisations/${organisationId}/import`,\n method: 'POST',\n body: formData,\n }\n },\n invalidatesTags: ['Organisation'],\n }),\n\n reinitializeOrganisation: builder.mutation({\n query: ({ cuttingTechnologyPower, organisationId }) => ({\n url: `/organisations/${organisationId}/reinitialise`,\n method: 'POST',\n body: { cuttingTechnologyPowerOverride: cuttingTechnologyPower },\n }),\n invalidatesTags: ['Organisation'],\n }),\n\n uploadStoreLogo: builder.mutation({\n query: ({ file, organisationId }) => {\n const formData = new FormData()\n formData.append('file', file)\n return {\n url: `/organisations/${organisationId}/store/logo`,\n method: 'POST',\n prepareHeaders: (headers) => {\n headers.set('Content-Type', 'multipart/form-data')\n return headers\n },\n body: formData,\n }\n },\n invalidatesTags: ['Organisation'],\n }),\n\n uploadStoreFavicon: builder.mutation({\n query: ({ file, organisationId }) => {\n const formData = new FormData()\n formData.append('file', file)\n return {\n url: `/organisations/${organisationId}/store/favicon`,\n method: 'POST',\n prepareHeaders: (headers) => {\n headers.set('Content-Type', 'multipart/form-data')\n return headers\n },\n body: formData,\n }\n },\n invalidatesTags: ['Organisation'],\n }),\n\n validateSubdomain: builder.mutation({\n query: ({ organisationId, subDomain }) => ({\n url: `/organisations/${organisationId}/subdomain-validation`,\n method: 'GET',\n params: { subDomain },\n }),\n }),\n }),\n })\n\nexport const selectStoreSettings = (state, query) => {\n return organisationApi.endpoints.getStoreSettings.select(query)(state)?.data ?? {}\n}\n\nexport const {\n useChangeDrawingUnitsMutation,\n useDownloadOrganisationAsJsonMutation,\n useExportOrganisationAsJsonMutation,\n useGetOrganisationQuery,\n useGetSharedOrganisationQuery,\n useGetStoreSettingsQuery,\n useImportOrganisationMutation,\n useLazyGetOrganisationQuery,\n useLazyGetSharedOrganisationQuery,\n useReinitializeOrganisationMutation,\n useToggleLegacyMaterialsMutation,\n useTransferOwnershipMutation,\n useUpdateOrganisationMutation,\n useUploadOrganisationLogoMutation,\n useUploadStoreFaviconMutation,\n useUploadStoreLogoMutation,\n useValidateSubdomainMutation,\n} = organisationApi\n","import { createSlice, isAnyOf } from '@reduxjs/toolkit'\n\nimport { getCurrencySymbol } from '@/common/utils'\n\nimport { organisationApi } from '../services/organisation'\n\nconst getPlanMetadataTypeReducer = (type) => {\n switch (type) {\n case 'number':\n return (acc, curr) => Math.max(parseInt(acc) || 0, parseInt(curr) || 0)\n default:\n return (acc, curr) => acc || curr === 'true'\n }\n}\n\nconst PLAN_METADATA_TYPES = {\n '3d': 'boolean',\n customer_pricing: 'boolean',\n customer_supplied: 'boolean',\n fixed_price_parts: 'boolean',\n is_enterprise_plan: 'boolean',\n material_consumption: 'boolean',\n materials_revamp: 'boolean',\n max_users: 'number',\n miscellaneous_items: 'boolean',\n part_library: 'boolean',\n pdf: 'boolean',\n plan: 'free',\n quote_history: 'boolean',\n sells_drawing_credits: 'boolean',\n tube: 'boolean',\n webstore: 'boolean',\n showViewer3D: 'boolean',\n showViewerTube: 'boolean',\n shipping: 'boolean',\n}\n\nconst initialState = {\n organisation: {},\n organisationId: undefined,\n lastPaymentStatus: undefined,\n paidFeatures: undefined,\n useImperialUnits: undefined,\n currencySymbol: undefined,\n currencyCode: undefined,\n isOnTrial: false,\n reduceTrialAccess: false,\n planMetadata: undefined,\n resolvedPlanMetadata: undefined,\n isOnFreePlan: false,\n showAccelerationSettings: false,\n defaultTaxRateId: undefined,\n hasActiveSubscription: false,\n}\n\nexport const organisationSlice = createSlice({\n name: 'organisationSlice',\n initialState,\n reducers: {\n setOrganisation: (state, { payload }) => {\n const isOnTrialValue = (payload?.trial && !payload?.trial.hasExpired) ?? false\n const isOnFreePlan = Boolean(payload?.freePlan)\n const reduceTrialAccessValue =\n ((payload?.trial &&\n !payload?.trial?.hasExpired &&\n !payload.hasActiveSubscription &&\n !import.meta.env.VITE_RESELLER) ||\n (payload?.freePlan && !payload.hasActiveSubscription && !import.meta.env.VITE_RESELLER)) ??\n false\n\n const metadata = payload?.subscriptions\n ?.map((s) => s.subscriptionItems)\n ?.reduce((array, current) => {\n current.forEach((si) => array.push(si.price))\n return array\n }, [])\n ?.reduce((map, curr) => {\n if (curr && curr.metadata) {\n const { $type, ...metadataDto } = curr.metadata\n map[curr.priceId] = metadataDto\n }\n return map\n }, {})\n\n const resolvedMetadata = metadata\n ? Object.values(metadata).reduce((acc, currPlanMetadata) => {\n return Object.keys(currPlanMetadata).reduce((innerAcc, metadataKey) => {\n innerAcc[metadataKey] = getPlanMetadataTypeReducer(PLAN_METADATA_TYPES[metadataKey])(\n innerAcc[metadataKey],\n currPlanMetadata[metadataKey]\n )\n return innerAcc\n }, acc)\n }, {})\n : {}\n const paidFeatures = {\n isFreePlan: resolvedMetadata?.plan?.toString().toLowerCase() === 'free',\n hasMaterialConsumptionModeOptions:\n isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.material_consumption),\n hasCustomerPricing:\n isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.customer_pricing),\n hasAdvancedMaterials:\n isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.materials_revamp),\n hasCustomerSuppliedMaterial:\n isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.customer_supplied),\n hasCreditConsumption: true,\n hasUnfoldTube: isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.tube),\n hasPdfUpload: isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.pdf),\n hasPartLibrary: isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.part_library),\n hasWebStore: isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.webstore),\n has3d:\n isOnTrialValue ||\n (payload?.hasActiveSubscription && resolvedMetadata ? resolvedMetadata['3d'] : false),\n hasMiscellaneousItems:\n isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.miscellaneous_items),\n hasFixedPriceParts:\n isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.fixed_price_parts),\n hasShowViewer3D: payload?.hasActiveSubscription && resolvedMetadata?.ShowViewer3D,\n hasShowViewerTube: payload?.hasActiveSubscription && resolvedMetadata?.ShowViewerTube,\n hasShipping: isOnTrialValue || (payload?.hasActiveSubscription && resolvedMetadata?.shipping),\n }\n\n const useImperialUnits =\n payload?.defaultDrawingUnits && payload?.defaultDrawingUnits?.toLowerCase() === 'imperial'\n\n const currencySymbol = getCurrencySymbol(payload?.currencyCode, payload?.locale)\n\n state.resolvedPlanMetadata = resolvedMetadata\n state.planMetadata = metadata\n state.paidFeatures = paidFeatures\n state.useImperialUnits = useImperialUnits\n state.currencySymbol = currencySymbol\n state.currencyCode = payload.currencyCode\n state.reduceTrialAccess = reduceTrialAccessValue\n state.isOnTrial = isOnTrialValue\n state.isOnFreePlan = isOnFreePlan\n state.organisationId = payload.organisationId\n state.organisation = payload\n state.hasActiveSubscription = payload.hasActiveSubscription\n state.showAccelerationSettings = payload.showAccelerationSettings\n },\n setLastPaymentStatus: (state, action) => {\n state.lastPaymentStatus = action.payload\n },\n setPaidFeatures: (state, action) => {\n state.paidFeatures = action.payload\n },\n setUseImperialUnits: (state, action) => {\n state.useImperialUnits = action.payload\n },\n setCurrencySymbol: (state, action) => {\n state.currencySymbol = action.payload\n },\n setcurrencyCode: (state, action) => {\n state.currencyCode = action.payload\n },\n setIsOnTrial: (state, action) => {\n state.isOnTrial = action.payload\n },\n setDefaultTaxRateId: (state, action) => {\n state.defaultTaxRateId = action.payload\n },\n },\n extraReducers: (builder) => {\n builder.addMatcher(isAnyOf(organisationApi.endpoints.getOrganisation.matchFulfilled), (state, action) => {\n organisationSlice.caseReducers.setOrganisation(state, action)\n })\n },\n})\n\nexport const selectOrganisation = (state) => state.organisationSlice.organisation\nexport const selectOrganisationId = (state) => state.organisationSlice.organisationId\nexport const selectPlanMetadata = (state) => state.organisationSlice.planMetadata\nexport const selectLastPaymentStatus = (state) => state.organisationSlice.lastPaymentStatus\nexport const selectUseImperialUnits = (state) => state.organisationSlice.useImperialUnits\nexport const selectCurrencySymbol = (state) => state.organisationSlice.currencySymbol\nexport const selectCurrencyCode = (state) => state.organisationSlice.currencyCode\nexport const selectPaidFeatures = (state) => state.organisationSlice.paidFeatures\nexport const selectIsOnTrial = (state) => state.organisationSlice.isOnTrial\nexport const selectReduceTrialAccess = (state) => state.organisationSlice.reduceTrialAccess\nexport const selectResolvedPlanMetadata = (state) => state.organisationSlice.resolvedPlanMetadata\nexport const selectIsOnFreePlan = (state) => state.organisationSlice.isOnFreePlan\nexport const selectShowAccelerationSettings = (state) => state.organisationSlice.showAccelerationSettings\nexport const selectDefaultTaxRateId = (state) => state.organisationSlice.defaultTaxRateId\nexport const selectMaxUsers = (state) => state.organisationSlice.resolvedPlanMetadata?.max_users ?? 0\nexport const selectHasActiveSubscription = (state) => state.organisationSlice.hasActiveSubscription\n\nexport const {\n setCurrencySymbol,\n setDefaultTaxRateId,\n setIsOnTrial,\n setLastPaymentStatus,\n setOrganisation,\n setPaidFeatures,\n setUseImperialUnits,\n} = organisationSlice.actions\n\nexport default organisationSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedItemsIds: [],\n expandedItemsIds: [],\n collapsedItemsIds: [],\n filteredItemsIds: [],\n updateQueue: [],\n filterOption: 'showAllItems',\n isCalculatingQuote: false,\n selectedQuoteItemDrawingId: null,\n openDrawingDoctorModal: false,\n openPartQuoteHistoryModal: false,\n quoteStatus: null,\n quotePaymentStatus: null,\n selectedPartFromLibrary: null,\n}\n\nexport const quoteItemsSlice = createSlice({\n name: 'quoteItemsSlice',\n initialState,\n reducers: {\n setSelectedItemsIds: (state, action) => {\n state.selectedItemsIds = action.payload\n },\n addSelectedItem: (state, action) => {\n state.selectedItemsIds.push(action.payload)\n },\n removeSelectedItem: (state, action) => {\n state.selectedItemsIds = state.selectedItemsIds.filter((item) => item !== action.payload)\n },\n clearSelectedItems: (state) => {\n state.selectedItemsIds = []\n },\n setExpandedItemsIds: (state, action) => {\n state.expandedItemsIds = action.payload\n },\n addExpandedItem: (state, action) => {\n state.expandedItemsIds.push(action.payload)\n },\n removeExpandedItem: (state, action) => {\n state.expandedItemsIds = state.expandedItemsIds.filter((item) => item !== action.payload)\n },\n clearExpandedItems: (state) => {\n state.expandedItemsIds = []\n },\n setCollapsedItemsIds: (state, action) => {\n state.collapsedItemsIds = action.payload\n },\n addCollapsedItem: (state, action) => {\n state.collapsedItemsIds.push(action.payload)\n },\n removeCollapsedItem: (state, action) => {\n state.collapsedItemsIds = state.collapsedItemsIds.filter((item) => item !== action.payload)\n },\n clearCollapsedItems: (state) => {\n state.collapsedItemsIds = []\n },\n setFilterOption: (state, action) => {\n state.filterOption = action.payload\n },\n setIsCalculatingQuote: (state, action) => {\n state.isCalculatingQuote = action.payload\n },\n\n setSelectedQuoteItemDrawingId: (state, action) => {\n state.selectedQuoteItemDrawingId = action.payload\n },\n setOpenDrawingDoctorModal: (state, action) => {\n state.openDrawingDoctorModal = action.payload\n },\n setOpenPartQuoteHistoryModal: (state, action) => {\n state.openPartQuoteHistoryModal = action.payload\n },\n setSelectedQuoteStatus: (state, action) => {\n state.quoteStatus = action.payload\n },\n setSelectedQuotePaymentStatus: (state, action) => {\n state.quotePaymentStatus = action.payload\n },\n setSelectedPartFromLibrary: (state, action) => {\n state.selectedPartFromLibrary = action.payload\n },\n setFilteredItemsIds: (state, action) => {\n state.filteredItemsIds = action.payload\n },\n addItemToUpdateQueue: (state, action) => {\n if (!state.updateQueue.includes(action.payload)) {\n state.updateQueue.push(action.payload)\n }\n },\n removeItemFromUpdateQueue: (state, action) => {\n state.updateQueue = state.updateQueue.filter((item) => item !== action.payload)\n },\n clearUpdateQueue: (state) => {\n state.updateQueue = []\n },\n },\n})\n\nexport const selectSelectedItems = (state) => state.quoteItemsSlice.selectedItemsIds\nexport const selectExpandedItems = (state) => state.quoteItemsSlice.expandedItemsIds\nexport const selectCollapsedItems = (state) => state.quoteItemsSlice.collapsedItemsIds\nexport const selectFilterOption = (state) => state.quoteItemsSlice.filterOption\nexport const selectIsCalculatingQuote = (state) => state.quoteItemsSlice.isCalculatingQuote\nexport const selectQuoteItemDrawingId = (state) => state.quoteItemsSlice.selectedQuoteItemDrawingId\nexport const selectOpenDrawingDoctorModal = (state) => state.quoteItemsSlice.openDrawingDoctorModal\nexport const selectOpenPartQuoteHistoryModal = (state) => state.quoteItemsSlice.openPartQuoteHistoryModal\nexport const selectedQuoteStatus = (state) => state.quoteItemsSlice.quoteStatus\nexport const selectQuotePaymentStatus = (state) => state.quoteItemsSlice.quotePaymentStatus\nexport const selectSelectedPartFromLibrary = (state) => state.quoteItemsSlice.selectedPartFromLibrary\nexport const selectFilteredItemsIds = (state) => state.quoteItemsSlice.filteredItemsIds\nexport const selectIsUpdateQueueEmpty = (state) => state.quoteItemsSlice.updateQueue.length === 0\n\nexport const {\n addCollapsedItem,\n addExpandedItem,\n addSelectedItem,\n clearCollapsedItems,\n clearExpandedItems,\n clearSelectedItems,\n removeCollapsedItem,\n removeExpandedItem,\n removeSelectedItem,\n setCollapsedItemsIds,\n setExpandedItemsIds,\n setFilterOption,\n setIsCalculatingQuote,\n setOpenDrawingDoctorModal,\n setOpenPartQuoteHistoryModal,\n setSelectedItemsIds,\n setSelectedPartFromLibrary,\n setSelectedQuoteItemDrawingId,\n setSelectedQuotePaymentStatus,\n setSelectedQuoteStatus,\n setFilteredItemsIds,\n addItemToUpdateQueue,\n removeItemFromUpdateQueue,\n clearUpdateQueue,\n} = quoteItemsSlice.actions\n\nexport default quoteItemsSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedRate: null,\n selectedRates: [],\n}\n\nexport const ratesSlice = createSlice({\n name: 'rateSlice',\n initialState,\n reducers: {\n setSelectedRate: (state, action) => {\n state.selectedRate = action.payload\n },\n\n setSelectedRates: (state, action) => {\n state.selectedRates = action.payload\n },\n\n addToSelectedRates: (state, action) => {\n state.selectedRates.push(action.payload)\n },\n\n removeFromSelectedRates: (state, action) => {\n state.selectedRates = state.selectedRates.filter((rate) => rate.rateId !== action.payload.rateId)\n },\n\n clearSelectedRates: (state) => {\n state.selectedRates = []\n },\n },\n})\n\nexport const selectSelectedRate = (state) => state.ratesSlice.selectedRate\nexport const selectSelectedRates = (state) => state.ratesSlice.selectedRates\n\nexport const { addToSelectedRates, clearSelectedRates, removeFromSelectedRates, setSelectedRate, setSelectedRates } =\n ratesSlice.actions\n\nexport default ratesSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedRateTableId: null,\n showAddRateTableForm: false,\n}\n\nexport const rateTablesSlice = createSlice({\n name: 'rateTablesSlice',\n initialState,\n reducers: {\n setSelectedRateTableId: (state, action) => {\n state.selectedRateTableId = action.payload\n },\n setShowAddRateTableForm: (state, action) => {\n state.showAddRateTableForm = action.payload\n },\n },\n})\n\nexport const selectSelectedRateTableId = (state) => state.rateTablesSlice.selectedRateTableId\nexport const selectShowAddRateTableForm = (state) => state.rateTablesSlice.showAddRateTableForm\n\nexport const { setSelectedRateTableId, setShowAddRateTableForm } = rateTablesSlice.actions\n\nexport default rateTablesSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedSheet: null,\n selectedSheets: [],\n}\n\nexport const sheetsSlice = createSlice({\n name: 'sheetsSlice',\n initialState,\n reducers: {\n setSelectedSheet: (state, action) => {\n state.selectedSheet = action.payload\n },\n\n setSelectedSheets: (state, action) => {\n state.selectedSheets = action.payload\n },\n\n addToSelectedSheets: (state, action) => {\n state.selectedSheets.push(action.payload)\n },\n\n removeFromSelectedSheets: (state, action) => {\n state.selectedSheets = state.selectedSheets.filter((sheet) => sheet.sheetId !== action.payload.sheetId)\n },\n\n clearSelectedSheets: (state) => {\n state.selectedSheets = []\n },\n },\n})\n\n//export const selectSheets = (state) => state.sheetsSlice.sheets\nexport const selectSelectedSheet = (state) => state.sheetsSlice.selectedSheet\nexport const selectSelectedSheets = (state) => state.sheetsSlice.selectedSheets\n\nexport const {\n addToSelectedSheets,\n clearSelectedSheets,\n removeFromSelectedSheets,\n setSelectedSheet,\n setSelectedSheets,\n} = sheetsSlice.actions\n\nexport default sheetsSlice.reducer\n","import { webStoreApi } from './webStoreApi'\n\nexport const webStoreAuthApi = webStoreApi.injectEndpoints({\n endpoints: (builder) => ({\n login: builder.mutation({\n query: ({ emailAddress, organisationId }) => ({\n url: `/${organisationId}/temp-login`,\n method: 'POST',\n body: { emailAddress },\n }),\n }),\n validateCode: builder.mutation({\n query: ({ code, emailAddress, organisationId }) => ({\n url: `/${organisationId}/temp-login/verify`,\n method: 'POST',\n body: { emailAddress, code },\n }),\n }),\n }),\n})\n\nexport const { useLoginMutation, useValidateCodeMutation } = webStoreAuthApi\n","const getCookie = (name) => {\n const matches = document.cookie.match(\n new RegExp('(?:^|; )' + name.replace(/([.$?*|{}()[\\]\\\\/+^])/g, '\\\\$1') + '=([^;]*)')\n )\n return matches ? decodeURIComponent(matches[1]) : undefined\n}\n\nconst setCookie = (name, value, options = {}) => {\n options = {\n path: '/',\n // add other defaults here if necessary\n ...options,\n }\n\n if (options.expires instanceof Date) {\n options.expires = options.expires.toUTCString()\n }\n\n let updatedCookie = encodeURIComponent(name) + '=' + encodeURIComponent(value)\n\n for (const optionKey in options) {\n updatedCookie += '; ' + optionKey\n const optionValue = options[optionKey]\n if (optionValue !== true) {\n updatedCookie += '=' + optionValue\n }\n }\n\n document.cookie = updatedCookie\n}\n\nconst deleteCookie = (name) => {\n setCookie(name, '', {\n 'max-age': -1,\n })\n}\n\nexport { deleteCookie, getCookie, setCookie }\n","import { webStoreApi } from './webStoreApi'\n\nexport const webStoreCustomerApi = webStoreApi\n .enhanceEndpoints({\n addTagTypes: ['WebStoreCustomer', 'WebStoreContact'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getCustomer: builder.query({\n query: ({ customerId, organisationId }) => `/${organisationId}/customer/${customerId}`,\n providesTags: ['WebStoreCustomer'],\n }),\n\n getContact: builder.query({\n query: ({ contactId, customerId, organisationId }) =>\n `/${organisationId}/customer/${customerId}/contacts/${contactId}`,\n providesTags: (result, error, arg) => [{ type: 'WebStoreContact', id: arg.contactId }],\n }),\n\n getContacts: builder.query({\n query: ({ customerId, organisationId }) => `/${organisationId}/customer/${customerId}/contacts`,\n providesTags: (result, _error, _arg) => [\n { type: 'Contact', id: 'LIST' },\n ...(result?.map(({ id }) => ({ type: 'WebStoreContact', id })) ?? []),\n ],\n }),\n\n createCustomer: builder.mutation({\n query: ({ customer, organisationId }) => ({\n url: `/${organisationId}/customer`,\n method: 'POST',\n body: customer,\n headers: {\n 'Content-Type': 'application/json',\n },\n }),\n invalidatesTags: ['WebStoreCustomer'],\n }),\n\n updateCustomer: builder.mutation({\n query: ({ customer, customerId, organisationId }) => ({\n url: `/${organisationId}/customer/${customerId}`,\n method: 'PUT',\n body: customer,\n }),\n invalidatesTags: ['WebStoreCustomer'],\n }),\n\n createContact: builder.mutation({\n query: ({ contact, customerId, organisationId }) => ({\n url: `/${organisationId}/customer/${customerId}/contacts`,\n method: 'POST',\n body: contact,\n headers: {\n 'Content-Type': 'application/json',\n },\n }),\n invalidatesTags: ['WebStoreContact'],\n }),\n\n updateContact: builder.mutation({\n query: ({ contact, contactId, customerId, organisationId }) => ({\n url: `/${organisationId}/customer/${customerId}/contacts/${contactId}`,\n method: 'PUT',\n body: contact,\n }),\n invalidatesTags: (result, error, arg) => [{ type: 'WebStoreContact', id: arg.contactId }],\n }),\n }),\n })\n\nexport const {\n useCreateContactMutation,\n useCreateCustomerMutation,\n useGetContactQuery,\n useGetContactsQuery,\n useGetCustomerQuery,\n useLazyGetContactQuery,\n useLazyGetCustomerQuery,\n useUpdateContactMutation,\n useUpdateCustomerMutation,\n} = webStoreCustomerApi\n","import { createSlice } from '@reduxjs/toolkit'\n\nimport { webStoreAuthApi } from '@/app/services/web-store/webStoreAuth'\n\nimport { deleteCookie, getCookie, setCookie } from '../../../features/web-store/helpers/auth/cookie'\nimport { webStoreCustomerApi } from '../../services/web-store/webStoreCustomer'\n\nconst isContactLoggedInCookie = getCookie('isContactLoggedIn') ? JSON.parse(getCookie('isContactLoggedIn')) : false\nconst isGuestLoggedInCookie = getCookie('isGuestLoggedIn') ? JSON.parse(getCookie('isGuestLoggedIn')) : false\nconst currentContactCookie = getCookie('currentContact') ? JSON.parse(getCookie('currentContact')) : null\nconst currentCustomerCookie = getCookie('currentCustomer') ? JSON.parse(getCookie('currentCustomer')) : null\nconst guestUserDataCookie = getCookie('guestUserData') ? JSON.parse(getCookie('guestUserData')) : null\n\nconst initialState = {\n isContactLoggedIn: isContactLoggedInCookie,\n isGuestLoggedIn: isGuestLoggedInCookie,\n contact: currentContactCookie,\n customer: currentCustomerCookie,\n guest: guestUserDataCookie,\n token: null,\n openLoginDialog: false,\n}\n\nconst webStoreAuthSlice = createSlice({\n name: 'webStoreAuth',\n initialState,\n reducers: {\n unsetCurrentUser: (state) => {\n state.contact = null\n state.customer = null\n state.guest = null\n state.isContactLoggedIn = false\n state.isGuestLoggedIn = false\n deleteCookie('isContactLoggedIn')\n deleteCookie('isGuestLoggedIn')\n deleteCookie('currentContact')\n deleteCookie('currentCustomer')\n deleteCookie('guestUserData')\n },\n setGuestUserLoggin: (state) => {\n state.isGuestLoggedIn = true\n setCookie('isGuestLoggedIn', 'true', {\n samesite: 'strict',\n 'max-age': 3600,\n })\n },\n setGuestUserData: (state, action) => {\n state.guest = action.payload\n setCookie('guestUserData', JSON.stringify(action.payload), {\n samesite: 'strict',\n 'max-age': 3600,\n })\n },\n setOpenLoginDialog: (state, action) => {\n state.openLoginDialog = action.payload\n },\n },\n extraReducers: (builder) => {\n builder\n .addMatcher(webStoreAuthApi.endpoints.validateCode.matchFulfilled, (state, { payload }) => {\n state.contact = payload\n state.isContactLoggedIn = true\n state.isGuestLoggedIn = false\n state.guest = null\n setCookie('isContactLoggedIn', 'true', {\n samesite: 'strict',\n 'max-age': 3600,\n })\n setCookie('currentContact', JSON.stringify(payload), {\n samesite: 'strict',\n 'max-age': 3600,\n })\n deleteCookie('isGuestLoggedIn')\n deleteCookie('guestUserData')\n })\n .addMatcher(webStoreCustomerApi.endpoints.getCustomer.matchFulfilled, (state, { payload }) => {\n state.customer = payload\n const cookieData = structuredClone(payload)\n delete cookieData.addresses\n\n setCookie('currentCustomer', JSON.stringify(cookieData), {\n samesite: 'strict',\n 'max-age': 3600,\n })\n })\n },\n})\n\nexport const selectCurrentContact = (state) => state.webStoreAuth.contact\nexport const selectCurrentCustomer = (state) => state.webStoreAuth.customer\nexport const selectGuestUser = (state) => state.webStoreAuth.guest\nexport const selectIsContactLoggedIn = (state) => state.webStoreAuth.isContactLoggedIn\nexport const selectIsGuestLoggedIn = (state) => state.webStoreAuth.isGuestLoggedIn\nexport const selectOpenLoginDialog = (state) => state.webStoreAuth.openLoginDialog\n\nexport const { setGuestUserData, setGuestUserLoggin, setOpenLoginDialog, unsetCurrentUser } = webStoreAuthSlice.actions\n\nexport default webStoreAuthSlice.reducer\n","import { createSlice } from '@reduxjs/toolkit'\n\nconst initialState = {\n selectedItemsIds: [],\n expandedItemsIds: [],\n}\n\nexport const webStoreQuoteItemsSlice = createSlice({\n name: 'webStoreQuoteItems',\n initialState,\n reducers: {\n setSelectedItemsIds: (state, action) => {\n state.selectedItemsIds = action.payload\n },\n addSelectedItem: (state, action) => {\n state.selectedItemsIds.push(action.payload)\n },\n removeSelectedItem: (state, action) => {\n state.selectedItemsIds = state.selectedItemsIds.filter((item) => item !== action.payload)\n },\n clearSelectedItems: (state) => {\n state.selectedItemsIds = []\n },\n setExpandedItemsIds: (state, action) => {\n state.expandedItemsIds = action.payload\n },\n addExpandedItem: (state, action) => {\n state.expandedItemsIds.push(action.payload)\n },\n removeExpandedItem: (state, action) => {\n state.expandedItemsIds = state.expandedItemsIds.filter((item) => item !== action.payload)\n },\n clearExpandedItems: (state) => {\n state.expandedItemsIds = []\n },\n },\n})\n\nexport const selectSelectedItems = (state) => state.webStoreQuoteItems.selectedItemsIds\nexport const selectExpandedItems = (state) => state.webStoreQuoteItems.expandedItemsIds\n\nexport const {\n addExpandedItem,\n addSelectedItem,\n clearExpandedItems,\n clearSelectedItems,\n removeExpandedItem,\n removeSelectedItem,\n setExpandedItemsIds,\n setSelectedItemsIds,\n} = webStoreQuoteItemsSlice.actions\n\nexport default webStoreQuoteItemsSlice.reducer\n","import { configureStore } from '@reduxjs/toolkit'\nimport { setupListeners } from '@reduxjs/toolkit/query'\nimport { FLUSH, PAUSE, PERSIST, persistReducer, persistStore, PURGE, REGISTER, REHYDRATE } from 'redux-persist'\nimport storage from 'redux-persist/lib/storage'\n\nimport { api } from './services/api'\nimport { quickPartsApi } from './services/quickPartsApi'\nimport { webStoreApi } from './services/web-store/webStoreApi'\nimport appReducer from './slices/appSlice'\nimport contactsReducer from './slices/contactsSlice'\nimport customersReducer from './slices/customersSlice'\nimport cuttingTechnologiesReducer from './slices/cuttingTechnologiesSlice'\nimport materialsReducer from './slices/materialsSlice'\nimport miscItemsReducer from './slices/miscItemsSlice'\nimport organisationReducer from './slices/organisationSlice'\nimport quoteItemsReducer from './slices/quoteItemsSlice'\nimport ratesReducer from './slices/ratesSlice'\nimport rateTablesReducer from './slices/rateTablesSlice'\nimport sheetsReducer from './slices/sheetsSlice'\nimport webStoreAuthSlice from './slices/web-store/webStoreAuthSlice'\nimport webStoreQuoteItemsReducer from './slices/web-store/webStoreQuoteItemsSlice'\n\nconst persistConfig = {\n key: 'qt',\n version: 3,\n storage,\n whitelist: ['displayQuotePricing'],\n}\n\nconst persistedAppReducer = persistReducer(persistConfig, appReducer)\n\nexport const store = configureStore({\n reducer: {\n [api.reducerPath]: api.reducer,\n [quickPartsApi.reducerPath]: quickPartsApi.reducer,\n [webStoreApi.reducerPath]: webStoreApi.reducer,\n appSlice: persistedAppReducer,\n contactsSlice: contactsReducer,\n customersSlice: customersReducer,\n cuttingTechnologiesSlice: cuttingTechnologiesReducer,\n materialsSlice: materialsReducer,\n miscItemsSlice: miscItemsReducer,\n organisationSlice: organisationReducer,\n quoteItemsSlice: quoteItemsReducer,\n ratesSlice: ratesReducer,\n rateTablesSlice: rateTablesReducer,\n sheetsSlice: sheetsReducer,\n webStoreAuth: webStoreAuthSlice,\n webStoreQuoteItems: webStoreQuoteItemsReducer,\n },\n middleware: (getDefaultMiddleware) =>\n getDefaultMiddleware({\n serializableCheck: {\n ignoredActions: [FLUSH, REHYDRATE, PAUSE, PERSIST, PURGE, REGISTER],\n },\n })\n .concat(api.middleware)\n .concat(quickPartsApi.middleware)\n .concat(webStoreApi.middleware),\n devTools: process.env.NODE_ENV === 'development',\n})\n\nsetupListeners(store.dispatch)\n\nexport const persistor = persistStore(store)\n","import { useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { FormControl, InputLabel, MenuItem, Select } from '@mui/material'\nimport allCountries from 'country-region-data'\nimport PropTypes from 'prop-types'\n\nconst CountrySelectInput = ({ disabled = false, onChange, required = false, value }) => {\n const { t } = useTranslation()\n const [selectedValue, setSelectedValue] = useState('')\n\n useEffect(() => {\n setSelectedValue(value)\n }, [value])\n\n const handleChange = (event) => {\n const newCountryCode = event.target.value\n const newCountryIndex = allCountries?.findIndex((country) => country.countryShortCode === newCountryCode)\n const newSelectedCountry = allCountries[newCountryIndex].countryShortCode\n\n setSelectedValue(newSelectedCountry)\n\n if (typeof onChange === 'function') {\n onChange(newSelectedCountry)\n }\n }\n\n return (\n \n \n {t('Select country')}\n \n \n {allCountries.map((country) => {\n return (\n \n {country.countryName}\n \n )\n })}\n \n \n )\n}\n\nCountrySelectInput.propTypes = {\n disabled: PropTypes.bool,\n required: PropTypes.bool,\n value: PropTypes.string,\n onChange: PropTypes.func,\n}\n\nexport default CountrySelectInput\n","import { useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport Script from 'react-load-script'\nimport { TextField } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nlet autocomplete\n\nconst LocationSearchInput = ({ autoFocus, fullWidth, label, onChange, placeholder, required, value }) => {\n const { t } = useTranslation()\n\n const [query, setQuery] = useState('')\n\n const handleGeolocation = () => {\n if (navigator.geolocation) {\n navigator.geolocation.getCurrentPosition((position) => {\n const geolocation = {\n lat: position.coords.latitude,\n lng: position.coords.longitude,\n }\n const circle = new window.google.maps.Circle({\n center: geolocation,\n radius: position.coords.accuracy,\n })\n autocomplete.setBounds(circle.getBounds())\n })\n }\n }\n\n const handleScriptLoad = () => {\n // Declare Options For Autocomplete\n const options = {\n types: ['geocode', 'establishment'],\n }\n\n // Initialize Google Autocomplete\n autocomplete = new window.google.maps.places.Autocomplete(document.getElementById('autocomplete'), options)\n\n // Avoid paying for data that you don't need by restricting the set of\n // place fields that are returned to just the address components and formatted\n // address.\n autocomplete.setFields(['address_components', 'formatted_address'])\n\n // Fire Event when a suggested name is selected\n autocomplete.addListener('place_changed', handlePlaceSelect)\n\n handleGeolocation()\n }\n\n const getValueFromAddressComponent = (address, typeName, attributeName) => {\n attributeName = attributeName ?? 'long_name'\n\n for (let i = 0; i < address.length; i++) {\n const component = address[i]\n const typesIndex = component.types.findIndex((t) => t === typeName)\n\n if (typesIndex > -1) {\n return component[attributeName] ?? ''\n }\n }\n\n return ''\n }\n\n const getUnitAddressFromAddress = (address) => {\n return getValueFromAddressComponent(address, 'subpremise', 'short_name')\n }\n\n const getStreetAddressFromAddress = (address) => {\n let streetAddress = ''\n\n streetAddress += getUnitAddressFromAddress(address)\n\n streetAddress += streetAddress ? '/' : ''\n\n streetAddress += getValueFromAddressComponent(address, 'street_number', 'long_name')\n\n streetAddress += streetAddress ? ' ' : ''\n\n streetAddress += getValueFromAddressComponent(address, 'route', 'long_name')\n\n return streetAddress\n }\n\n const getCityFromAddress = (address) => {\n return getValueFromAddressComponent(address, 'locality', 'short_name')\n }\n\n const getCountryFromAddress = (address) => {\n return getValueFromAddressComponent(address, 'country', 'short_name')\n }\n\n const getPostcodeFromAddress = (address) => {\n return getValueFromAddressComponent(address, 'postal_code', 'long_name')\n }\n\n const getStateFromAddress = (address) => {\n return getValueFromAddressComponent(address, 'administrative_area_level_1', 'short_name')\n }\n\n const handlePlaceSelect = () => {\n const addressObject = autocomplete.getPlace()\n const address = addressObject.address_components\n setQuery(addressObject.formatted_address)\n\n if (address && typeof onChange === 'function') {\n const processedAddress = {\n line1: getStreetAddressFromAddress(address),\n line2: '',\n city: getCityFromAddress(address),\n country: getCountryFromAddress(address),\n postcode: getPostcodeFromAddress(address),\n state: getStateFromAddress(address),\n }\n\n onChange(processedAddress)\n }\n }\n\n const handleInputFocus = (event) => {\n event.target.select()\n }\n\n const handleInputBlur = () => {\n const addressObject = autocomplete ? autocomplete.getPlace() : null\n if (addressObject) {\n setQuery(addressObject.formatted_address)\n } else {\n setQuery(value)\n }\n }\n\n useEffect(() => {\n setQuery(value)\n }, [value])\n\n return (\n <>\n \n setQuery(event.target.value)}\n onClick={handleGeolocation}\n onFocus={handleInputFocus}\n />\n \n )\n}\n\nLocationSearchInput.propTypes = {\n autoFocus: PropTypes.bool,\n fullWidth: PropTypes.bool,\n label: PropTypes.string,\n placeholder: PropTypes.string,\n required: PropTypes.bool,\n value: PropTypes.string,\n onChange: PropTypes.func,\n}\n\nexport default LocationSearchInput\n","import { useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { FormControl, InputLabel, MenuItem, Select } from '@mui/material'\nimport allCountries from 'country-region-data'\nimport PropTypes from 'prop-types'\n\nconst RegionSelectInput = ({ country, disabled = false, onChange, required = false, value = '' }) => {\n const { t } = useTranslation()\n\n const [selectedValue, setSelectedValue] = useState('')\n const [regions, setRegions] = useState([])\n\n useEffect(() => {\n setSelectedValue(value)\n }, [value])\n\n useEffect(() => {\n if (!country) {\n setRegions([])\n } else {\n const countryIndex = allCountries.findIndex((c) => c.countryShortCode === country)\n if (countryIndex > -1) {\n const selectedCountry = allCountries[countryIndex]\n\n setRegions(selectedCountry.regions ?? [])\n } else {\n setRegions([])\n }\n }\n }, [country])\n\n const handleChange = (event) => {\n const newRegionCode = event.target.value\n const newRegionIndex = regions.findIndex((region) => region.shortCode === newRegionCode)\n\n if (newRegionIndex > -1) {\n const newSelectedRegion = regions[newRegionIndex].shortCode\n\n setSelectedValue(newSelectedRegion)\n\n if (typeof onChange === 'function') {\n onChange(newSelectedRegion)\n }\n }\n }\n\n return (\n \n \n {t('Select region')}\n \n \n {regions.map((region) => {\n return (\n \n {region.name}\n \n )\n })}\n \n \n )\n}\n\nRegionSelectInput.propTypes = {\n country: PropTypes.string,\n disabled: PropTypes.bool,\n required: PropTypes.bool,\n value: PropTypes.string,\n onChange: PropTypes.func,\n}\n\nexport default RegionSelectInput\n","import { useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { Box, TextField } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nimport CountrySelectInput from '../CountrySelectInput/CountrySelectInput'\nimport LocationSearchInput from '../LocationSearchInput/LocationSearchInput'\nimport RegionSelectInput from '../RegionSelectInput/RegionSelectInput'\n\nconst AddressInput = ({\n address,\n autoFocus,\n autocompletePlaceholder,\n fullWidth = true,\n labelText,\n onChange,\n showAddressInputsOnValidation = false,\n sx,\n validateInput = true,\n}) => {\n const { t } = useTranslation()\n\n const [addressState, setAddressState] = useState({\n line1: '',\n line2: '',\n city: '',\n state: '',\n country: '',\n postcode: '',\n })\n\n const getFormattedAddress = (address) => {\n if (!address) {\n return ''\n }\n\n let addressStr = address.line1 ? address.line1 + ' ' : ''\n addressStr += address.line2 ? address.line2 + ' ' : ''\n addressStr += address.city ? address.city + ' ' : ''\n addressStr += address.state ? address.state + ' ' : ''\n addressStr += address.postcode ? address.postcode + ' ' : ''\n addressStr += address.country ? address.country + ' ' : ''\n\n return addressStr\n }\n\n const handleAddressChange = (newAddress) => {\n setAddressState({\n line1: newAddress.line1 ?? '',\n line2: newAddress.line2 ?? '',\n city: newAddress.city ?? '',\n state: newAddress.state ?? '',\n country: newAddress.country ?? '',\n postcode: newAddress.postcode ?? '',\n })\n\n if (typeof onChange === 'function') {\n onChange(newAddress)\n }\n }\n\n const handleChange = (propertyName, newValue) => {\n const newAddress = { ...addressState }\n\n newAddress[propertyName] = newValue\n\n handleAddressChange(newAddress)\n }\n\n const handleSearchLocationDidUpdate = (address) => {\n handleAddressChange(address)\n }\n\n useEffect(() => {\n if (address) {\n setAddressState(address)\n }\n }, [address])\n\n return (\n \n {validateInput ? (\n \n ) : null}\n {!validateInput || showAddressInputsOnValidation ? (\n \n \n handleChange('line1', event.target.value)}\n />\n handleChange('line2', event.target.value)}\n />\n handleChange('postcode', event.target.value)}\n />\n \n \n handleChange('city', event.target.value)}\n />\n handleChange('country', country)}\n />\n handleChange('state', region)}\n />\n \n \n ) : null}\n \n )\n}\n\nAddressInput.propTypes = {\n address: PropTypes.shape({\n city: PropTypes.string,\n country: PropTypes.string,\n line1: PropTypes.string,\n line2: PropTypes.string,\n postcode: PropTypes.string,\n state: PropTypes.string,\n }),\n autocompletePlaceholder: PropTypes.string,\n autoFocus: PropTypes.bool,\n fullWidth: PropTypes.bool,\n labelPosition: PropTypes.oneOf(['start', 'end', 'bottom', 'top']),\n labelText: PropTypes.string,\n ref: PropTypes.func,\n showAddressInputsOnValidation: PropTypes.bool,\n sx: PropTypes.oneOfType([\n PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.func, PropTypes.object, PropTypes.bool])),\n PropTypes.func,\n PropTypes.object,\n ]),\n validateInput: PropTypes.bool,\n onChange: PropTypes.func,\n onClick: PropTypes.func,\n onSubmit: PropTypes.func,\n}\n\nexport default AddressInput\n","import { useTranslation } from 'react-i18next'\nimport { Button, Dialog, DialogActions, DialogContent, DialogContentText, DialogTitle } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst AlertDialog = ({\n buttonVariant = 'text',\n cancelButtonText = 'Cancel',\n content,\n hideCancelButton = false,\n hideOkButton = false,\n okButtonText = 'Ok',\n onCancelClose,\n onOkClose,\n open,\n title,\n ...rest\n}) => {\n const { t } = useTranslation()\n\n return (\n \n {title}\n \n {content}\n \n \n {!hideCancelButton ? (\n \n {cancelButtonText}\n \n ) : null}\n {!hideOkButton ? (\n \n {okButtonText}\n \n ) : null}\n \n \n )\n}\n\nAlertDialog.propTypes = {\n buttonVariant: PropTypes.oneOf(['text', 'contained', 'outlined']),\n cancelButtonText: PropTypes.string,\n content: PropTypes.oneOfType([PropTypes.object, PropTypes.string]),\n hideCancelButton: PropTypes.bool,\n hideOkButton: PropTypes.bool,\n okButtonText: PropTypes.string,\n open: PropTypes.bool,\n title: PropTypes.string,\n onCancelClose: PropTypes.func,\n onOkClose: PropTypes.func,\n}\n\nexport default AlertDialog\n","import { AppBar as MuiAppBar, Toolbar as MuiToolbar } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst AppBar = ({ children, className, classes, color = 'default', position = 'fixed' }) => {\n return (\n \n {children}\n \n )\n}\n\nAppBar.propTypes = {\n children: PropTypes.node,\n classes: PropTypes.object,\n className: PropTypes.object,\n color: PropTypes.oneOf(['default', 'inherit', 'primary', 'secondary', 'transparent']),\n position: PropTypes.oneOf(['absolute', 'fixed', 'relative', 'static', 'sticky']),\n}\n\nexport default AppBar\n","import { useCallback } from 'react'\nimport { useSelector } from 'react-redux'\n\nimport { selectLocale } from '@/app/slices/appSlice'\nimport { selectCurrencyCode } from '@/app/slices/organisationSlice'\nimport { getCurrencyFormatter } from '@/common/utils'\n\nexport function useCurrencyFormatter(props) {\n const { currency, locale, ...options } = props || {}\n const appCurrency = useSelector(selectCurrencyCode)\n const appLocale = useSelector(selectLocale)\n const defaultCurrency = 'USD'\n const defaultLocale = 'en-US'\n\n const localeToUse = locale || appLocale || defaultLocale\n const currencyCodeToUse = currency || appCurrency || defaultCurrency\n\n const formatter = getCurrencyFormatter(currencyCodeToUse, localeToUse, options)\n\n const formatCurrency = useCallback(\n (value) => (value != null && !isNaN(value) ? formatter.format(value) : null),\n\n [formatter]\n )\n\n return {\n formatCurrency,\n formatter,\n }\n}\n","import { useCallback, useEffect, useReducer } from 'react'\n\nconst blacklistedTargets = ['INPUT', 'TEXTAREA']\n\nconst keysReducer = (state, action) => {\n switch (action.type) {\n case 'set-key-down':\n return { ...state, [action.key]: true }\n case 'set-key-up':\n return { ...state, [action.key]: false }\n default:\n return state\n }\n}\n\nconst useKeyboardShortcut = (shortcutKeys, callback) => {\n if (!Array.isArray(shortcutKeys)) {\n throw new Error(\n 'The first parameter to `useKeyboardShortcut` must be an ordered array of `KeyboardEvent.key` strings.'\n )\n }\n\n if (!shortcutKeys.length) {\n throw new Error(\n 'The first parameter to `useKeyboardShortcut` must contain atleast one `KeyboardEvent.key` string.'\n )\n }\n\n const initalKeyMapping = shortcutKeys.reduce((currentKeys, key) => {\n currentKeys[key.toLowerCase()] = false\n return currentKeys\n }, {})\n\n const [keys, setKeys] = useReducer(keysReducer, initalKeyMapping)\n\n const keydownListener = useCallback(\n (keydownEvent) => {\n const { key, repeat, target } = keydownEvent\n if (key !== undefined) {\n const loweredKey = key.toLowerCase()\n\n if (repeat) return\n if (blacklistedTargets.includes(target.tagName)) return\n if (keys[loweredKey] === undefined) return\n\n if (keys[loweredKey] === false) {\n setKeys({ type: 'set-key-down', key: loweredKey })\n }\n }\n },\n [keys]\n )\n\n const keyupListener = useCallback(\n (keyupEvent) => {\n const { key, target } = keyupEvent\n if (key !== undefined) {\n const loweredKey = key.toLowerCase()\n\n if (blacklistedTargets.includes(target.tagName)) return\n if (keys[loweredKey] === undefined) return\n\n if (keys[loweredKey] === true) {\n setKeys({ type: 'set-key-up', key: loweredKey })\n }\n }\n },\n [keys]\n )\n\n useEffect(() => {\n if (!Object.values(keys).filter((value) => !value).length) callback(keys)\n }, [callback, keys])\n\n useEffect(() => {\n window.addEventListener('keydown', keydownListener, true)\n return () => window.removeEventListener('keydown', keydownListener, true)\n }, [keydownListener])\n\n useEffect(() => {\n window.addEventListener('keyup', keyupListener, true)\n return () => window.removeEventListener('keyup', keyupListener, true)\n }, [keyupListener])\n}\n\nexport default useKeyboardShortcut\n","import { useSelector } from 'react-redux'\n\nimport { selectLocale } from '@/app/slices/appSlice'\n\nexport default function useNumberFormatter(props) {\n const config = props || {}\n const appLocale = useSelector(selectLocale)\n const defaultLocale = 'en-US'\n\n const locale = config.locale || appLocale || defaultLocale\n const numberOfDecimalPlaces = config.numberOfDecimalPlaces ?? 2\n\n const formatterOptions = {\n style: 'decimal',\n unitDisplay: 'long',\n signDisplay: 'auto',\n useGrouping: true,\n ...config,\n }\n\n formatterOptions.minimumFractionDigits = numberOfDecimalPlaces\n formatterOptions.maximumFractionDigits = numberOfDecimalPlaces\n\n const formatter = new Intl.NumberFormat(locale, formatterOptions)\n\n const n = (value) => {\n return value ? formatter.format(value) : value\n }\n\n return {\n n,\n }\n}\n","import { useEffect, useRef } from 'react'\n\nexport default function useEffectOnlyOnUpdate(callback, dependencies) {\n const didMount = useRef(false)\n\n useEffect(() => {\n if (didMount.current) {\n callback(dependencies)\n } else {\n didMount.current = true\n }\n }, [callback, dependencies])\n}\n","import { useEffect, useRef, useState } from 'react'\n\nexport const useOnAllImagesLoaded = (ref) => {\n const [isImageLoading, setIsImageLoading] = useState(true)\n const imageElements = useRef([])\n\n const updateStatus = () => {\n setIsImageLoading(imageElements?.current?.some((image) => !image.complete))\n }\n\n const cleanUpEventHandlers = () => {\n imageElements?.current?.forEach((image) => {\n image.removeEventListener('load', updateStatus)\n image.removeEventListener('error', updateStatus)\n })\n }\n\n useEffect(() => {\n imageElements.current =\n typeof ref?.current?.querySelectorAll === 'function' ? Array.from(ref.current.querySelectorAll('img')) : []\n\n cleanUpEventHandlers()\n\n if (!imageElements?.current?.length) {\n setIsImageLoading(true)\n return\n }\n\n imageElements?.current?.forEach((image) => {\n image.addEventListener('load', updateStatus)\n image.addEventListener('error', updateStatus)\n })\n\n updateStatus()\n\n return cleanUpEventHandlers\n })\n\n return {\n isImageLoading,\n }\n}\n","import { api } from './api'\n\nconst apiVersion = 'v2'\n\nexport const cuttingTechnologiesApi = api\n .enhanceEndpoints({\n addTagTypes: ['CuttingTechnologies', 'MaterialThicknesses', 'ThicknessSheets'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getCuttingTechnologies: builder.query({\n query: ({ organisationId, params }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies`,\n params,\n }),\n transformResponse: (response) => {\n delete response.$type\n\n return response\n },\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'CuttingTechnologies', id: 'LIST' },\n ...(Object.keys(result).map((cuttingTechnologyId) => ({\n type: 'CuttingTechnologies',\n id: cuttingTechnologyId,\n })) ?? []),\n ]\n : [{ type: 'CuttingTechnologies', id: 'LIST' }],\n }),\n\n getCuttingTechnology: builder.query({\n query: ({ cuttingTechnologyId, organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies/${cuttingTechnologyId}`,\n }),\n providesTags: (_result, _error, arg) => [{ type: 'CuttingTechnologies', id: arg.cuttingTechnologyId }],\n }),\n\n getCuttingTechnologyMaterials: builder.query({\n query: ({ cuttingTechnologyId, organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies/${cuttingTechnologyId}/materials`,\n method: 'GET',\n }),\n transformResponse: (response) => {\n return response?.sort((a, b) => {\n if (a.materialName?.toLocaleLowerCase() < b.materialName?.toLocaleLowerCase()) {\n return -1\n }\n if (a.materialName?.toLocaleLowerCase() > b.materialName?.toLocaleLowerCase()) {\n return 1\n }\n return 0\n })\n },\n }),\n\n getMaterialThicknesses: builder.query({\n query: ({ cuttingTechnologyId, materialId, organisationId, params }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies/${cuttingTechnologyId}/materials/${materialId}/thicknesses`,\n method: 'GET',\n params,\n }),\n providesTags: ['MaterialThicknesses'],\n }),\n getThicknessSheets: builder.query({\n query: ({ cuttingTechnologyId, materialId, organisationId, params }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies/${cuttingTechnologyId}/materials/${materialId}/thicknesses/sheets`,\n method: 'GET',\n params,\n }),\n providesTags: ['ThicknessSheets'],\n }),\n\n createCuttingTechnology: builder.mutation({\n query: ({ cuttingTechnology, organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies/`,\n method: 'POST',\n body: cuttingTechnology,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'CuttingTechnologies', id: arg.cuttingTechnology.cuttingTechnologyId },\n { type: 'CuttingTechnologies', id: 'LIST' },\n ],\n }),\n updateCuttingTechnology: builder.mutation({\n query: ({ cuttingTechnology, organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/cutting-technologies/${cuttingTechnology.cuttingTechnologyId}`,\n method: 'PUT',\n body: cuttingTechnology,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'CuttingTechnologies', id: arg.cuttingTechnology.cuttingTechnologyId },\n { type: 'CuttingTechnologies', id: 'LIST' },\n ],\n }),\n\n deleteCuttingTechnology: builder.mutation({\n query: ({ cuttingTechnologyId, organisationId }) => ({\n url: `/organisations/${organisationId}/cutting-technology/${cuttingTechnologyId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'CuttingTechnologies', id: 'LIST' },\n { type: 'CuttingTechnologies', id: arg.cuttingTechnologyId },\n ],\n }),\n }),\n })\n\nexport const {\n useCreateCuttingTechnologyMutation,\n useDeleteCuttingTechnologyMutation,\n useGetCuttingTechnologiesQuery,\n useGetCuttingTechnologyQuery,\n useLazyGetCuttingTechnologyMaterialsQuery,\n useLazyGetMaterialThicknessesQuery,\n useLazyGetThicknessSheetsQuery,\n useUpdateCuttingTechnologyMutation,\n} = cuttingTechnologiesApi\n","import { api } from './api'\n\nconst apiVersion = 'v2'\n\nexport const materialsApi = api\n .enhanceEndpoints({\n addTagTypes: ['Materials'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getMaterials: builder.query({\n query: ({ organisationId }) => `/${apiVersion}/organisation/${organisationId}/materials`,\n transformResponse: (response) => {\n delete response.$type\n\n return response\n },\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'Materials', id: 'LIST' },\n\n ...(Object.values(result)?.map(({ materialId }) => ({\n type: 'Materials',\n id: materialId,\n })) ?? []),\n ]\n : [{ type: 'Materials', id: 'LIST' }],\n }),\n\n getMaterial: builder.query({\n query: ({ materialId, organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/materials/${materialId}`,\n }),\n providesTags: (result, _error, arg) => [\n { type: 'Materials', id: arg.materialId },\n { type: 'Sheets', id: 'LIST' },\n ...(result.sheets.map(({ sheetId }) => ({ type: 'Sheets', id: sheetId })) ?? []),\n ],\n }),\n\n createMaterial: builder.mutation({\n query: ({ material, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/create`,\n method: 'POST',\n body: { ...material, organisationId },\n }),\n invalidatesTags: [{ type: 'Materials', id: 'LIST' }],\n }),\n\n checkNameAvailability: builder.query({\n query: ({ materialName, materialType, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/check-name-availability/${materialName}/${materialType}`,\n method: 'GET',\n }),\n }),\n\n updateMaterial: builder.mutation({\n query: ({ material, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/${material.materialId}`,\n method: 'PUT',\n body: material,\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'Materials', id: arg.material.materialId }],\n }),\n\n archiveMaterial: builder.mutation({\n query: ({ materialId, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/${materialId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Materials', id: arg.materialId },\n { type: 'QuoteItems' },\n { type: 'Quotes' }\n ],\n }),\n\n unarchiveMaterial: builder.mutation({\n query: ({ materialId, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/${materialId}/unarchive`,\n method: 'PUT',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Materials', id: arg.materialId },\n { type: 'QuoteItems' },\n { type: 'Quotes' }\n ],\n }),\n\n updateMaterialRateTable: builder.mutation({\n query: ({ materialRateTable, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/material-rate-table/${materialRateTable.materialRateTableId}`,\n method: 'PUT',\n body: materialRateTable,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Materials', id: arg.materialRateTable.materialId },\n ],\n }),\n\n deleteMaterialRateTable: builder.mutation({\n query: ({ materialRateTableId, organisationId }) => ({\n url: `/organisations/${organisationId}/materials/material-rate-table/${materialRateTableId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Materials', id: arg.materialRateTableId.materialId },\n ],\n }),\n\n createMaterialList: builder.mutation({\n query: ({ organisationId }) => ({\n url: `/organisations/${organisationId}/materials`,\n method: 'POST',\n }),\n }),\n\n createQuoteMaterialList: builder.mutation({\n query: ({ location, organisationId }) => {\n let url = `/organisations/${organisationId}/quote-materials`\n if (location && location.length > 0) {\n url += `?location=${location}`\n }\n return {\n url,\n method: 'POST',\n }\n },\n }),\n }),\n })\n\nexport const {\n useArchiveMaterialMutation,\n useCheckNameAvailabilityQuery,\n useCreateMaterialListMutation,\n useCreateMaterialMutation,\n useCreateQuoteMaterialListMutation,\n useDeleteMaterialRateTableMutation,\n useGetMaterialQuery,\n useGetMaterialsQuery,\n useLazyGetMaterialQuery,\n useLazyGetMaterialsQuery,\n useUnarchiveMaterialMutation,\n useUpdateMaterialMutation,\n useUpdateMaterialRateTableMutation,\n} = materialsApi\n","import { api } from './api'\n\nconst apiVersion = 'v2'\n\nexport const rateTablesApi = api\n .enhanceEndpoints({\n addTagTypes: ['RateTables'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getRateTables: builder.query({\n query: ({ organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/rate-tables`,\n }),\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'RateTables', id: 'LIST' },\n ...(Object.values(result).map(({ rateTableId }) => ({\n type: 'RateTables',\n id: rateTableId,\n })) ?? []),\n ]\n : [{ type: 'RateTables', id: 'LIST' }],\n }),\n\n getRateTable: builder.query({\n query: ({ organisationId, rateTableId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/rate-tables/${rateTableId}`,\n }),\n providesTags: (result, _error, arg) => [\n { type: 'RateTables', id: arg.rateTableId },\n { type: 'Rates', id: 'LIST' },\n ...(result.rates.map(({ rateId }) => ({ type: 'Rates', id: rateId })) ?? []),\n ],\n }),\n\n createRateTable: builder.mutation({\n query: ({ organisationId, rateTable }) => ({\n url: `/organisations/${organisationId}/rate-table`,\n method: 'PUT',\n body: rateTable,\n }),\n invalidatesTags: [{ type: 'RateTables', id: 'LIST' }],\n }),\n\n updateRateTable: builder.mutation({\n query: ({ organisationId, rateTable }) => ({\n url: `/organisations/${organisationId}/rate-table`,\n method: 'PUT',\n body: rateTable,\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'RateTables', id: arg.rateTable.rateTableId }],\n }),\n\n archiveRateTable: builder.mutation({\n query: ({ organisationId, rateTableId }) => ({\n url: `/organisations/${organisationId}/rate-table/${rateTableId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'RateTables', id: 'LIST' },\n { type: 'RateTables', id: arg.rateTableId },\n ],\n }),\n\n unarchiveRateTable: builder.mutation({\n query: ({ organisationId, rateTableId }) => ({\n url: `/organisations/${organisationId}/rate-table/${rateTableId}/unarchive`,\n }),\n invalidatesTags: (result, _error, arg) => [{ type: 'RateTables', id: arg.rateTableId }],\n }),\n\n duplicateRateTable: builder.mutation({\n query: ({ newRateTableName, organisationId, rateTableId }) => ({\n url: `/organisations/${organisationId}/rate-table/${rateTableId}/duplicate/${newRateTableName}`,\n method: 'POST',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'RateTables', id: 'LIST' },\n { type: 'RateTables', id: arg.rateTableId },\n ],\n }),\n\n validateRateTableName: builder.mutation({\n query: ({ organisationId, requestBody }) => ({\n url: `/organisations/${organisationId}/rate-table/validate-name`,\n method: 'POST',\n body: requestBody,\n }),\n }),\n }),\n })\n\nexport const {\n useArchiveRateTableMutation,\n useCreateRateTableMutation,\n useDuplicateRateTableMutation,\n useGetRateTableQuery,\n useGetRateTablesQuery,\n useLazyGetRateTableQuery,\n useLazyGetRateTablesQuery,\n useUnarchiveRateTableMutation,\n useUpdateRateTableMutation,\n useValidateRateTableNameMutation,\n} = rateTablesApi\n","import { api } from './api'\n\nconst apiVersion = 'v2'\n\nexport const sheetsApi = api\n .enhanceEndpoints({\n addTagTypes: ['Sheets'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getSheets: builder.query({\n query: ({ organisationId }) => ({\n url: `/${apiVersion}/organisation/${organisationId}/sheets`,\n }),\n transformResponse: (response) => {\n delete response.$type\n\n return response\n },\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'Sheets', id: 'LIST' },\n\n ...(Object.keys(result)?.map((sheetId) => ({\n type: 'Sheets',\n id: sheetId,\n })) ?? []),\n ]\n : [{ type: 'Sheets', id: 'LIST' }],\n }),\n\n updateSheet: builder.mutation({\n query: ({ organisationId, sheet }) => ({\n url: `/organisations/${organisationId}/sheets`,\n method: 'PUT',\n body: sheet,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Sheets', id: 'LIST' },\n { type: 'Sheets', id: arg.sheet.sheetId },\n { type: 'Materials', id: arg.sheet.materialId },\n { type: 'QuoteItems' },\n { type: 'Quotes' },\n { type: 'MaterialThicknesses' },\n { type: 'ThicknessSheets' },\n ],\n }),\n\n sheetsBulkUpdate: builder.mutation({\n query: ({ organisationId, updatedSheets }) => ({\n url: `/organisations/${organisationId}/sheets/bulk-update`,\n method: 'PUT',\n body: { sheets: updatedSheets },\n }),\n invalidatesTags: (_result, _error, arg) =>\n arg.updatedSheets.map((sheet) => ({ type: 'Sheets', id: sheet.sheetId })),\n }),\n\n updateSheetCuttingTechnology: builder.mutation({\n query: ({ organisationId, sheetCuttingTechnology }) => ({\n url: `/organisations/${organisationId}/sheets/sheet-cutting-technology`,\n method: 'PUT',\n body: sheetCuttingTechnology,\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Sheets', id: arg.sheetCuttingTechnology.sheetId },\n { type: 'CuttingTechnologies', id: arg.sheetCuttingTechnology.cuttingTechnologyId },\n ],\n }),\n\n deleteSheet: builder.mutation({\n query: ({ organisationId, sheetId }) => ({\n url: `/organisations/${organisationId}/sheets/${sheetId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Sheets', id: 'LIST' },\n { type: 'Sheets', id: arg.sheetId },\n { type: 'QuoteItems' },\n { type: 'Quotes' },\n ],\n }),\n\n deleteSheetCuttingTechnology: builder.mutation({\n query: ({ organisationId, sheetCuttingTechnologyId }) => ({\n url: `/organisations/${organisationId}/sheets/sheet-cutting-technology/${sheetCuttingTechnologyId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Sheets', id: 'LIST' },\n { type: 'Sheets', id: arg.sheetCuttingTechnology.sheetId },\n { type: 'CuttingTechnologies', id: 'LIST' },\n { type: 'CuttingTechnologies', id: arg.sheetCuttingTechnology.cuttingTechnologyId },\n ],\n }),\n\n setSheetAsDefault: builder.mutation({\n query: ({ organisationId, sheetId }) => ({\n url: `/organisations/${organisationId}/sheets/${sheetId}/default`,\n method: 'PUT',\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'Sheets', id: arg.sheetId }],\n }),\n\n setSheetAsWebStoreEnabled: builder.mutation({\n query: ({ isWebStoreEnabled, organisationId, sheetId }) => ({\n url: `/organisations/${organisationId}/sheets/${sheetId}/store/${\n isWebStoreEnabled ? 'enable' : 'disable'\n }`,\n method: 'PUT',\n }),\n invalidatesTags: (_result, _error, arg) => [{ type: 'Sheets', id: arg.sheetId }],\n }),\n\n importSheets: builder.mutation({\n query: ({ emptyFieldsTreatment, file, materialId, organisationId }) => {\n const formData = new FormData()\n formData.append('file', file, file.name)\n return {\n url: `/organisations/${organisationId}/sheets/${materialId}/import?emptyFieldsTreatment=${emptyFieldsTreatment}`,\n method: 'POST',\n body: formData,\n }\n },\n invalidatesTags: (_result, _error, arg) => [\n { type: 'Sheets', id: 'LIST' },\n { type: 'Materials', id: arg.materialId },\n ],\n }),\n\n exportSheets: builder.query({\n query: ({ materialId, organisationId }) => ({\n url: `/organisations/${organisationId}/sheets/${materialId}/export`,\n responseHandler: (response) => {\n return response.text()\n },\n }),\n }),\n\n downloadSheetTemplate: builder.query({\n query: ({ materialId, organisationId }) => ({\n url: `/organisations/${organisationId}/sheets/${materialId}/export-template`,\n responseHandler: (response) => {\n return response.text()\n },\n }),\n }),\n }),\n })\n\nexport const {\n useDeleteSheetCuttingTechnologyMutation,\n useDeleteSheetMutation,\n useGetSheetsQuery,\n useImportSheetsMutation,\n useLazyDownloadSheetTemplateQuery,\n useLazyExportSheetsQuery,\n useSetSheetAsDefaultMutation,\n useSetSheetAsWebStoreEnabledMutation,\n useSheetsBulkUpdateMutation,\n useUpdateSheetCuttingTechnologyMutation,\n useUpdateSheetMutation,\n} = sheetsApi\n","import { useCallback, useMemo } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { useSelector } from 'react-redux'\n\nimport { useGetCuttingTechnologiesQuery } from '@/app/services/cuttingTechnologies'\nimport { useGetMaterialsQuery } from '@/app/services/materials'\nimport { useGetRateTablesQuery } from '@/app/services/rateTables'\nimport { useGetSheetsQuery } from '@/app/services/sheets'\nimport { selectOrganisation, selectOrganisationId } from '@/app/slices/organisationSlice'\nimport { isProfileSizeFit, thicknessSameWithTolerance } from '@/common/utils/SheetUtils/SheetUtils'\n\nconst compareMaterials = (a, b) => {\n if (a.materialName.toLocaleLowerCase() < b.materialName.toLocaleLowerCase()) {\n return -1\n }\n if (a.materialName.toLocaleLowerCase() > b.materialName.toLocaleLowerCase()) {\n return 1\n }\n return 0\n}\n\nexport function useQuoteItemPropertyOptions({\n cuttingTechnologyId: cuttingTechnologyValue,\n materialId: materialValue,\n mode,\n partDimensions,\n thickness: thicknessValue,\n}) {\n const { t } = useTranslation()\n const organisationId = useSelector(selectOrganisationId)\n const organisation = useSelector(selectOrganisation)\n const { data: materials } = useGetMaterialsQuery({ organisationId })\n const { data: sheets } = useGetSheetsQuery({ organisationId })\n const { data: cuttingTechnologies } = useGetCuttingTechnologiesQuery({ organisationId })\n const { data: rateTables } = useGetRateTablesQuery({ organisationId })\n\n const customersMaterialOptions = [\n { label: t('Yes'), value: 'true' },\n { label: t('No'), value: 'false' },\n ]\n const consumptionModeOptions = [\n {\n label: t(mode === 'flat' ? 'Whole Sheet' : 'Whole Length'),\n value: 'WholeSheet',\n },\n { label: t('Nest Bounds'), value: 'NestBounds' },\n ]\n\n const grainDirectionOptions = [\n { label: t('None'), value: 'None' },\n { label: t('Horizontal'), value: 'Horizontal' },\n { label: t('Vertical'), value: 'Vertical' },\n ]\n\n const filterSheet = useCallback(\n (sheet) => {\n const cuttingTech = sheet?.sheetCuttingTechnology?.find(\n (cuttingTech) => cuttingTech?.cuttingTechnologyId === cuttingTechnologyValue\n )\n\n const isNotDisabled = cuttingTech?.isEnabled ?? true\n return !sheet?.isDeleted && isNotDisabled && isProfileSizeFit(sheet, partDimensions, organisation)\n },\n [cuttingTechnologyValue, partDimensions, organisation]\n )\n\n const cuttingTechOptions = useMemo(() => {\n if (cuttingTechnologies)\n return Object.values(cuttingTechnologies).filter(\n (tech) => tech?.type?.toLowerCase() === mode && !tech?.isDeleted\n )\n }, [cuttingTechnologies, mode])\n\n const materialRateTableFilter = useCallback(\n (materialRateTable) =>\n rateTables && rateTables[materialRateTable.rateTableId]?.cuttingTechnologyId === cuttingTechnologyValue,\n [rateTables, cuttingTechnologyValue]\n )\n\n const materialFilter = useCallback(\n (material) =>\n material.materialRateTables.some(materialRateTableFilter) &&\n !material.isDeleted &&\n material.type.toLowerCase() === mode,\n [materialRateTableFilter, mode]\n )\n\n const materialOptions = useMemo(() => {\n if (!materials) {\n return\n }\n return Object.values(materials).filter(materialFilter)?.sort(compareMaterials)\n }, [materials, materialFilter])\n\n const thicknessOptions = useMemo(() => {\n if (!materials || !sheets) {\n return\n }\n const material = materials[materialValue]\n const sheetsForMaterial = material?.sheetIds.map((sheetId) => sheets[sheetId]).filter(filterSheet)\n const thicknesses = sheetsForMaterial?.map((sheet) => sheet?.thickness)?.sort((a, b) => a - b)\n return [...new Set(thicknesses ?? [])]\n }, [materialValue, materials, sheets, filterSheet])\n\n const sheetOptions = useMemo(() => {\n if (!materials || !sheets) {\n return\n }\n const material = materials[materialValue]\n const sheetsForMaterial = material?.sheetIds\n ?.map((sheetId) => sheets[sheetId])\n ?.filter((sheet) => sheet?.thickness === thicknessValue)\n ?.filter(filterSheet)\n ?.filter((sheet) =>\n mode === 'flat'\n ? sheet?.thickness === thicknessValue\n : thicknessSameWithTolerance(sheet, thicknessValue, partDimensions, organisation)\n )\n return (\n sheetsForMaterial?.map((sheet) => ({\n sheetId: sheet?.sheetId,\n sheetLength: sheet?.materialLength,\n sheetWidth: sheet?.sheetWidth,\n sheetHeight: sheet?.sheetHeight,\n sheetSize: `${sheet?.sheetWidth ?? ''} x ${sheet?.sheetHeight ?? ''}`,\n sheetCost: sheet?.sheetCost,\n sheetRatePrice: sheet?.sheetRatePrice,\n expiryDate: sheet?.expiryDate,\n isSoldByOrganisation: sheet?.isSoldByOrganisation,\n defaultMaterialConsumptionMode: sheet?.defaultMaterialConsumptionMode,\n })) ?? []\n )\n }, [materialValue, materials, sheets, filterSheet, thicknessValue])\n\n return {\n cuttingTechOptions,\n materialOptions,\n thicknessOptions,\n sheetOptions,\n customersMaterialOptions,\n consumptionModeOptions,\n grainDirectionOptions,\n }\n}\n","import { useMemo } from 'react'\nimport { useSelector } from 'react-redux'\nimport { useSplitTreatments } from '@splitsoftware/splitio-react'\n\nimport { selectCurrentUser } from '@/app/slices/appSlice'\nimport { getProductDetails, RESELLERS } from '@/common/utils'\n\nconst splitTreatments = {\n show3dButton: 'ToolBox_Show3DPartsButton',\n replaceSplinesWithLinearSegments: 'Toolbox_ReplaceSplineWithLinearSegmentsUnfold3d',\n showPricing: 'ToolBox_quoting_iteration1',\n showQuotingIteration2: 'ToolBox_quoting_iteration2',\n showQuickBooks: 'ToolBox_QuickBooks',\n showOrdering: 'ToolBox_ordering',\n showCustomers: 'ToolBox_customers',\n showFolding: 'ToolBox_folding',\n showLibellulaWhitelabelling: 'ToolBox_libellula_whitelabel',\n showStrikerWhitelabelling: 'ToolBox_Striker_whitelabel',\n showSecondaryProcesses: 'ToolBox_secondary_operations',\n showFoldingProductionWarnings: 'ToolBox_FoldingProductionWarnings',\n showAdminTools: 'ToolBox_ShowAdminTools',\n showCustomerCentral: 'ToolBox_CustomerCentral',\n showMaterialRevamp: 'ToolBox_material_revamp',\n showNestView: 'ToolBox_DisplayNestView',\n showCuttingTechManagement: 'ToolBox_ShowCuttingTechManagement',\n showPartLibrary: 'ToolBox_PartLibrary',\n showTube: 'ToolBox_Tube',\n showXero: 'ToolBox_Xero',\n isReady: false,\n showCsvImportExportButtons: 'ToolBox_ImportExportSheetsRateTables',\n showWebStore: 'ToolBox_WebStore',\n showPdfButton: 'ToolBox_PDF',\n showVoucherCode: 'ToolBox_VoucherCode',\n showUserflowTooltips: 'ToolBox_TooltipsToggle',\n showEmailCustomerModal: 'ToolBox_EmailCustomerModal',\n showPDFAttachmentCheckbox: 'ToolBox_PDFAttachments',\n previewResellerName: undefined,\n showIntegrations: false,\n showPdfFooterFields: 'ToolBox_PDFFooterFields',\n showDownloadPdfs: 'ToolBox_DownloadPDFs',\n showProductionPDFs: 'ToolBox_ProductionPDFs',\n showMiscItems: 'ToolBox_misc_items',\n showFixedPriceParts: 'ToolBox_FixedPriceParts',\n showMultipleTaxRates: 'ToolBox_MultipleTaxRates',\n showPayments: 'ToolBox_Payments',\n showCustomerTaxRates: 'ToolBox_CustomerTaxRates',\n showViewer3D: 'ToolBox_ShowViewer3D',\n showViewerTube: 'ToolBox_ShowViewerTube',\n showRetryIntegration: 'ToolBox_RetryIntegration',\n showShipping: 'ToolBox_Shipping',\n showIntegrationStatus: 'ToolBox_QBOXeroSuccessStatus',\n showGrainDirection: 'ToolBox_GrainDirection',\n showWebStoreSubdomains: 'ToolBox_WebStoreSubdomains',\n showGoogleDriveIntegration: 'ToolBox_GoogleDrive',\n showSecondaryProcessesWebstore: 'ToolBox_SecProcWebstore',\n}\n\nconst resellerName = import.meta.env.VITE_RESELLER\n\nexport function useToolBoxTreatments() {\n const currentUser = useSelector(selectCurrentUser)\n\n const productDetails = getProductDetails()\n\n const attributes = {\n emailAddress: currentUser ? currentUser.emailAddress : undefined,\n reseller: resellerName ? RESELLERS[resellerName].name : undefined,\n }\n\n const cleanSplitTreatments = { ...splitTreatments }\n delete cleanSplitTreatments.isReady\n delete cleanSplitTreatments.previewResellerName\n delete cleanSplitTreatments.showIntegrations\n\n const { isReady, treatments } = useSplitTreatments({ names: Object.values(cleanSplitTreatments), attributes })\n\n const resultTreatments = useMemo(() => {\n const result = {\n ...splitTreatments,\n }\n return Object.keys(splitTreatments).reduce((result, key) => {\n result[key] = treatments[splitTreatments[key]]?.treatment === 'on'\n return result\n }, result)\n }, [treatments])\n\n resultTreatments.isReady = isReady\n resultTreatments.showLibellulaWhitelabelling =\n resultTreatments.showLibellulaWhitelabelling || import.meta.env.VITE_RESELLER === RESELLERS.libellula.name\n resultTreatments.showStrikerWhitelabelling =\n resultTreatments.showStrikerWhitelabelling || import.meta.env.VITE_RESELLER === RESELLERS.striker.name\n\n resultTreatments.previewResellerName = resultTreatments.showLibellulaWhitelabelling\n ? RESELLERS.libellula.name\n : resultTreatments.showStrikerWhitelabelling\n ? RESELLERS.striker.name\n : undefined\n resultTreatments.showWebStore = resultTreatments.showWebStore && productDetails.hasWebStoreAccess\n resultTreatments.showIntegrations = resultTreatments.showXero || resultTreatments.showQuickBooks\n\n return resultTreatments\n}\n","import { api } from './api'\n\nexport const organistionUsersApi = api\n .enhanceEndpoints({\n addTagTypes: ['OrganisationUsers'],\n })\n .injectEndpoints({\n endpoints: (builder) => ({\n getOrganisationUsers: builder.query({\n query: ({ organisationId }) => `/organisations/${organisationId}/users`,\n providesTags: (result, _error, _arg) =>\n result\n ? [\n { type: 'OrganisationUsers', id: 'LIST' },\n ...result.map(({ userId }) => ({ type: 'OrganisationUsers', id: userId })),\n ]\n : [{ type: 'OrganisationUsers', id: 'LIST' }],\n }),\n\n getOrganisationUser: builder.query({\n query: ({ organisationId, userId }) => `/organisations/${organisationId}/users/${userId}`,\n providesTags: (_result, _error, arg) => [{ type: 'OrganisationUsers', id: arg.userId }],\n }),\n\n addOrganisationUser: builder.mutation({\n query: ({ emailAddress, firstName, lastName, organisationId, role }) => ({\n url: `/organisations/${organisationId}/users`,\n method: 'POST',\n body: { firstName, lastName, emailAddress, role },\n }),\n invalidatesTags: [{ type: 'OrganisationUsers', id: 'LIST' }],\n }),\n\n updateOrganisationUser: builder.mutation({\n query: ({ organisationId, role, userId }) => ({\n url: `/organisations/${organisationId}/users/${userId}`,\n method: 'POST',\n body: { role },\n }),\n //invalidates updated user only\n invalidatesTags: (_result, _error, arg) => [{ type: 'OrganisationUsers', id: arg.userId }],\n }),\n\n removeOrganisationUser: builder.mutation({\n query: ({ organisationId, userId }) => ({\n url: `/organisations/${organisationId}/users/${userId}`,\n method: 'DELETE',\n }),\n invalidatesTags: (result, _error, arg) => [{ type: 'OrganisationUsers', id: arg.userId }],\n }),\n }),\n })\n\nexport const {\n useAddOrganisationUserMutation,\n useGetOrganisationUserQuery,\n useGetOrganisationUsersQuery,\n useLazyGetOrganisationUserQuery,\n useLazyGetOrganisationUsersQuery,\n useRemoveOrganisationUserMutation,\n useUpdateOrganisationUserMutation,\n} = organistionUsersApi\n","const channelInstances = {}\n\nexport const getSingletonChannel = (name) => {\n if (!channelInstances[name]) {\n channelInstances[name] = new BroadcastChannel(name)\n }\n\n return channelInstances[name]\n}\n","import { useCallback, useEffect, useMemo, useRef } from 'react'\n\nimport { getSingletonChannel } from '../utils/GetSingletonChannel'\n\nexport function useBroadcastChannel(channelName, onMessageReceived) {\n const channel = useMemo(() => getSingletonChannel(channelName), [channelName])\n const isSubscribed = useRef(false)\n\n useEffect(() => {\n if (!isSubscribed.current) {\n if (!channel) {\n console.error('Channel is not defined')\n } else {\n channel.onmessage = (event) => onMessageReceived(event.data)\n }\n }\n\n return () => {\n if (isSubscribed.current) {\n channel.close()\n isSubscribed.current = true\n }\n }\n }, [])\n\n const postMessage = useCallback(\n (message) => {\n try {\n channel?.postMessage(message)\n } catch (error) {\n console.error('Failed to post message:', error)\n }\n },\n [channel]\n )\n\n return {\n postMessage,\n }\n}\n","import { useTranslation } from 'react-i18next'\nimport { useDispatch, useSelector } from 'react-redux'\nimport { useNavigate } from 'react-router-dom'\nimport { useAppInsightsContext } from '@microsoft/applicationinsights-react-js'\n\nimport { api } from '@/app/services/api'\nimport { useLazyGetMaterialsQuery } from '@/app/services/materials'\nimport { useLazyGetOrganisationQuery } from '@/app/services/organisation'\nimport { useLazyGetOrganisationUserQuery } from '@/app/services/organisationUsers'\nimport { useGetOrCreateUserContextMutation, useLazyGetUserOrganisationsQuery } from '@/app/services/user'\nimport {\n selectCurrentUser,\n setAppIsLoading,\n setCurrentUser,\n setLanguage,\n setLocale,\n setUserRole,\n} from '@/app/slices/appSlice'\nimport { setLastPaymentStatus, setOrganisation } from '@/app/slices/organisationSlice'\nimport { getLanguageToUse, getLocaleToUse, Paths } from '@/common/utils'\n\nimport { useBroadcastChannel } from './useBroadcastChannel'\n\nexport const useUserContextSwitcher = () => {\n const navigate = useNavigate()\n\n const dispatch = useDispatch()\n const appInsights = useAppInsightsContext()\n const { i18n } = useTranslation()\n\n const currentUser = useSelector(selectCurrentUser)\n\n const [getOrCreateUserContext] = useGetOrCreateUserContextMutation()\n const [getOrganisation] = useLazyGetOrganisationQuery()\n const [getOrganisationUser] = useLazyGetOrganisationUserQuery()\n const [getMaterials] = useLazyGetMaterialsQuery()\n const [getUserOrganisations] = useLazyGetUserOrganisationsQuery()\n\n const { postMessage } = useBroadcastChannel('orgChangeChannel', (message) => {\n try {\n if (message === 'ORG_CHANGED') {\n navigate(Paths.DASHBOARD_PATHNAME)\n window.location.reload()\n }\n } catch (error) {\n console.error('Failed to process message:', error)\n }\n })\n\n const trackEvent = ({ data, eventName }) => appInsights.trackEvent({ name: eventName, properties: data })\n\n const determineRedirectPath = (organisationResult) => {\n if (\n !organisationResult?.hasActiveSubscription &&\n (!organisationResult.trial || organisationResult.trial.hasExpired) &&\n !organisationResult.freePlan\n ) {\n return Paths.NO_ACTIVE_SUBSCRIPTION_PATHNAME\n }\n return Paths.DASHBOARD_PATHNAME\n }\n\n const setSelectedOrganisation = async (newSelectedOrganisationId) => {\n dispatch(setAppIsLoading(true))\n dispatch(api.util.resetApiState())\n\n try {\n const userContext = await getOrCreateUserContext({\n user: { ...currentUser, lastAccessedOrganisationId: newSelectedOrganisationId },\n }).unwrap()\n\n const appUser = userContext.user\n const selectedOrganisationId = userContext.selectedOrganisationId\n\n let organisationResult\n try {\n organisationResult = await getOrganisation({ organisationId: selectedOrganisationId }).unwrap()\n } catch (error) {\n console.error('An error occurred handling organisation.', error)\n }\n\n let organisationUserResult\n try {\n organisationUserResult = await getOrganisationUser({\n organisationId: selectedOrganisationId,\n userId: appUser.userId,\n }).unwrap()\n } catch (error) {\n console.error('An error occurred handling organisation user.', error)\n }\n\n try {\n await getMaterials({ organisationId: selectedOrganisationId })\n } catch (error) {\n console.error('An error occurred handling materials.', error)\n }\n\n await getUserOrganisations({ userId: appUser.userId })\n\n const role = organisationUserResult?.role?.toLowerCase() || 'user'\n const lastPaymentStatus = userContext.lastPayment?.status?.toLowerCase() || 'unknown'\n const localeToUse = getLocaleToUse(organisationResult)\n const languageToUse = getLanguageToUse(appUser, organisationResult)\n\n i18n.changeLanguage(languageToUse)\n dispatch(setCurrentUser(appUser))\n dispatch(setOrganisation(organisationResult))\n dispatch(setUserRole(role))\n dispatch(setLastPaymentStatus(lastPaymentStatus))\n dispatch(setLanguage(languageToUse))\n dispatch(setLocale(localeToUse))\n dispatch(setAppIsLoading(false))\n\n const redirectTo = determineRedirectPath(organisationResult)\n\n trackEvent({\n eventName: `User redirected to ${redirectTo}`,\n data: { currentUser: appUser, organisation: organisationResult },\n })\n\n navigate(redirectTo)\n\n postMessage('ORG_CHANGED')\n return organisationResult\n } catch (e) {\n dispatch(setAppIsLoading(false))\n navigate(Paths.ERROR_PATHNAME)\n }\n }\n\n return {\n setSelectedOrganisation,\n }\n}\n","import { Box } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nimport { useToolBoxTreatments } from '@/common/hooks'\nimport { getProductDetails } from '@/common/utils'\n\nconst classes = {\n root: {\n height: 128,\n px: 2,\n py: 1,\n },\n}\n\nconst AppLogo = ({ className, logoUrl = null, onClick }) => {\n const { previewResellerName } = useToolBoxTreatments()\n\n const productDetails = getProductDetails(previewResellerName)\n const defaultLogoUrl = `${productDetails.logo}`\n\n const logoUrlToUse = logoUrl || defaultLogoUrl\n\n return (\n \n )\n}\n\nAppLogo.propTypes = {\n className: PropTypes.object,\n logoUrl: PropTypes.string,\n onClick: PropTypes.func,\n}\n\nexport default AppLogo\n","import { forwardRef } from 'react'\nimport { Button as MuiButton } from '@mui/material'\n\nconst Button = forwardRef((props, ref) => {\n const { children, ...other } = props\n\n return (\n \n {children}\n \n )\n})\n\nButton.displayName = 'Button'\n\nexport default Button\n","import { styled } from '@mui/material/styles'\nimport Tooltip, { tooltipClasses } from '@mui/material/Tooltip'\nimport PropTypes from 'prop-types'\n\nconst StyledTooltip = styled(({ className, ...props }) => (\n \n))(({ theme }) => ({\n [`& .${tooltipClasses.tooltip}`]: {\n fontFamily: theme.typography.inter,\n color: 'white',\n backgroundColor: 'rgba(0, 0, 0, 0.9)',\n fontWeight: 400,\n fontSize: '12px',\n lineHeight: '18px',\n padding: '6px 12px',\n },\n [`& .${tooltipClasses.arrow}`]: {\n '&:before': {\n backgroundColor: 'rgba(0, 0, 0, 0.9)',\n },\n },\n}))\n\nconst TbxTooltip = ({\n arrow = true,\n children,\n disableFocusListener = false,\n disableTouchListener = false,\n followCursor = false,\n leaveDelay,\n title,\n}) => {\n return (\n \n {children}\n \n )\n}\n\nTbxTooltip.propTypes = {\n children: PropTypes.node.isRequired,\n title: PropTypes.oneOfType([PropTypes.string, PropTypes.node]).isRequired,\n arrow: PropTypes.bool,\n disableFocusListener: PropTypes.bool,\n disableTouchListener: PropTypes.bool,\n followCursor: PropTypes.bool,\n leaveDelay: PropTypes.number,\n}\n\nexport default TbxTooltip\n","import { useRef, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { ArrowDropDown as ArrowDropDownIcon } from '@mui/icons-material'\nimport {\n Box,\n Button,\n ButtonGroup as MuiButtonGroup,\n ClickAwayListener,\n Grow,\n MenuItem,\n MenuList,\n Paper,\n Popper,\n} from '@mui/material'\nimport PropTypes from 'prop-types'\n\nimport TbxTooltip from '../TbxTooltip/TbxTooltip'\n\nconst classes = {\n popper: {\n zIndex: 1001,\n position: 'absolute',\n top: 0,\n left: 0,\n transform: 'translate3d(140px, 50px, 0px)',\n willChange: 'transform',\n },\n dropdownArrow: {\n marginRight: 0.5,\n },\n}\n\nconst ButtonGroup = ({ className, color, defaultSelectedIndex, disabled, options, splitButton, variant }) => {\n const { t } = useTranslation()\n\n const quoteButtonGroupAnchorRef = useRef()\n const [selectedIndex, setSelectedIndex] = useState(defaultSelectedIndex ? defaultSelectedIndex : 0)\n\n const [menuOpen, setMenuOpen] = useState(false)\n\n const handleToggleMenu = () => {\n setMenuOpen((prevOpen) => !prevOpen)\n }\n\n const handleButtonEvent = async (index) => {\n if (!splitButton && index === 0) {\n return handleToggleMenu()\n }\n\n if (options[index]) {\n options[index].handler()\n }\n }\n\n const handleButtonClick = () => {\n handleButtonEvent(selectedIndex)\n }\n\n const handleMenuItemClick = (event, index) => {\n if (splitButton) {\n setSelectedIndex(index)\n }\n\n setMenuOpen(false)\n handleButtonEvent(index)\n }\n\n const handleCloseButton = (event) => {\n if (quoteButtonGroupAnchorRef.current && quoteButtonGroupAnchorRef.current.contains(event.target)) {\n return\n }\n setMenuOpen(false)\n }\n\n return options.length > 1 ? (\n
\n \n \n {options[selectedIndex].label}\n {!splitButton ? : null}\n \n {splitButton ? (\n \n \n \n ) : null}\n \n \n {({ TransitionProps, placement }) => (\n \n \n \n \n {splitButton\n ? options.map((option, index) => {\n const { id, tagAttrs } = option\n const title = !tagAttrs.disabled\n ? ''\n : id === 'addFromPartLibrary'\n ? t('Select a customer to enable Part Library')\n : id === 'sendToPartLibrary'\n ? t('Disabled until the order is confirmed')\n : ''\n\n return (\n \n \n {\n handleMenuItemClick(event, index)\n if (option.id === 'addFromPartLibrary') {\n setSelectedIndex(0)\n }\n }}\n {...option.tagAttrs}\n >\n {option.label}\n \n \n \n )\n })\n : options.slice(1).map((option, index) => {\n const { id, label, tagAttrs } = option\n const title = !tagAttrs.disabled\n ? ''\n : id === 'sendToPartLibrary'\n ? t('Disabled until the order is confirmed')\n : ''\n return (\n \n \n handleMenuItemClick(event, index + 1)}\n {...tagAttrs}\n >\n {label}\n \n \n \n )\n })}\n \n \n \n \n )}\n \n
\n ) : (\n \n {options[0]?.label}\n \n )\n}\n\nButtonGroup.propTypes = {\n className: PropTypes.string,\n color: PropTypes.string,\n defaultSelectedIndex: PropTypes.number,\n disabled: PropTypes.bool,\n hasPartLibraryEntry: PropTypes.bool,\n options: PropTypes.arrayOf(\n PropTypes.shape({\n handler: PropTypes.func,\n label: PropTypes.string,\n tagAttrs: PropTypes.object,\n })\n ),\n quoteCustomer: PropTypes.object,\n splitButton: PropTypes.bool,\n variant: PropTypes.string,\n}\n\nexport default ButtonGroup\n","import { useRef, useState } from 'react'\nimport { ArrowDropDown as ArrowDropDownIcon } from '@mui/icons-material'\nimport {\n Box,\n Button,\n ButtonGroup as MuiButtonGroup,\n ClickAwayListener,\n Grow,\n MenuItem,\n MenuList,\n Paper,\n Popper,\n} from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n popper: {\n zIndex: 1001,\n position: 'absolute',\n top: 0,\n left: 0,\n transform: 'translate3d(140px, 50px, 0px)',\n willChange: 'transform',\n },\n dropdownArrow: {\n marginRight: 0.5,\n },\n}\nconst ButtonGroupOLD = ({\n buttonLabels,\n buttonOnClickHandlers,\n className,\n color,\n defaultSelectedIndex,\n disabled,\n projectCustomer,\n splitButton,\n variant,\n}) => {\n const quoteButtonGroupAnchorRef = useRef()\n\n const [selectedIndex, setSelectedIndex] = useState(defaultSelectedIndex ? defaultSelectedIndex : 0)\n\n const [menuOpen, setMenuOpen] = useState(false)\n\n const handleToggleMenu = () => {\n setMenuOpen((prevOpen) => !prevOpen)\n }\n\n const handleButtonEvent = async (index) => {\n if (!splitButton & (index === 0)) {\n setMenuOpen(true)\n }\n\n if (buttonOnClickHandlers[index]) {\n buttonOnClickHandlers[index]()\n }\n }\n\n const handleButtonClick = () => {\n handleButtonEvent(selectedIndex)\n }\n\n const handleMenuItemClick = (event, index) => {\n if (splitButton) {\n setSelectedIndex(index)\n }\n setMenuOpen(false)\n\n handleButtonEvent(index)\n }\n\n const handleCloseButton = (event) => {\n if (quoteButtonGroupAnchorRef.current && quoteButtonGroupAnchorRef.current.contains(event.target)) {\n return\n }\n setMenuOpen(false)\n }\n\n return buttonLabels.length > 1 ? (\n
\n \n \n {buttonLabels[selectedIndex]}\n {!splitButton ? : null}\n \n {splitButton ? (\n \n \n \n ) : null}\n \n \n {({ TransitionProps, placement }) => (\n \n \n \n \n {\n splitButton\n ? buttonLabels.map((option, index) => (\n handleMenuItemClick(event, index)}\n >\n {option}\n \n ))\n : // if (\n // option === t('Add from Part Library') &&\n // !projectCustomer\n // ) {\n // return (\n // \n //
\n // \n // {option}\n //
\n // \n // )\n // } else {\n // return (\n // \n // handleMenuItemClick(event, index)\n // }\n // >\n // {option}\n // )\n // }\n buttonLabels.slice(1).map((option, index) => (\n handleMenuItemClick(event, index + 1)}\n >\n {option}\n \n ))\n // if (\n // option === t('Add from Part Library') &&\n // !projectCustomer\n // ) {\n // return (\n // \n //
\n // \n // {option}\n //
\n // \n // )\n // } else {\n // return (\n // \n // handleMenuItemClick(event, index + 1)\n // }\n // >\n // {option}\n // )\n // }\n }\n
\n
\n
\n \n )}\n \n
\n ) : (\n \n )\n}\n\nButtonGroupOLD.propTypes = {\n buttonLabels: PropTypes.array,\n buttonOnClickHandlers: PropTypes.array,\n className: PropTypes.string,\n color: PropTypes.string,\n defaultSelectedIndex: PropTypes.number,\n disabled: PropTypes.bool,\n projectCustomer: PropTypes.object,\n splitButton: PropTypes.bool,\n variant: PropTypes.string,\n}\n\nexport default ButtonGroupOLD\n","import { Box } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n dashedContainer: (alignItems) => {\n let itemAlignment = 'center'\n if (alignItems === 'top') {\n itemAlignment = 'flex-start'\n } else if (alignItems === 'bottom') {\n itemAlignment = 'flex-end'\n }\n\n return {\n border: (theme) => `dashed 2px ${theme.palette.text.primary}`,\n width: '100%',\n height: '100%',\n display: 'flex',\n alignItems: itemAlignment,\n justifyContent: 'center',\n padding: 2,\n boxSizing: 'border-box',\n }\n },\n}\n\nconst DashedBorderContainer = ({ alignItems = 'center', children, sx }) => {\n return {children}\n}\n\nDashedBorderContainer.propTypes = {\n children: PropTypes.node.isRequired,\n alignItems: PropTypes.oneOf(['top', 'bottom', 'center']),\n sx: PropTypes.oneOfType([\n PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.func, PropTypes.object, PropTypes.bool])),\n PropTypes.func,\n PropTypes.object,\n ]),\n}\n\nexport default DashedBorderContainer\n","import 'dayjs/locale/it'\nimport 'dayjs/locale/fr'\nimport 'dayjs/locale/de'\nimport 'dayjs/locale/en-gb'\nimport 'dayjs/locale/en-au'\nimport 'dayjs/locale/es'\nimport 'dayjs/locale/pt'\n\nimport { useMemo } from 'react'\nimport dayjs from 'dayjs'\nimport timezone from 'dayjs/plugin/timezone'\nimport utc from 'dayjs/plugin/utc'\nimport PropTypes from 'prop-types'\n\nimport { getSupportedLocale } from '@/common/utils'\n\ndayjs.extend(utc)\ndayjs.extend(timezone)\n\nconst DateTime = ({ children, format = 'YYYY-MM-DD HH:mm:ss', locale = 'en-US', ...props }) => {\n const userTimeZone = dayjs.tz.guess()\n const supportedLocale = useMemo(() => getSupportedLocale(locale), [locale])\n\n const dateTime = dayjs.utc(children).locale(supportedLocale).tz(userTimeZone).format()\n const date = dayjs.utc(children).locale(supportedLocale).tz(userTimeZone).format(format)\n\n return (\n \n {date}\n \n )\n}\n\nDateTime.propTypes = {\n children: PropTypes.node,\n format: PropTypes.string,\n locale: PropTypes.string,\n}\n\nexport default DateTime\n","import SvgIcon from '@mui/material/SvgIcon'\n\n// https://materialdesignicons.com/icon/window-close\n\nconst CrossIcon = (props) => {\n return (\n \n \n \n )\n}\n\nexport default CrossIcon\n","import MuiFab from '@mui/material/Fab'\nimport MuiIconButton from '@mui/material/IconButton'\nimport PropTypes from 'prop-types'\n\nconst IconButton = ({ className, classes, color = 'default', disabled, size, type, ...props }) => {\n return type === 'floating' ? (\n \n ) : (\n \n )\n}\n\nIconButton.propTypes = {\n children: PropTypes.node,\n classes: PropTypes.object,\n className: PropTypes.object,\n color: PropTypes.oneOf(['default', 'inherit', 'primary', 'secondary']),\n disabled: PropTypes.bool,\n size: PropTypes.oneOf(['large', 'medium', 'small']),\n type: PropTypes.oneOf(['default', 'floating']),\n onClick: PropTypes.func,\n}\n\nexport default IconButton\n","import { Box, Dialog as MuiDialog } from '@mui/material'\nimport DialogActions from '@mui/material/DialogActions'\nimport DialogContent from '@mui/material/DialogContent'\nimport DialogTitle from '@mui/material/DialogTitle'\nimport { alpha } from '@mui/material/styles'\nimport PropTypes from 'prop-types'\n\nimport CrossIcon from '../../icons/CrossIcon/CrossIcon'\nimport IconButton from '../IconButton/IconButton'\n\nconst classes = {\n titleContainer: {\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'space-between',\n },\n closeIcon: {\n marginRight: '-12px',\n },\n actions: {\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n my: 2,\n },\n actionsStart: {\n justifyContent: 'flex-start',\n },\n actionsEnd: {\n justifyContent: 'flex-end',\n },\n dialogBackdrop: {\n backgroundColor: (theme) => alpha(theme.palette.text.primary, 0.9),\n },\n}\n\nconst Dialog = ({\n actions = [],\n actionsPosition = 'center',\n children,\n className,\n contentClassName,\n fullWidth,\n maxWidth,\n onClose,\n open = false,\n showCloseButton = true,\n title,\n ...rest\n}) => {\n const dialogActionsStyle = Object.assign(\n {},\n classes.actions,\n actionsPosition === 'start' ? classes.actionsStart : actionsPosition === 'end' ? classes.actionsEnd : {}\n )\n\n return (\n \n \n \n {title}\n {showCloseButton ? (\n \n \n \n ) : null}\n \n \n {children}\n {actions && actions.length > 0 ? (\n \n {actions.map((action) => {\n return action\n })}\n \n ) : null}\n \n )\n}\n\nDialog.propTypes = {\n actions: PropTypes.arrayOf(PropTypes.node),\n actionsPosition: PropTypes.oneOf(['end', 'start', 'center']),\n children: PropTypes.node,\n className: PropTypes.object,\n contentClassName: PropTypes.object,\n fullWidth: PropTypes.bool,\n maxWidth: PropTypes.oneOf(['xs', 'sm', 'md', 'lg', 'xl', false]),\n open: PropTypes.bool,\n showCloseButton: PropTypes.bool,\n title: PropTypes.string,\n onClose: PropTypes.func,\n}\n\nexport default Dialog\n","import MuiDivider from '@mui/material/Divider'\nimport PropTypes from 'prop-types'\n\nconst Divider = ({\n absolute = false,\n flexItem = false,\n light = false,\n orientation = 'horizontal',\n variant = 'fullWidth',\n ...props\n}) => {\n return (\n \n )\n}\n\nDivider.propTypes = {\n absolute: PropTypes.bool,\n classes: PropTypes.object,\n flexItem: PropTypes.bool,\n light: PropTypes.bool,\n orientation: PropTypes.oneOf(['horizontal', 'vertical']),\n variant: PropTypes.oneOf(['fullWidth', 'inset', 'middle']),\n}\n\nexport default Divider\n","import { useTranslation } from 'react-i18next'\nimport { Alert, Snackbar } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst ErrorSnackbar = ({ autoHideDuration, children, onClose, open }) => {\n const { t } = useTranslation()\n\n const defaultAutoHideDuration = 7500\n const alertContent = children ?? t('$t(An error occurred).')\n const showtime = autoHideDuration || autoHideDuration === 0 ? autoHideDuration : defaultAutoHideDuration\n\n return (\n \n \n {alertContent}\n \n \n )\n}\n\nErrorSnackbar.propTypes = {\n open: PropTypes.bool.isRequired,\n onClose: PropTypes.func.isRequired,\n autoHideDuration: PropTypes.number,\n children: PropTypes.node,\n}\n\nexport default ErrorSnackbar\n","import { forwardRef, useCallback } from 'react'\nimport { useDropzone } from 'react-dropzone'\nimport { useTranslation } from 'react-i18next'\nimport { Box } from '@mui/material'\nimport Typography from '@mui/material/Typography'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n dragContainer: {\n position: 'fixed',\n top: 0,\n left: 0,\n right: 0,\n bottom: 0,\n backgroundColor: 'hsla(0, 0%, 0%, 0.5)',\n width: '100%',\n height: '100%',\n zIndex: 20,\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'center',\n justifyContent: 'center',\n },\n dragInstruction: {\n color: 'background.paper',\n },\n uploaderContainer: {\n width: '100%',\n height: '100%',\n },\n}\n\nconst FileUploader = forwardRef(\n (\n {\n acceptedFileTypes,\n children,\n dragActiveText,\n maxSize,\n onFilesAccepted,\n onFilesRejected,\n singleFileUpload = false,\n },\n ref\n ) => {\n const { t } = useTranslation()\n\n // Dropzone configuration\n const onDropAccepted = useCallback(\n (acceptedFiles) => {\n if (typeof onFilesAccepted === 'function') {\n onFilesAccepted(acceptedFiles)\n }\n },\n [onFilesAccepted]\n )\n\n const onDropRejected = useCallback(\n (acceptedFiles) => {\n if (typeof onFilesRejected === 'function') {\n onFilesRejected(acceptedFiles)\n }\n },\n [onFilesRejected]\n )\n\n const { getInputProps, getRootProps, isDragActive } = useDropzone({\n onDropAccepted,\n onDropRejected,\n accept: acceptedFileTypes,\n noDragEventsBubbling: true,\n noClick: true,\n noKeyboard: true,\n multiple: !singleFileUpload,\n maxSize,\n })\n\n return (\n \n \n {isDragActive ? (\n \n \n {dragActiveText ?? t('Drop files here')}\n \n {maxSize ? (\n \n {t('Maximum file size {{maxSize}} MB', {\n maxSize: Math.floor(maxSize / 1e6),\n })}\n \n ) : null}\n \n ) : null}\n {children}\n \n )\n }\n)\n\nFileUploader.displayName = 'FileUploader'\n\nFileUploader.propTypes = {\n acceptedFileTypes: PropTypes.oneOfType([PropTypes.array, PropTypes.object]),\n children: PropTypes.node,\n dragActiveText: PropTypes.string,\n maxSize: PropTypes.number,\n singleFileUpload: PropTypes.bool,\n onFilesAccepted: PropTypes.func,\n onFilesRejected: PropTypes.func,\n}\n\nexport default FileUploader\n","import { forwardRef } from 'react'\nimport MuiMenuItem from '@mui/material/MenuItem'\nimport PropTypes from 'prop-types'\n\nconst SelectOption = forwardRef(function MenuItem(\n {\n children,\n className,\n classes,\n dense,\n disableGutters = false,\n disabled,\n role,\n selected,\n tabIndex,\n value,\n ...other\n },\n ref\n) {\n return (\n \n {children}\n \n )\n})\n\nSelectOption.propTypes = {\n children: PropTypes.node,\n classes: PropTypes.object,\n className: PropTypes.object,\n dense: PropTypes.bool,\n disabled: PropTypes.bool,\n disableGutters: PropTypes.bool,\n role: PropTypes.string,\n selected: PropTypes.bool,\n tabIndex: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n value: PropTypes.any,\n}\n\nexport default SelectOption\n","import {\n Box,\n InputAdornment as MuiInputAdornment,\n InputLabel as MuiInputLabel,\n TextField as MuiTextField,\n} from '@mui/material'\nimport PropTypes from 'prop-types'\n\nimport SelectOption from '../SelectOption/SelectOption'\n\nconst styleClasses = {\n container: (props) => ({\n display: 'flex',\n alignItems: props.labelPosition === 'start' || props.labelPosition === 'end' ? 'center' : 'normal',\n flexDirection: props.labelPosition === 'top' || props.labelPosition === 'bottom' ? 'column' : 'row',\n justifyContent: props.justifycontent,\n }),\n inputLabelTopOrBottom: {\n mx: 0,\n my: 1,\n },\n labelRoot: {\n margin: 1,\n overflow: 'inherit',\n whiteSpace: 'inherit',\n },\n denseInputAdornment: {\n marginRight: '4px',\n marginLeft: '4px',\n },\n selectPlaceholder: {\n color: (theme) => theme.palette.text.disabled,\n },\n input: {},\n adornment: {},\n inputDisabled: {},\n}\n\nconst Input = ({\n FormHelperTextProps,\n InputProps,\n adornmentDense,\n adornmentEnd,\n adornmentStart,\n autoComplete,\n autoFocus,\n children,\n className,\n classes,\n color,\n defaultValue,\n disabled,\n error,\n fullWidth,\n helperText,\n hideLabel,\n inputProps,\n inputRef,\n justifycontent = 'space-between',\n label,\n labelPosition = 'start',\n labelProps,\n labelText,\n multiline,\n name,\n onBlur,\n onChange,\n onEnter,\n onEscape,\n onFocus,\n onKeyDown,\n onKeyUp,\n placeholder,\n readOnly,\n required,\n rows,\n rowsMax,\n textAlign = 'right',\n type = 'text',\n value,\n variant,\n ...rest\n}) => {\n const inputRootClass =\n labelPosition === 'top' || labelPosition === 'bottom' ? styleClasses.inputLabelTopOrBottom : {}\n\n const handleKeyUp = (event) => {\n switch (event.which) {\n case 13:\n if (typeof onEnter === 'function') {\n onEnter(event)\n }\n break\n case 27:\n if (typeof onEscape === 'function') {\n onEscape(event)\n }\n break\n default:\n break\n }\n\n if (typeof onKeyUp === 'function') {\n onKeyUp(event)\n }\n }\n\n const getTextfieldInput = (isSelect) => {\n return (\n \n {adornmentEnd}\n \n ),\n startAdornment: adornmentStart && (\n \n {adornmentStart}\n \n ),\n ...InputProps,\n inputProps: { ...inputProps, style: { textAlign: textAlign } },\n readOnly: readOnly,\n }}\n inputRef={inputRef}\n label={label}\n maxRows={rowsMax}\n multiline={multiline}\n name={name}\n placeholder={placeholder ? placeholder : isSelect ? '-' : null}\n required={required}\n rows={rows}\n select={isSelect}\n SelectProps={inputProps}\n sx={inputRootClass}\n type={type}\n value={isSelect && !value ? '-1' : value}\n variant={variant}\n onBlur={onBlur}\n onChange={onChange}\n onFocus={onFocus}\n onKeyDown={onKeyDown}\n onKeyUp={handleKeyUp}\n >\n {isSelect && !value && value !== 0 ? (\n \n {placeholder ?? '-'}\n \n ) : null}\n {children}\n \n )\n }\n\n const renderInput = () => {\n switch (type) {\n case 'select':\n return getTextfieldInput(true)\n default:\n return getTextfieldInput(false)\n }\n }\n\n const renderLabel = () => {\n return (\n labelText &&\n !hideLabel && (\n \n {labelText}\n \n )\n )\n }\n\n const containerClass = Object.assign({}, styleClasses.container({ labelPosition, justifycontent }), className)\n\n return (\n \n {!labelPosition || labelPosition === 'top' || labelPosition === 'start' ? renderLabel() : null}\n {renderInput()}\n {labelPosition === 'bottom' || labelPosition === 'end' ? renderLabel() : null}\n \n )\n}\n\nexport const inputPropTypes = {\n labelPosition: PropTypes.oneOf(['start', 'end', 'bottom', 'top', '']).isRequired,\n textAlign: PropTypes.oneOf(['left', 'center', 'right']).isRequired,\n type: PropTypes.oneOf(['text', 'number', 'radio', 'checkbox', 'select']).isRequired,\n adornmentDense: PropTypes.bool,\n adornmentEnd: PropTypes.node,\n adornmentStart: PropTypes.node,\n autoComplete: PropTypes.string,\n autoFocus: PropTypes.bool,\n children: PropTypes.node,\n classes: PropTypes.object,\n className: PropTypes.object,\n color: PropTypes.oneOf(['primary', 'secondary']),\n defaultValue: PropTypes.any,\n disabled: PropTypes.bool,\n error: PropTypes.oneOfType([PropTypes.bool, PropTypes.string]),\n FormHelperTextProps: PropTypes.object,\n fullWidth: PropTypes.bool,\n helperText: PropTypes.node,\n hideLabel: PropTypes.bool,\n inputProps: PropTypes.object,\n InputProps: PropTypes.object,\n inputRef: PropTypes.any,\n justifycontent: PropTypes.oneOf(['start', 'end', 'center', 'space-between', 'space-around']),\n label: PropTypes.node,\n labelProps: PropTypes.object,\n labelText: PropTypes.string,\n margin: PropTypes.oneOf(['dense', 'none', 'normal']),\n multiline: PropTypes.bool,\n name: PropTypes.string,\n placeholder: PropTypes.string,\n readOnly: PropTypes.bool,\n required: PropTypes.bool,\n rows: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n rowsMax: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n style: PropTypes.object,\n value: PropTypes.any,\n variant: PropTypes.oneOf(['filled', 'outlined', 'standard']),\n onBlur: PropTypes.func,\n onChange: PropTypes.func,\n onEnter: PropTypes.func,\n onEscape: PropTypes.func,\n onFocus: PropTypes.func,\n onKeyDown: PropTypes.func,\n onKeyUp: PropTypes.func,\n}\n\nInput.propTypes = inputPropTypes\n\nexport default Input\n","import { Box } from '@mui/material'\nimport Typography from '@mui/material/Typography'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n container: {\n width: '100%',\n },\n title: {\n color: 'text.primary',\n mr: 1,\n my: 1,\n },\n}\n\nconst InputGroup = ({ children, className, title }) => {\n const containerStyle = Object.assign({}, classes.container, className)\n\n return (\n \n \n {title}\n \n {children}\n \n )\n}\n\nInputGroup.propTypes = {\n children: PropTypes.node,\n title: PropTypes.string,\n}\n\nexport default InputGroup\n","import MuiFormHelperText from '@mui/material/FormHelperText'\nimport PropTypes from 'prop-types'\n\nconst InputHelperText = ({\n children,\n className,\n classes,\n disabled,\n error,\n filled,\n focused,\n margin,\n required,\n variant,\n}) => {\n return (\n \n {children}\n \n )\n}\n\nInputHelperText.propTypes = {\n children: PropTypes.node,\n classes: PropTypes.object,\n className: PropTypes.string,\n disabled: PropTypes.bool,\n error: PropTypes.bool,\n filled: PropTypes.bool,\n focused: PropTypes.bool,\n margin: PropTypes.oneOf(['dense']),\n required: PropTypes.bool,\n variant: PropTypes.oneOf(['filled', 'outlined', 'standard']),\n}\n\nexport default InputHelperText\n","import MuiLinearProgress from '@mui/material/LinearProgress'\nimport PropTypes from 'prop-types'\n\nconst LinearProgress = ({ className, classes, color, value, valueBuffer, variant }) => {\n return (\n \n )\n}\n\nLinearProgress.propTypes = {\n classes: PropTypes.object,\n className: PropTypes.object,\n color: PropTypes.oneOf(['primary', 'secondary']),\n value: PropTypes.number,\n valueBuffer: PropTypes.number,\n variant: PropTypes.oneOf(['buffer', 'determinate', 'indeterminate', 'query']),\n}\n\nexport default LinearProgress\n","import MuiLink from '@mui/material/Link'\nimport PropTypes from 'prop-types'\n\nconst Link = (props) => {\n return \n}\n\nLink.propTypes = {\n classes: PropTypes.object,\n color: PropTypes.oneOf(['initial', 'inherit', 'primary', 'secondary', 'textPrimary', 'textSecondary', 'error']),\n TypographyClasses: PropTypes.object,\n underline: PropTypes.oneOf(['none', 'hover', 'always']),\n}\n\nexport default Link\n","import { useTranslation } from 'react-i18next'\nimport { Box, CircularProgress, Typography } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n wrapper: {\n position: 'relative',\n width: '100%',\n height: 'calc(100vh - 64px)',\n overflow: 'hidden',\n },\n transparentOverlay: {\n position: 'absolute',\n top: 0,\n left: 0,\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n zIndex: (theme) => theme.zIndex.appBar - 1,\n width: '100%',\n height: '100%',\n },\n overlay: {\n position: 'absolute',\n top: 0,\n left: 0,\n background: 'rgba(0,0,0,0.2)',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n zIndex: (theme) => theme.zIndex.appBar - 1,\n width: '100%',\n height: '100%',\n transition: 'opacity 500ms ease-in',\n },\n loadingComponent: {\n width: '100%',\n height: '100%',\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'center',\n justifyContent: 'center',\n },\n loadingText: {\n marginTop: 2,\n },\n}\n\nconst LoadingOverlay = ({\n active = true,\n children,\n className,\n labelText = 'Loading',\n loadingComponent = ,\n showLoadingComponent = true,\n showLoadingLabel = true,\n transparent,\n}) => {\n const { t } = useTranslation()\n\n const containerStyle = Object.assign({}, classes.wrapper, className)\n return (\n \n {active ? (\n \n {showLoadingComponent ? (\n \n {loadingComponent}\n {showLoadingLabel ? (\n \n {t(labelText)}...\n \n ) : null}\n \n ) : null}\n \n ) : null}\n {children}\n \n )\n}\n\nLoadingOverlay.propTypes = {\n active: PropTypes.bool,\n children: PropTypes.node,\n className: PropTypes.object,\n labelText: PropTypes.string,\n loadingComponent: PropTypes.node,\n showLoadingComponent: PropTypes.bool,\n showLoadingLabel: PropTypes.bool,\n transparent: PropTypes.bool,\n}\n\nexport default LoadingOverlay\n","import { Children, cloneElement, useEffect, useRef, useState } from 'react'\nimport { Box } from '@mui/material'\nimport MuiButton from '@mui/material/Button'\nimport MuiClickAwayListener from '@mui/material/ClickAwayListener'\nimport MuiGrow from '@mui/material/Grow'\nimport MuiMenuList from '@mui/material/MenuList'\nimport MuiPaper from '@mui/material/Paper'\nimport MuiPopper from '@mui/material/Popper'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n root: {\n display: 'flex',\n zIndex: (theme) => theme.zIndex.appBar - 1,\n },\n paper: {\n marginRight: 2,\n },\n}\n\nconst Menu = ({ children, disabled, open = false, title }) => {\n const [menuOpen, setMenuOpen] = useState(open)\n const anchorRef = useRef(null)\n\n useEffect(() => {\n setMenuOpen(open)\n }, [open])\n\n const handleToggle = () => {\n setMenuOpen((prevOpen) => !prevOpen)\n }\n\n const handleClose = (event) => {\n if (anchorRef.current && anchorRef.current.contains(event.target)) {\n return\n }\n\n setMenuOpen(false)\n }\n\n const handleChildOnClick = (event, childClickHandler) => {\n if (typeof childClickHandler === 'function') {\n childClickHandler(event)\n }\n\n handleClose(event)\n }\n\n function handleListKeyDown(event) {\n if (event.key === 'Tab') {\n event.preventDefault()\n setMenuOpen(false)\n }\n }\n\n // return focus to the button when we transitioned from !menuOpen -> menuOpen\n const prevOpen = useRef(menuOpen)\n useEffect(() => {\n if (prevOpen.current === true && menuOpen === false) {\n anchorRef.current.focus()\n }\n\n prevOpen.current = menuOpen\n }, [menuOpen])\n\n return (\n \n \n {title}\n \n \n {({ TransitionProps }) => (\n \n \n \n \n {/* {children} */}\n {Children.map(children, (child) => {\n const childClickHandler = child.props.onClick\n return cloneElement(child, {\n onClick: (event) => handleChildOnClick(event, childClickHandler),\n })\n })}\n \n \n \n \n )}\n \n \n )\n}\n\nMenu.propTypes = {\n title: PropTypes.string.isRequired,\n children: PropTypes.node,\n disabled: PropTypes.bool,\n open: PropTypes.bool,\n}\n\nexport default Menu\n","import MuiMenuItem from '@mui/material/MenuItem'\nimport PropTypes from 'prop-types'\n\nconst MenuItem = ({ children, disabled = false, onClick }) => {\n return (\n \n {children}\n \n )\n}\n\nMenuItem.propTypes = {\n children: PropTypes.node,\n disabled: PropTypes.bool,\n onClick: PropTypes.func,\n}\n\nexport default MenuItem\n","import { useNavigate } from 'react-router-dom'\nimport { AppBar, Box, Button, Toolbar, Typography } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n root: {\n flexGrow: 0,\n },\n toolbar: {\n backgroundColor: 'white',\n alignItems: 'center',\n },\n menuButton: {\n marginRight: 2,\n },\n title: {\n flexGrow: 1,\n color: 'primary.main',\n fontWeight: 600,\n },\n actionContainer: {\n display: 'flex',\n alignItems: 'center',\n },\n actionText: {\n color: '#FD0D1B',\n fontWeight: 600,\n },\n actionButton: {\n my: 0,\n mx: 2,\n },\n}\n\nconst NotificationBar = ({\n actionButtonText,\n actionButtonUrl,\n external = false,\n hideButton,\n leftText,\n reverse = false,\n rightText,\n}) => {\n const navigate = useNavigate()\n\n const handleActionButtonClicked = () => {\n if (external || actionButtonUrl.startsWith('http')) {\n window.open(actionButtonUrl)\n } else {\n navigate(actionButtonUrl)\n }\n }\n\n const getLeftText = () => {\n return (\n \n {leftText}\n \n )\n }\n\n const getRightText = () => {\n return (\n \n \n {rightText}\n \n {actionButtonText && !hideButton ? \n {actionButtonText}\n : null}\n \n )\n }\n\n return (\n \n \n {reverse ? getRightText() : getLeftText()}\n {reverse ? getLeftText() : getRightText()}\n \n \n )\n}\n\nNotificationBar.propTypes = {\n actionButtonText: PropTypes.string,\n actionButtonUrl: PropTypes.string,\n external: PropTypes.bool,\n hideButton: PropTypes.bool,\n leftText: PropTypes.node,\n reverse: PropTypes.bool,\n rightText: PropTypes.node,\n}\n\nexport default NotificationBar\n","import { useEffect, useState } from 'react'\nimport { Box } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nimport { useToolBoxTreatments } from '@/common/hooks'\nimport { getProductDetails } from '@/common/utils/GeneralUtils/GeneralUtils'\n\nconst classes = {\n wrapper: ({ wrapperHeight, wrapperWidth }) => ({\n width: wrapperWidth,\n height: wrapperHeight,\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n }),\n img: {\n height: '100%',\n width: '100%',\n objectFit: 'contain',\n objectPosition: 'center',\n },\n}\n\nconst OrganisationLogo = ({ organisationId, organisationLogoUri, sx, wrapperHeight = 100, wrapperWidth = 100 }) => {\n const { previewResellerName } = useToolBoxTreatments()\n\n const productDetails = getProductDetails(previewResellerName)\n const defaultLogoUrl = `/${productDetails.logo}`\n\n const [logo, setLogo] = useState(null)\n const [isMounted, setIsMounted] = useState(true)\n\n const loadDefaultLogo = async () => {\n const logoResponse = await fetch(defaultLogoUrl)\n const blobResponse = logoResponse && (await logoResponse.blob())\n if (blobResponse?.size > 0) {\n isMounted && setLogo(URL.createObjectURL(blobResponse))\n }\n }\n\n useEffect(() => {\n setIsMounted(true)\n\n if (organisationId && organisationLogoUri) {\n setLogo(organisationLogoUri)\n } else {\n loadDefaultLogo()\n }\n\n return () => {\n setIsMounted(false)\n }\n }, [organisationId, organisationLogoUri])\n\n return (\n \n \n \n )\n}\n\nOrganisationLogo.propTypes = {\n organisationId: PropTypes.string.isRequired,\n asBase64: PropTypes.bool,\n organisationLogoUri: PropTypes.string,\n sx: PropTypes.oneOfType([\n PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.func, PropTypes.object, PropTypes.bool])),\n PropTypes.func,\n PropTypes.object,\n ]),\n wrapperHeight: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n wrapperWidth: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n}\n\nexport default OrganisationLogo\n","import { useEffect, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { useSelector } from 'react-redux'\nimport { FormControl, InputLabel, MenuItem, Select } from '@mui/material'\nimport PropTypes from 'prop-types'\n\nimport { selectCurrentUserOrganisations } from '@/app/slices/appSlice'\n\nconst OrganisationSelect = ({\n disabled = false,\n inputProps,\n label = 'Organisation',\n onChange,\n onClick,\n size = 'medium',\n sx,\n value = '',\n variant = 'standard',\n}) => {\n const { t } = useTranslation()\n\n const currentUserOrganisations = useSelector(selectCurrentUserOrganisations)\n\n const [selectedValue, setSelectedValue] = useState('')\n\n const handleChange = (event) => {\n const selectedOrganisationId = event.target.value\n setSelectedValue(selectedOrganisationId)\n\n if (typeof onChange === 'function') {\n onChange(selectedOrganisationId)\n }\n }\n\n useEffect(() => {\n setSelectedValue(value)\n }, [value])\n\n return (\n \n {t(label)}\n \n {currentUserOrganisations?.map((org) => (\n \n {org.name}\n \n ))}\n \n \n )\n}\n\nOrganisationSelect.propTypes = {\n disabled: PropTypes.bool,\n inputProps: PropTypes.object,\n label: PropTypes.string,\n size: PropTypes.oneOf(['small', 'medium']),\n sx: PropTypes.oneOfType([\n PropTypes.arrayOf(PropTypes.oneOfType([PropTypes.func, PropTypes.object, PropTypes.bool])),\n PropTypes.func,\n PropTypes.object,\n ]),\n value: PropTypes.string,\n variant: PropTypes.oneOf(['filled', 'outlined', 'standard']),\n onChange: PropTypes.func,\n onClick: PropTypes.func,\n}\n\nexport default OrganisationSelect\n","import { Box } from '@mui/material'\nimport CircularProgress from '@mui/material/CircularProgress'\n\nconst classes = {\n progress: {\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'center',\n justifyContent: 'center',\n width: '100vw',\n height: '100vh',\n gap: 2,\n },\n}\n\nconst PageSpinner = () => {\n return (\n \n \n \n )\n}\n\nexport default PageSpinner\n","import { forwardRef } from 'react'\nimport MuiPopover from '@mui/material/Popover'\nimport { HTMLElementType } from '@mui/utils'\nimport PropTypes from 'prop-types'\n\nconst Popover = forwardRef(\n (\n {\n anchorEl,\n anchorOrigin = { horizontal: 'left', vertical: 'top' },\n anchorPosition,\n anchorReference = 'anchorEl',\n children,\n className,\n marginThreshold,\n onClose,\n open,\n transformOrigin = { horizontal: 'left', vertical: 'top' },\n ...other\n },\n ref\n ) => {\n return (\n \n {children}\n \n )\n }\n)\n\nPopover.displayName = 'Popover'\n\nPopover.propTypes = {\n open: PropTypes.bool.isRequired,\n anchorEl: PropTypes.oneOfType([HTMLElementType, PropTypes.func]),\n anchorOrigin: PropTypes.shape({\n horizontal: PropTypes.oneOfType([PropTypes.oneOf(['center', 'left', 'right']), PropTypes.number]).isRequired,\n vertical: PropTypes.oneOfType([PropTypes.oneOf(['bottom', 'center', 'top']), PropTypes.number]).isRequired,\n }),\n anchorPosition: PropTypes.shape({\n left: PropTypes.number.isRequired,\n top: PropTypes.number.isRequired,\n }),\n anchorReference: PropTypes.oneOf(['anchorEl', 'anchorPosition', 'none']),\n children: PropTypes.node,\n className: PropTypes.string,\n marginThreshold: PropTypes.number,\n transformOrigin: PropTypes.shape({\n horizontal: PropTypes.oneOfType([PropTypes.oneOf(['center', 'left', 'right']), PropTypes.number]).isRequired,\n vertical: PropTypes.oneOfType([PropTypes.oneOf(['bottom', 'center', 'top']), PropTypes.number]).isRequired,\n }),\n onClose: PropTypes.func,\n}\n\nexport default Popover\n","import PropTypes from 'prop-types'\n\nimport { getFormattedPrice } from '@/common/utils/CurrencyUtils/CurrencyUtils'\n\nconst Price = ({ amount, currencyCode, locale }) => {\n return <>{getFormattedPrice(amount, currencyCode, locale)}\n}\n\nPrice.propTypes = {\n amount: PropTypes.number,\n currencyCode: PropTypes.string,\n locale: PropTypes.string,\n}\n\nexport default Price\n","import { forwardRef } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { FiberManualRecord } from '@mui/icons-material'\nimport Box from '@mui/material/Box'\nimport Typography from '@mui/material/Typography'\nimport PropTypes from 'prop-types'\n\nimport { WEB_STORE_QUOTE_STATUS } from '@/common/utils/Constants/Constants'\n\nconst CommonProjectStatusStyles = {\n fontWeight: '500',\n fontSize: '0.75rem',\n}\n\nconst classes = {\n text: {\n color: 'text.primary',\n },\n NotCalculated: {\n color: '#F78F1E',\n ...CommonProjectStatusStyles,\n },\n Calculated: {\n color: '#007DAB',\n ...CommonProjectStatusStyles,\n },\n Ordered: {\n color: '#00BE35',\n ...CommonProjectStatusStyles,\n },\n Invoiced: {\n color: '#1EBEF7',\n ...CommonProjectStatusStyles,\n },\n Voided: {\n color: '#C62828',\n ...CommonProjectStatusStyles,\n },\n Issued: {\n color: '#0288D1',\n ...CommonProjectStatusStyles,\n },\n Lost: {\n color: '#800080',\n ...CommonProjectStatusStyles,\n },\n Cancelled: {\n color: '#E12626',\n ...CommonProjectStatusStyles,\n },\n PendingOrderConfirmation: {\n color: '#203E96',\n ...CommonProjectStatusStyles,\n },\n Rejected: {\n color: '#130F0F',\n ...CommonProjectStatusStyles,\n },\n}\n\nconst ProjectStatusIcon = forwardRef(({ className, value }, ref) => {\n const { t } = useTranslation()\n\n let projectStatusStyle = undefined\n let label = ''\n switch (value) {\n case WEB_STORE_QUOTE_STATUS.NotCalculated:\n projectStatusStyle = classes.NotCalculated\n label = t('Not Calculated')\n break\n case WEB_STORE_QUOTE_STATUS.Calculated:\n projectStatusStyle = classes.Calculated\n label = t('Calculated')\n break\n case WEB_STORE_QUOTE_STATUS.Ordered:\n projectStatusStyle = classes.Ordered\n label = t('Ordered')\n break\n case WEB_STORE_QUOTE_STATUS.Invoiced:\n projectStatusStyle = classes.Invoiced\n label = t('Invoiced')\n break\n case WEB_STORE_QUOTE_STATUS.Voided:\n projectStatusStyle = classes.Voided\n label = t('Voided')\n break\n case WEB_STORE_QUOTE_STATUS.Lost:\n projectStatusStyle = classes.Lost\n label = t('Lost')\n break\n case WEB_STORE_QUOTE_STATUS.Cancelled:\n projectStatusStyle = classes.Cancelled\n label = t('Cancelled')\n break\n case WEB_STORE_QUOTE_STATUS.PendingOrderConfirmation:\n projectStatusStyle = classes.PendingOrderConfirmation\n label = t('Pending Order Confirmation')\n break\n case WEB_STORE_QUOTE_STATUS.Rejected:\n projectStatusStyle = classes.Rejected\n label = t('Rejected')\n break\n case WEB_STORE_QUOTE_STATUS.Issued:\n projectStatusStyle = classes.Issued\n label = t('Issued')\n break\n default:\n break\n }\n return (\n \n {value !== undefined ? (\n \n \n \n {label}\n \n \n ) : null}\n \n )\n})\n\nProjectStatusIcon.displayName = 'ProjectStatusIcon'\n\nProjectStatusIcon.propTypes = {\n className: PropTypes.object,\n value: PropTypes.oneOf([\n WEB_STORE_QUOTE_STATUS.NotCalculated,\n WEB_STORE_QUOTE_STATUS.Calculated,\n WEB_STORE_QUOTE_STATUS.Ordered,\n WEB_STORE_QUOTE_STATUS.Invoiced,\n WEB_STORE_QUOTE_STATUS.Voided,\n WEB_STORE_QUOTE_STATUS.Lost,\n WEB_STORE_QUOTE_STATUS.Cancelled,\n ]),\n}\n\nexport default ProjectStatusIcon\n","import { forwardRef } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport { FiberManualRecord } from '@mui/icons-material'\nimport Box from '@mui/material/Box'\nimport Typography from '@mui/material/Typography'\nimport PropTypes from 'prop-types'\n\nimport { GET_STATUS_LABEL, QUOTE_STATUS } from '@/common/utils/Constants/Constants'\n\nconst CommonProjectStatusStyles = {\n fontWeight: '500',\n fontSize: '0.75rem',\n}\n\nconst classes = {\n text: {\n color: 'text.primary',\n },\n NotCalculated: {\n color: '#F78F1E',\n ...CommonProjectStatusStyles,\n },\n Calculated: {\n color: '#007DAB',\n ...CommonProjectStatusStyles,\n },\n Ordered: {\n color: '#00BE35',\n ...CommonProjectStatusStyles,\n },\n Invoiced: {\n color: '#1EBEF7',\n ...CommonProjectStatusStyles,\n },\n Voided: {\n color: '#C62828',\n ...CommonProjectStatusStyles,\n },\n Issued: {\n color: '#0288D1',\n ...CommonProjectStatusStyles,\n },\n Lost: {\n color: '#800080',\n ...CommonProjectStatusStyles,\n },\n Cancelled: {\n color: '#E12626',\n ...CommonProjectStatusStyles,\n },\n PendingOrderConfirmation: {\n color: '#203E96',\n ...CommonProjectStatusStyles,\n },\n Rejected: {\n color: '#130F0F',\n ...CommonProjectStatusStyles,\n },\n}\n\nconst QuoteStatusIcon = forwardRef(({ className, value }, ref) => {\n const { t } = useTranslation()\n\n let projectStatusStyle = undefined\n let label = ''\n switch (value) {\n case QUOTE_STATUS.NotCalculated:\n projectStatusStyle = classes.NotCalculated\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Calculated:\n projectStatusStyle = classes.NotCalculated\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Ordered:\n projectStatusStyle = classes.Ordered\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Invoiced:\n projectStatusStyle = classes.Invoiced\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Voided:\n projectStatusStyle = classes.Voided\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Lost:\n projectStatusStyle = classes.Lost\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Cancelled:\n projectStatusStyle = classes.Cancelled\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.PendingOrderConfirmation:\n projectStatusStyle = classes.PendingOrderConfirmation\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Rejected:\n projectStatusStyle = classes.Rejected\n label = t(GET_STATUS_LABEL(value))\n break\n case QUOTE_STATUS.Issued:\n projectStatusStyle = classes.Issued\n label = t(GET_STATUS_LABEL(value))\n break\n default:\n break\n }\n return (\n \n {value !== undefined ? (\n \n \n \n {label}\n \n \n ) : null}\n \n )\n})\n\nQuoteStatusIcon.displayName = 'QuoteStatusIcon'\n\nQuoteStatusIcon.propTypes = {\n className: PropTypes.object,\n value: PropTypes.oneOf([\n QUOTE_STATUS.NotCalculated,\n QUOTE_STATUS.Calculated,\n QUOTE_STATUS.Ordered,\n QUOTE_STATUS.Invoiced,\n QUOTE_STATUS.Voided,\n QUOTE_STATUS.Lost,\n QUOTE_STATUS.Cancelled,\n QUOTE_STATUS.Rejected,\n QUOTE_STATUS.PendingOrderConfirmation,\n QUOTE_STATUS.Dispatched,\n QUOTE_STATUS.Issued,\n ]),\n}\n\nexport default QuoteStatusIcon\n","import React, { useEffect, useState } from 'react'\nimport { Editor } from 'react-draft-wysiwyg'\nimport { TextField } from '@mui/material'\nimport { ContentState, convertToRaw, EditorState } from 'draft-js'\nimport draftToHtml from 'draftjs-to-html'\nimport htmlToDraft from 'html-to-draftjs'\nimport PropTypes from 'prop-types'\n\nimport 'react-draft-wysiwyg/dist/react-draft-wysiwyg.css'\n\nconst RichContentField = React.forwardRef(function DraftField(props, ref) {\n const { component: Component, editorRef, handleOnChange, ...rest } = props\n\n React.useImperativeHandle(ref, () => ({\n focus: () => editorRef?.current?.focus(),\n }))\n\n return (\n \n )\n})\n\nRichContentField.propTypes = {\n component: PropTypes.any,\n editorRef: PropTypes.any,\n handleOnChange: PropTypes.func,\n}\n\nconst RichContentEditor = ({\n dataTestId,\n error,\n fullWidth,\n helperText,\n label,\n onBlur,\n onChange,\n placeholder,\n spellCheck,\n value,\n variant,\n}) => {\n const [editorState, setEditorState] = useState(EditorState.createEmpty())\n const [updated, setUpdated] = useState(false)\n const editorRef = React.useRef(null)\n\n useEffect(() => {\n if (!updated) {\n const defaultValue = value ? value : ''\n const blocksFromHtml = htmlToDraft(defaultValue)\n const contentState = ContentState.createFromBlockArray(\n blocksFromHtml.contentBlocks,\n blocksFromHtml.entityMap\n )\n const newEditorState = EditorState.createWithContent(contentState)\n setEditorState(newEditorState)\n }\n }, [value])\n\n const onEditorStateChange = (editorState) => {\n setUpdated(true)\n setEditorState(editorState)\n\n return onChange(draftToHtml(convertToRaw(editorState.getCurrentContent())))\n }\n\n return (\n \n )\n}\n\nRichContentEditor.propTypes = {\n onChange: PropTypes.func.isRequired,\n dataTestId: PropTypes.string,\n error: PropTypes.bool,\n fullWidth: PropTypes.bool,\n helperText: PropTypes.any,\n label: PropTypes.string,\n placeholder: PropTypes.string,\n spellCheck: PropTypes.bool,\n value: PropTypes.string,\n variant: PropTypes.string,\n onBlur: PropTypes.func,\n}\n\nexport default RichContentEditor\n","import { useCallback, useEffect, useState } from 'react'\nimport { Search } from '@mui/icons-material'\nimport { Box, Input } from '@mui/material'\nimport { debounce } from 'lodash'\nimport PropTypes from 'prop-types'\n\nconst classes = (theme) => ({\n search: {\n position: 'relative',\n marginLeft: 0,\n width: '100%',\n [theme.breakpoints.up('sm')]: {\n marginLeft: 0,\n width: 'auto',\n },\n },\n searchIcon: {\n width: 30,\n height: '100%',\n position: 'absolute',\n pointerEvents: 'none',\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n '& svg': {\n color: 'text.primary',\n },\n },\n underline: {\n '&:before': {\n borderBottom: `1px solid ${theme.palette.text.primary}`,\n },\n '&:hover:not(.Mui-disabled):before': {\n borderBottom: `2px solid ${theme.palette.text.primary}`,\n },\n '&:after': {\n borderBottom: `2px solid ${theme.palette.text.primary}`,\n },\n },\n inputRoot: {\n height: '30px',\n },\n inputInput: {\n paddingTop: 1,\n paddingRight: 1,\n paddingBottom: 1,\n paddingLeft: 4,\n transition: theme.transitions.create('width'),\n color: 'text.primary',\n borderBottomColor: 'text.primary',\n },\n searchInput: {\n display: 'block',\n width: '100%',\n },\n label: {\n marginBottom: '4px',\n },\n})\nconst SearchInput = ({ debounceDelay = 800, debounceInput = true, disabled = false, onChange, placeholder, value }) => {\n const [searchTerm, setSearchTerm] = useState(value)\n\n useEffect(() => {\n setSearchTerm(value)\n }, [value])\n\n const searchTermChangedHandler = useCallback(\n debounce(\n (newValue) => {\n if (typeof onChange === 'function') {\n onChange(newValue)\n }\n },\n !debounceInput ? 0 : debounceDelay\n ),\n []\n )\n\n const handleSearchTermChange = (event) => {\n setSearchTerm(event.target.value)\n searchTermChangedHandler(event.target.value)\n }\n\n return (\n \n \n \n \n \n \n \n \n )\n}\n\nSearchInput.propTypes = {\n debounceDelay: PropTypes.number,\n debounceInput: PropTypes.bool,\n disabled: PropTypes.bool,\n placeholder: PropTypes.string,\n value: PropTypes.string,\n onChange: PropTypes.func,\n}\n\nexport default SearchInput\n","import { useCallback, useEffect, useState } from 'react'\nimport { Search } from '@mui/icons-material'\nimport { TextField } from '@mui/material'\nimport InputAdornment from '@mui/material/InputAdornment'\nimport { debounce } from 'lodash'\nimport PropTypes from 'prop-types'\n\nconst SearchTextField = ({\n debounceDelay = 800,\n debounceInput = true,\n disabled = false,\n label,\n onChange,\n placeholder,\n value,\n}) => {\n const [searchTerm, setSearchTerm] = useState(value)\n\n useEffect(() => {\n setSearchTerm(value)\n }, [value])\n\n const searchTermChangedHandler = useCallback(\n debounce(\n (newValue) => {\n if (typeof onChange === 'function') {\n onChange(newValue)\n }\n },\n !debounceInput ? 0 : debounceDelay\n ),\n []\n )\n\n const handleSearchTermChange = (event) => {\n setSearchTerm(event.target.value)\n searchTermChangedHandler(event.target.value)\n }\n\n return (\n \n \n \n ),\n }}\n label={label}\n placeholder={placeholder}\n size=\"small\"\n value={searchTerm}\n variant=\"outlined\"\n onChange={handleSearchTermChange}\n />\n )\n}\n\nSearchTextField.propTypes = {\n debounceDelay: PropTypes.number,\n debounceInput: PropTypes.bool,\n disabled: PropTypes.bool,\n label: PropTypes.string,\n placeholder: PropTypes.string,\n value: PropTypes.string,\n onChange: PropTypes.func,\n}\n\nexport default SearchTextField\n","import { useRef, useState } from 'react'\nimport { ArrowDropDown } from '@mui/icons-material'\nimport Button from '@mui/material/Button'\nimport ButtonGroup from '@mui/material/ButtonGroup'\nimport ClickAwayListener from '@mui/material/ClickAwayListener'\nimport Grow from '@mui/material/Grow'\nimport MenuItem from '@mui/material/MenuItem'\nimport MenuList from '@mui/material/MenuList'\nimport Paper from '@mui/material/Paper'\nimport Popper from '@mui/material/Popper'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n popper: {\n zIndex: (theme) => theme.zIndex.appBar + 1,\n },\n}\n\nconst SelectButton = ({\n disabled = false,\n hideSelectedOptionFromMenu = false,\n options = [],\n switchButtonToSelected = true,\n}) => {\n const anchorRef = useRef(null)\n const [open, setOpen] = useState(false)\n const [selectedIndex, setSelectedIndex] = useState(0)\n\n const handleClick = (event) => {\n const selectedOption = options[selectedIndex]\n\n if (selectedOption && typeof selectedOption.onClick === 'function') {\n selectedOption.onClick(event)\n }\n }\n\n const handleMenuItemClick = (event, index) => {\n if (switchButtonToSelected) {\n setSelectedIndex(index)\n }\n\n const selectedOption = options[index]\n if (typeof selectedOption.onClick === 'function') {\n selectedOption.onClick(event)\n }\n\n setOpen(false)\n }\n\n const handleToggle = () => {\n setOpen((prevOpen) => !prevOpen)\n }\n\n const handleClose = (event) => {\n if (anchorRef.current && anchorRef.current.contains(event.target)) {\n return\n }\n\n setOpen(false)\n }\n\n return (\n <>\n \n \n {options[selectedIndex] ? options[selectedIndex].labelText : null}\n \n \n \n \n \n \n {({ TransitionProps, placement }) => (\n \n \n \n \n {options.map(\n (option, index) =>\n !(index === selectedIndex && hideSelectedOptionFromMenu) && (\n handleMenuItemClick(event, index)}\n >\n {option.labelText}\n \n )\n )}\n \n \n \n \n )}\n \n \n )\n}\n\nSelectButton.propTypes = {\n disabled: PropTypes.bool,\n hideSelectedOptionFromMenu: PropTypes.bool,\n options: PropTypes.arrayOf(\n PropTypes.shape({\n labelText: PropTypes.string.isRequired,\n disabled: PropTypes.bool,\n onClick: PropTypes.func,\n })\n ),\n switchButtonToSelected: PropTypes.bool,\n}\n\nexport default SelectButton\n","import { useTranslation } from 'react-i18next'\nimport { useMsal } from '@azure/msal-react'\nimport { Box, Card, CardContent, Link, Typography } from '@mui/material'\n\nimport { AppLogo } from '@/common/components'\n\nimport { loginRequest } from '../../../authConfig'\n\nconst classes = {\n cardContainer: {\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n flexDirection: 'column',\n height: '100vh',\n },\n root: {\n borderRadius: 4,\n width: 860,\n height: 400,\n display: 'flex',\n flexDirection: 'column',\n },\n logoImgContainer: {\n display: 'flex',\n alignItems: 'center',\n justifyContent: 'center',\n flexDirection: 'column',\n width: '100%',\n marginBottom: 2,\n },\n logoImg: {\n height: 128,\n },\n contentContainer: {\n flexGrow: 1,\n display: 'flex',\n flexDirection: 'column',\n alignItems: 'center',\n justifyContent: 'center',\n },\n link: {\n cursor: 'pointer',\n },\n title: {\n marginBottom: 2,\n },\n}\n\nconst SignOut = () => {\n const { t } = useTranslation()\n const { instance: msalInstance } = useMsal()\n const handleSignInClicked = () => {\n msalInstance.loginRedirect({\n ...loginRequest,\n redirectStartPage: '/',\n })\n }\n\n return (\n \n \n \n \n \n \n \n {t('Successfully signed out')}\n \n \n {t('To sign back in') + ' '}\n \n {t('click here')}\n \n \n \n \n \n )\n}\n\nexport default SignOut\n","import { Box } from '@mui/material'\n\nconst HorizontalSpacer = () => {\n return \n}\n\nexport default HorizontalSpacer\n","import SvgIcon from '@mui/material/SvgIcon'\n\n// https://materialdesignicons.com/icon/delete\nconst DeleteIcon = (props) => {\n return (\n \n \n \n )\n}\n\nexport default DeleteIcon\n","import MuiTableCell from '@mui/material/TableCell'\nimport PropTypes from 'prop-types'\n\nconst TableCell = ({ align = 'inherit', children, colSpan = 1, padding, ...other }) => {\n return (\n \n {children}\n \n )\n}\n\nTableCell.propTypes = {\n align: PropTypes.oneOf(['center', 'inherit', 'justify', 'left', 'right']),\n children: PropTypes.node,\n colSpan: PropTypes.number,\n padding: PropTypes.oneOf(['default', 'checkbox', 'none']),\n}\n\nexport default TableCell\n","import { cloneElement, useEffect, useState } from 'react'\nimport PropTypes from 'prop-types'\n\nimport CheckMarkIcon from '../../../icons/CheckMarkIcon/CheckMarkIcon'\nimport CrossIcon from '../../../icons/CrossIcon/CrossIcon'\nimport DeleteIcon from '../../../icons/DeleteIcon/DeleteIcon'\nimport EditIcon from '../../../icons/EditIcon/EditIcon'\nimport IconButton from '../../IconButton/IconButton'\nimport TableCell from '../TableCell/TableCell'\n\nconst TableActionsCell = ({\n actions,\n className,\n hidden = false,\n isEditing = false,\n onDeleteClicked,\n onEditCancelClicked,\n onEditClicked,\n onEditSaveClicked,\n row,\n rowIndex,\n showDeleteAction = true,\n showEditAction = true,\n}) => {\n const actionWidth = 48\n const [isInEditingMode, setIsInEdtingMode] = useState(false)\n const [cellWidth, setCellWidth] = useState(actionWidth)\n\n useEffect(() => {\n setIsInEdtingMode(isEditing)\n }, [isEditing])\n\n useEffect(() => {\n let width = 0\n\n if (actions) {\n width += actions.length * actionWidth\n }\n if (showEditAction) {\n width += 2 * actionWidth\n }\n if (showDeleteAction) {\n width += actionWidth\n }\n\n setCellWidth(width)\n }, [actions, showDeleteAction, showEditAction])\n\n const handleEditButonClicked = (event) => {\n setIsInEdtingMode(true)\n\n if (typeof onEditClicked === 'function') {\n onEditClicked(event, row, rowIndex)\n }\n }\n\n const handleEditSaveButonClicked = (event) => {\n if (typeof onEditSaveClicked === 'function') {\n onEditSaveClicked(event, row, rowIndex)\n }\n\n setIsInEdtingMode(false)\n }\n\n const handleEditCancelButonClicked = (event) => {\n if (typeof onEditCancelClicked === 'function') {\n onEditCancelClicked(event, row, rowIndex)\n }\n\n setIsInEdtingMode(false)\n }\n\n const handleDeleteButtonClicked = (event) => {\n if (typeof onDeleteClicked === 'function') {\n onDeleteClicked(event, row, rowIndex)\n }\n }\n\n return (\n !hidden && (\n \n {actions && actions.length > 0\n ? actions.map((action, index) => {\n const actionClickHandler = action.props.onClick\n return cloneElement(action, {\n key: index,\n onClick: (event) => {\n event.stopPropagation()\n if (actionClickHandler) {\n actionClickHandler(event, row)\n }\n },\n })\n })\n : null}\n {showEditAction ? (\n isInEditingMode ? (\n <>\n {\n event.stopPropagation()\n handleEditSaveButonClicked(event)\n }}\n >\n \n \n {\n event.stopPropagation()\n handleEditCancelButonClicked(event)\n }}\n >\n \n \n \n ) : (\n {\n event.stopPropagation()\n handleEditButonClicked(event)\n }}\n >\n \n \n )\n ) : null}\n {showDeleteAction && !isInEditingMode ? (\n {\n event.stopPropagation()\n handleDeleteButtonClicked(event)\n }}\n >\n \n \n ) : null}\n \n )\n )\n}\n\nTableActionsCell.propTypes = {\n row: PropTypes.object.isRequired,\n rowIndex: PropTypes.number.isRequired,\n actions: PropTypes.arrayOf(PropTypes.object),\n hidden: PropTypes.bool,\n isEditing: PropTypes.bool,\n showDeleteAction: PropTypes.bool,\n showEditAction: PropTypes.bool,\n onDeleteClicked: PropTypes.func,\n onEditCancelClicked: PropTypes.func,\n onEditClicked: PropTypes.func,\n onEditSaveClicked: PropTypes.func,\n}\n\nexport default TableActionsCell\n","import { NumericFormat } from 'react-number-format'\nimport { useSelector } from 'react-redux'\nimport TextField from '@mui/material/TextField'\nimport PropTypes from 'prop-types'\n\nimport { selectLocale } from '@/app/slices/appSlice'\nimport { selectCurrencyCode } from '@/app/slices/organisationSlice'\nimport { getCurrencyFormat, getFormattedPrice } from '@/common/utils'\n\nconst CurrencyInput = ({\n InputLabelProps,\n allowNegativeValues = false,\n customPlaceholder,\n disabled = false,\n error,\n helperText,\n inputProps,\n inputRef,\n label = 'Currency Input',\n onBlur,\n onChange,\n required = false,\n size = 'small',\n value,\n}) => {\n const currencyCode = useSelector(selectCurrencyCode)\n const locale = useSelector(selectLocale)\n const placeholder =\n typeof customPlaceholder === 'number'\n ? getFormattedPrice(customPlaceholder, currencyCode, locale)\n : customPlaceholder\n\n const handleOnChange = (values) => {\n if (typeof onChange === 'function') {\n const { floatValue } = values\n onChange(floatValue)\n }\n }\n\n const currencyFormat = getCurrencyFormat(currencyCode, locale)\n const literal = currencyFormat['literal'] ?? ''\n const currencySymbol =\n currencyFormat['currencySymbolPosition'] === 'before'\n ? currencyFormat['currency'] + literal\n : literal + currencyFormat['currency']\n const prefix = currencyFormat['currencySymbolPosition'] === 'before' ? currencySymbol : undefined\n const suffix = currencyFormat['currencySymbolPosition'] === 'after' ? currencySymbol : undefined\n const decimalPlaces = currencyFormat['decimalPlaces']\n const groupSeparator = currencyFormat['group']\n const decimalSeparator = groupSeparator === '.' ? ',' : '.'\n\n return (\n \n )\n}\n\nexport const currencyInputPropTypes = {\n allowNegativeValues: PropTypes.bool,\n customPlaceholder: PropTypes.oneOfType([PropTypes.number, PropTypes.string]),\n disabled: PropTypes.bool,\n error: PropTypes.bool,\n helperText: PropTypes.string,\n InputLabelProps: PropTypes.object,\n inputProps: PropTypes.object,\n inputRef: PropTypes.object,\n label: PropTypes.string,\n required: PropTypes.bool,\n size: PropTypes.oneOf(['small', 'medium']),\n value: PropTypes.number,\n onBlur: PropTypes.func,\n onChange: PropTypes.func,\n onErrorChange: PropTypes.func,\n}\n\nCurrencyInput.propTypes = currencyInputPropTypes\n\nexport default CurrencyInput\n","import 'dayjs/locale/de'\nimport 'dayjs/locale/en'\nimport 'dayjs/locale/en-gb'\nimport 'dayjs/locale/es'\nimport 'dayjs/locale/fr'\nimport 'dayjs/locale/it'\nimport 'dayjs/locale/pt'\nimport 'dayjs/locale/vi'\n\nimport { useMemo } from 'react'\nimport { useSelector } from 'react-redux'\nimport { LocalizationProvider } from '@mui/x-date-pickers'\nimport { AdapterDayjs } from '@mui/x-date-pickers/AdapterDayjs'\nimport { deDE, enUS, esES, frFR, itIT, ptBR, viVN } from '@mui/x-date-pickers/locales'\nimport dayjs from 'dayjs'\nimport isBetween from 'dayjs/plugin/isBetween'\nimport timezone from 'dayjs/plugin/timezone'\nimport utc from 'dayjs/plugin/utc'\n\nimport { selectOrganisation } from '@/app/slices/organisationSlice'\n\ndayjs.extend(utc)\ndayjs.extend(timezone)\ndayjs.extend(isBetween)\n\nconst TbxLocalizationProvider = ({ children }) => {\n const organisation = useSelector(selectOrganisation)\n\n const getAdapterLocale = (language) => {\n const adapter = (language) => {\n switch (language) {\n case 'en-US':\n return 'en'\n case 'en-GB':\n return 'en-gb'\n case 'de-DE':\n return 'de'\n case 'es-ES':\n return 'es'\n case 'fr-FR':\n return 'fr'\n case 'it-IT':\n return 'it'\n case 'pt-PT':\n return 'pt'\n case 'vi-VN':\n return 'vi'\n default:\n return 'en'\n }\n }\n\n dayjs.locale(adapter(language))\n\n return adapter(language)\n }\n\n const getLocaleText = (language) => {\n switch (language) {\n case 'en-US':\n return enUS.components.MuiLocalizationProvider.defaultProps.localeText\n case 'en-GB':\n return enUS.components.MuiLocalizationProvider.defaultProps.localeText\n case 'de-DE':\n return deDE.components.MuiLocalizationProvider.defaultProps.localeText\n case 'es-ES':\n return esES.components.MuiLocalizationProvider.defaultProps.localeText\n case 'fr-FR':\n return frFR.components.MuiLocalizationProvider.defaultProps.localeText\n case 'it-IT':\n return itIT.components.MuiLocalizationProvider.defaultProps.localeText\n case 'pt-PT':\n return ptBR.components.MuiLocalizationProvider.defaultProps.localeText\n case 'vi-VN':\n return viVN.components.MuiLocalizationProvider.defaultProps.localeText\n default:\n return enUS.components.MuiLocalizationProvider.defaultProps.localeText\n }\n }\n\n const adapterLocale = useMemo(() => getAdapterLocale(organisation.language), [organisation.language])\n\n const localeText = useMemo(() => getLocaleText(organisation.language), [organisation.language])\n\n return (\n (params.contentType === 'letter' ? 'MMM' : 'MM'),\n }}\n >\n {children}\n \n )\n}\n\nexport default TbxLocalizationProvider\n","import Checkbox from '@mui/material/Checkbox'\nimport FormControl from '@mui/material/FormControl'\nimport FormControlLabel from '@mui/material/FormControlLabel'\nimport FormGroup from '@mui/material/FormGroup'\nimport FormHelperText from '@mui/material/FormHelperText'\nimport MenuItem from '@mui/material/MenuItem'\nimport Select from '@mui/material/Select'\nimport TextField from '@mui/material/TextField'\nimport { DateTimePicker, DesktopDatePicker, TimePicker } from '@mui/x-date-pickers'\nimport PropTypes from 'prop-types'\n\nimport CurrencyInput from '../../CurrencyInput/CurrencyInput'\nimport TbxLocalizationProvider from '../../TbxLocalizationProvider/TbxLocalizationProvider'\n\nconst TableEditComponent = ({\n allowNegativeValues,\n autoFocus = true,\n dateSetting,\n error,\n helperText,\n onBlur,\n onChange,\n onKeyDown,\n placeholder,\n readOnly,\n selectOptions,\n title,\n type = 'text',\n value,\n}) => {\n const renderLookupField = () => {\n return (\n \n onChange(event.target.value)}\n >\n {Object.keys(selectOptions).map((key) => (\n \n {selectOptions[key]}\n \n ))}\n \n {helperText ? {helperText} : null}\n \n )\n }\n\n const renderBooleanField = () => {\n return (\n \n \n onChange(event.target.checked)}\n />\n }\n label=\"\"\n />\n \n {helperText}\n \n )\n }\n\n const renderDateField = () => {\n const dateFormat = dateSetting && dateSetting.format ? dateSetting.format : 'dd-MM-yyyy'\n return (\n \n \n \n )\n }\n\n const renderTimeField = () => {\n return (\n \n \n \n )\n }\n\n const renderDateTimeField = () => {\n return (\n \n \n \n )\n }\n\n const renderTextField = () => {\n return (\n onChange(type === 'numeric' ? event.target.valueAsNumber : event.target.value)}\n />\n )\n }\n\n const renderCurrencyField = () => {\n return (\n onChange(event)}\n onKeyDown={onKeyDown}\n />\n )\n }\n\n const renderComponent = () => {\n let component = 'ok'\n\n if (type === 'select') {\n component = renderLookupField()\n } else if (type === 'boolean') {\n component = renderBooleanField()\n } else if (type === 'date') {\n component = renderDateField()\n } else if (type === 'time') {\n component = renderTimeField()\n } else if (type === 'datetime') {\n component = renderDateTimeField()\n } else if (type === 'currency') {\n component = renderCurrencyField()\n } else {\n component = renderTextField()\n }\n\n return component\n }\n\n return renderComponent()\n}\n\nTableEditComponent.propTypes = {\n autoFocus: PropTypes.bool,\n dateSetting: PropTypes.shape({\n format: PropTypes.string,\n }),\n error: PropTypes.string,\n helperText: PropTypes.string,\n placeholder: PropTypes.string,\n readOnly: PropTypes.bool,\n selectOptions: PropTypes.arrayOf(PropTypes.object),\n title: PropTypes.string,\n type: PropTypes.oneOf(['text', 'numeric', 'currency', 'datetime', 'select', 'boolean']),\n value: PropTypes.any,\n onChange: PropTypes.func,\n onKeyDown: PropTypes.func,\n}\n\nexport default TableEditComponent\n","import MuiTableRow from '@mui/material/TableRow'\nimport PropTypes from 'prop-types'\n\nconst TableRow = ({ children, hover = false, selected = false, ...other }) => {\n return (\n \n {children}\n \n )\n}\n\nTableRow.propTypes = {\n children: PropTypes.node,\n hover: PropTypes.bool,\n selected: PropTypes.bool,\n}\n\nexport default TableRow\n","import { Box } from '@mui/material'\nimport Checkbox from '@mui/material/Checkbox'\nimport MuiTableHead from '@mui/material/TableHead'\nimport MuiTableSortLabel from '@mui/material/TableSortLabel'\nimport PropTypes from 'prop-types'\n\nimport { getCellAlignmentForColumn } from '../Table'\nimport TableCell from '../TableCell/TableCell'\nimport TableRow from '../TableRow/TableRow'\n\nconst classes = {\n visuallyHidden: {\n border: 0,\n clip: 'rect(0 0 0 0)',\n height: 1,\n margin: -1,\n overflow: 'hidden',\n padding: 0,\n position: 'absolute',\n top: 20,\n width: 1,\n },\n tableCellColumnBorders: {\n border: 'solid 1px',\n borderTop: 'none',\n },\n tableCellFullBorders: {\n border: 'solid 1px',\n },\n tableCellNoBorder: {\n border: 'none',\n },\n tableCell: {\n zIndex: 900,\n backgroundColor: 'background.default',\n fontSize: '12px',\n fontWeight: 500,\n },\n headerContainer: {\n display: 'flex',\n flexDirection: 'row',\n alignItems: 'flex-start',\n justifyContent: 'flex-start',\n },\n invertedHeaderContainer: {\n display: 'flex',\n flexDirection: 'row-reverse',\n alignItems: 'flex-start',\n justifyContent: 'flex-start',\n },\n}\n\nconst TableHead = ({\n actionColumnPosition = 'end',\n allowMultiRowSelect = true,\n borderStyle = 'rows',\n cellStyle,\n columns = [],\n numberOfRowsSelected,\n onRequestSort,\n onSelectAllClick,\n order,\n orderBy,\n rowCount,\n showActionsColumn = false,\n}) => {\n const tableCellStyle = Object.assign(\n {},\n classes.tableCell,\n borderStyle === 'none' ? classes.tableCellNoBorder : {},\n borderStyle === 'columns' ? classes.tableCellColumnBorders : {},\n borderStyle === 'cells' ? classes.tableCellFullBorders : {}\n )\n\n const createSortHandler = (property) => (event) => {\n onRequestSort(event, property)\n }\n\n const renderActionsColumn = () => {\n return (\n showActionsColumn && (\n \n )\n )\n }\n\n const columnIsSortable = (column) => typeof column.sortable === 'undefined' || column.sortable\n\n return (\n \n \n {allowMultiRowSelect && columns.length > 0 ? \n 0 ? numberOfRowsSelected === rowCount : null}\n indeterminate={numberOfRowsSelected > 0 ? numberOfRowsSelected < rowCount : null}\n onChange={onSelectAllClick}\n />\n : null}\n {actionColumnPosition === 'start' ? renderActionsColumn() : null}\n {columns.map(\n (column, index) =>\n !column.hidden && (\n \n \n {column.label}\n {columnIsSortable(column) && orderBy === column.id ? (\n \n {order === 'desc' ? 'sorted descending' : 'sorted ascending'}\n \n ) : null}\n \n \n )\n )}\n {actionColumnPosition === 'end' ? renderActionsColumn() : null}\n \n \n )\n}\n\nTableHead.propTypes = {\n numberOfRowsSelected: PropTypes.number.isRequired,\n order: PropTypes.oneOf(['asc', 'desc']).isRequired,\n orderBy: PropTypes.string.isRequired,\n rowCount: PropTypes.number.isRequired,\n onRequestSort: PropTypes.func.isRequired,\n onSelectAllClick: PropTypes.func.isRequired,\n actionColumnPosition: PropTypes.oneOf(['end', 'start']),\n allowMultiRowSelect: PropTypes.bool,\n borderStyle: PropTypes.oneOf(['none', 'columns', 'rows', 'cells']),\n cellStyle: PropTypes.object,\n columns: PropTypes.arrayOf(PropTypes.object),\n showActionsColumn: PropTypes.bool,\n}\n\nexport default TableHead\n","import { useTranslation } from 'react-i18next'\nimport { Delete } from '@mui/icons-material'\nimport { Box, CircularProgress, Toolbar as MuiToolbar, Typography } from '@mui/material'\nimport IconButton from '@mui/material/IconButton'\nimport { lighten } from '@mui/material/styles'\nimport Tooltip from '@mui/material/Tooltip'\nimport PropTypes from 'prop-types'\n\nconst classes = {\n root: (showDivider) => ({\n paddingLeft: 2,\n paddingRight: 2,\n borderWidth: showDivider ? 1 : 0,\n borderBottomStyle: showDivider ? 'solid' : 'none',\n borderColor: (theme) => theme.palette.divider,\n }),\n highlight: (theme) =>\n theme.palette.mode === 'light'\n ? {\n color: theme.palette.secondary.main,\n backgroundColor: lighten(theme.palette.secondary.light, 0.85),\n }\n : {\n color: theme.palette.text.primary,\n backgroundColor: theme.palette.secondary.dark,\n },\n title: (titleTransform) => {\n let textTransform = 'none'\n\n if (titleTransform === 'uppercase') {\n textTransform = 'uppercase'\n } else if (titleTransform === 'lowercase') {\n textTransform = 'lowercase'\n }\n\n return {\n flex: '1 1 0%',\n textTransform: textTransform,\n color: 'text.primary',\n }\n },\n selection: {\n flex: '5 1 0%',\n },\n loadingContainer: {\n flex: '1 1 0%',\n marginRight: 2,\n },\n}\n\nconst TableToolbar = ({\n actionPanel,\n allowMultiRowDelete,\n isLoading = false,\n loadingComponent = ,\n numberOfRowsSelected,\n onDelete,\n selectionComponent,\n showDivider = false,\n showSelectionComponent = true,\n style,\n title,\n titleTransform = 'default',\n}) => {\n const { t } = useTranslation()\n\n const handleDeleteClicked = () => {\n if (typeof onDelete === 'function') {\n onDelete()\n }\n }\n const showSelection = showSelectionComponent && numberOfRowsSelected > 0\n\n const toolbarClass = [classes.root(showDivider), showSelection && classes.highlight]\n\n return (\n \n {isLoading ? {loadingComponent} : null}\n {showSelection ? (\n selectionComponent ? (\n {selectionComponent}\n ) : (\n \n {numberOfRowsSelected + ' ' + t('selected')}\n \n )\n ) : (\n !isLoading && (\n \n {title}\n \n )\n )}\n {allowMultiRowDelete && numberOfRowsSelected > 0 ? \n \n \n \n : null}\n {actionPanel}\n \n )\n}\n\nTableToolbar.propTypes = {\n actionPanel: PropTypes.node,\n allowMultiRowDelete: PropTypes.bool,\n isLoading: PropTypes.bool,\n loadingComponent: PropTypes.node,\n numberOfRowsSelected: PropTypes.number,\n selectionComponent: PropTypes.node,\n showDivider: PropTypes.bool,\n showSelectionComponent: PropTypes.bool,\n style: PropTypes.object,\n title: PropTypes.string,\n titleTransform: PropTypes.oneOf(['default', 'uppercase', 'lowercase']),\n onDelete: PropTypes.func,\n}\n\nexport default TableToolbar\n","import { Fragment, useCallback, useEffect, useRef, useState } from 'react'\nimport { useTranslation } from 'react-i18next'\nimport {\n Box,\n Checkbox,\n CircularProgress,\n darken,\n lighten,\n Table as MuiTable,\n TableBody,\n TableContainer,\n TablePagination,\n Tooltip,\n Typography,\n} from '@mui/material'\nimport _ from 'lodash'\nimport PropTypes from 'prop-types'\n\nimport { PARTS_PER_PAGE_OPTIONS, pascalToKebabCase, pascalToSentenceCase } from '@/common/utils'\n\nimport TableActionsCell from './TableActionsCell/TableActionsCell'\nimport TableCell from './TableCell/TableCell'\nimport TableEditComponent from './TableEditComponent/TableEditComponent'\nimport TableHead from './TableHead/TableHead'\nimport TableRow from './TableRow/TableRow'\nimport TableToolbar from './TableToolbar/TableToolbar'\n\nexport function getCellAlignmentForColumn(column) {\n if (!column) {\n return 'inherit'\n }\n\n if (column.align) {\n return column.align\n }\n\n switch (column.type) {\n case 'numeric':\n case 'datetime':\n case 'currency':\n return 'right'\n case 'boolean':\n return 'center'\n case 'text':\n return 'left'\n default:\n return 'inherit'\n }\n}\n\nconst classes = {\n root: {\n width: '100%',\n },\n tableFooter: {\n background: 'background.default',\n overflow: 'hidden',\n },\n tableContainerRoot: {\n overflowX: 'hidden',\n flexGrow: 1,\n },\n inputRoot: {\n width: '100%',\n },\n noDataCell: {\n textAlign: 'center',\n },\n noDataText: (theme) =>\n theme.palette.mode === 'light'\n ? {\n color: lighten(theme.palette.text.primary, 0.25),\n }\n : {\n color: darken(theme.palette.text.primary, 0.25),\n },\n editableTableCell: {\n cursor: 'pointer',\n },\n tableCellColumnBorders: {\n border: 'solid 1px',\n borderBottom: 'none',\n borderTop: 'none',\n },\n tableCellFullBorders: {\n border: 'solid 1px',\n },\n tableCellNoBorder: {\n border: 'none',\n },\n toolbar: {\n zIndex: 1000,\n backgroundColor: 'background.default',\n },\n}\n\nconst Table = ({\n actionColumnPosition = 'end',\n allowMultiRowDelete = true,\n allowMultiRowSelect = true,\n borderStyle = 'rows',\n className,\n columns = [],\n data = [],\n editMode = 'row',\n footerRows,\n hideEmptyTable = false,\n hideToolbar = false,\n isLoading = false,\n loadingComponent = ,\n locale,\n nestedData = [],\n noDataText,\n onPageChange,\n onRowClicked,\n onRowDeleted,\n onRowUpdated,\n onRowsDeleted,\n onRowsPerPageChange,\n onSelectionChanged,\n orderColumn = '',\n orderDirection = 'asc',\n page = 0,\n rowActions = [],\n rowsPerPage = 10,\n rowsPerPageLabel = 'Rows per page',\n selectedData,\n selectionComponent,\n showPageCount = true,\n showPagination = true,\n showRowDeleteAction = false,\n showRowsPerPage = true,\n showSelectionComponent = true,\n showToolbarDivider = false,\n stickyFooter = false,\n stickyHeader = false,\n stickyToolbar = false,\n tableContainerProps,\n title = '',\n titleTransform = 'default',\n toolbarActionPanelContent,\n}) => {\n const { t } = useTranslation()\n\n const rootRef = useRef()\n const toolbarRef = useRef()\n const [cellBeingUpdated, setCellBeingUpdated] = useState({\n rowIndex: -1,\n columnIndex: -1,\n })\n const [order, setOrder] = useState(orderDirection)\n const [orderBy, setOrderBy] = useState(orderColumn)\n const [rows, setRows] = useState([])\n const [rowsOld, setRowsOld] = useState([])\n const [selected, setSelected] = useState([])\n const [editingRow, setEditingRow] = useState(null)\n const [currentPage, setCurrentPage] = useState(page)\n const [currentRowsPerPage, setCurrentRowsPerPage] = useState(showPagination ? rowsPerPage : rows.length)\n\n useEffect(() => {\n if (!showPagination) {\n setCurrentRowsPerPage(rows.length)\n } else {\n setCurrentRowsPerPage(rowsPerPage)\n }\n }, [showPagination, rows, rowsPerPage])\n\n const descendingComparator = useCallback((a, b, orderBy) => {\n const first = a[orderBy]\n const next = b[orderBy]\n\n switch (typeof first) {\n case 'number':\n if (next < first) return -1\n if (next > first) return 1\n break\n case 'string':\n return next?.localeCompare(first, undefined, { sensitivity: 'base' })\n default:\n return 0\n }\n return 0\n }, [])\n\n const getComparator = useCallback(\n (order, orderBy) => {\n return order === 'desc'\n ? (a, b) => descendingComparator(a, b, orderBy)\n : (a, b) => -descendingComparator(a, b, orderBy)\n },\n [descendingComparator]\n )\n\n const stableSort = useCallback((array, comparator) => {\n const stabilizedThis = array.map((el, index) => [el, index])\n stabilizedThis.sort((a, b) => {\n const order = comparator(a[0], b[0])\n if (order !== 0) return order\n return a[1] - b[1]\n })\n return stabilizedThis.map((el) => el[0])\n }, [])\n\n useEffect(() => {\n setRows(_.cloneDeep(data))\n setRowsOld(_.cloneDeep(data))\n }, [data])\n\n useEffect(() => {\n setCurrentPage(page)\n }, [page])\n\n useEffect(() => {\n setCurrentRowsPerPage(showPagination ? rowsPerPage : rows.length)\n }, [showPagination, rowsPerPage, rows])\n\n useEffect(() => {\n if (selectedData) {\n setSelected(selectedData)\n }\n }, [selectedData])\n\n useEffect(() => {\n setRows(stableSort(data, getComparator(order, orderBy)))\n }, [order, orderBy, data, stableSort, getComparator])\n\n const revertRowState = () => {\n setRows(stableSort(_.cloneDeep(rowsOld), getComparator(order, orderBy)))\n }\n\n const saveRowState = () => {\n setRowsOld(_.cloneDeep(rows))\n }\n\n const selectedRowIndex = (row) =>\n selected.findIndex((element) => {\n return _.isEqual(row, element)\n })\n const isSelected = (row) => selectedRowIndex(row) !== -1\n\n const handleRequestSort = (_event, property) => {\n const isAsc = orderBy === property && order === 'asc'\n setOrder(isAsc ? 'desc' : 'asc')\n setOrderBy(property)\n }\n\n const handleSelectionChanged = (newSelection) => {\n setSelected(newSelection)\n\n if (typeof onSelectionChanged === 'function') {\n onSelectionChanged(newSelection)\n }\n }\n\n const handleSelectAllClick = (event) => {\n if (event.target.checked) {\n const newSelecteds = rows\n handleSelectionChanged(newSelecteds)\n return\n }\n handleSelectionChanged([])\n }\n\n const handleSelectClick = (event, row) => {\n event.stopPropagation()\n\n const selectedIndex = selectedRowIndex(row)\n let newSelected = []\n\n if (selectedIndex === -1) {\n newSelected = newSelected.concat(selected, row)\n } else if (selectedIndex === 0) {\n newSelected = newSelected.concat(selected.slice(1))\n } else if (selectedIndex === selected.length - 1) {\n newSelected = newSelected.concat(selected.slice(0, -1))\n } else if (selectedIndex > 0) {\n newSelected = newSelected.concat(selected.slice(0, selectedIndex), selected.slice(selectedIndex + 1))\n }\n\n handleSelectionChanged(newSelected)\n }\n\n const handleRowClick = (event, row) => {\n if (allowMultiRowSelect) {\n handleSelectionChanged([row])\n }\n if (typeof onRowClicked === 'function') {\n onRowClicked(event, row)\n }\n }\n\n const handleCellClick = (cellCoordinates) => {\n setCellBeingUpdated(cellCoordinates)\n }\n\n const handleCellValueChanged = (value, cellCoordinates) => {\n const newRows = [...rows]\n\n const columnId = columns[cellCoordinates.columnIndex].id\n newRows[cellCoordinates.rowIndex][columnId] = value\n\n setRows(newRows)\n }\n\n const handleCellInputBlur = (cellCoordinates) => {\n const columnId = columns[cellCoordinates.columnIndex].id\n const oldRow = rowsOld[cellCoordinates.rowIndex]\n const newRow = rows[cellCoordinates.rowIndex]\n\n if (editMode === 'cell') {\n if (typeof onRowUpdated === 'function') {\n if (!_.isEqual(oldRow[columnId], newRow[columnId])) {\n onRowUpdated(newRow)\n }\n }\n\n saveRowState()\n }\n\n setCellBeingUpdated({ rowIndex: -1, columnIndex: -1 })\n }\n\n const handleEditButonClicked = (_event, row) => {\n setEditingRow(row)\n }\n\n const handleEditSaveButtonClicked = (_event, row, _rowIndex) => {\n if (typeof onRowUpdated === 'function') {\n onRowUpdated(row)\n }\n\n saveRowState()\n setEditingRow(null)\n }\n\n const handleEditCancelButtonClicked = (_event, _row, _rowIndex) => {\n revertRowState()\n setEditingRow(null)\n }\n\n const handleRowDeleteClicked = (_event, row, _rowIndex) => {\n if (typeof onRowDeleted === 'function') {\n onRowDeleted(row)\n }\n }\n\n const handleMultiRowDeleteClicked = () => {\n if (typeof onRowsDeleted === 'function') {\n onRowsDeleted(selected)\n }\n }\n\n const handleChangePage = (_, newPage) => {\n setCurrentPage(newPage)\n\n if (typeof onPageChange === 'function') {\n onPageChange(newPage)\n }\n }\n\n const handleChangeRowsPerPage = (event) => {\n const newRowsPerPage = parseInt(event.target.value, 10)\n setCurrentRowsPerPage(newRowsPerPage)\n setCurrentPage(0)\n\n if (typeof onRowsPerPageChange === 'function') {\n onRowsPerPageChange(newRowsPerPage)\n }\n }\n\n const customRowActionsSpecified = rowActions && rowActions.length > 0\n const shouldRenderActionsColumn = showRowDeleteAction || editMode === 'row' || customRowActionsSpecified\n\n const tableCellStyle = Object.assign(\n {},\n borderStyle === 'none' ? classes.tableCellNoBorder : {},\n borderStyle === 'columns' ? classes.tableCellColumnBorders : {},\n borderStyle === 'cells' ? classes.tableCellFullBorders : {}\n )\n\n const renderActionCells = (row, rowIndex) => {\n const isEditingThisRow = _.isEqual(editingRow, row)\n\n return (\n