@voyant-travel/bookings-react 0.119.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +201 -0
- package/README.md +87 -0
- package/dist/admin/booking-contract-dialog.d.ts +22 -0
- package/dist/admin/booking-contract-dialog.d.ts.map +1 -0
- package/dist/admin/booking-contract-dialog.js +161 -0
- package/dist/admin/booking-detail-host.d.ts +103 -0
- package/dist/admin/booking-detail-host.d.ts.map +1 -0
- package/dist/admin/booking-detail-host.js +127 -0
- package/dist/admin/booking-detail-skeleton.d.ts +7 -0
- package/dist/admin/booking-detail-skeleton.d.ts.map +1 -0
- package/dist/admin/booking-detail-skeleton.js +24 -0
- package/dist/admin/booking-documents-table.d.ts +13 -0
- package/dist/admin/booking-documents-table.d.ts.map +1 -0
- package/dist/admin/booking-documents-table.js +259 -0
- package/dist/admin/booking-invoice-sheet.d.ts +18 -0
- package/dist/admin/booking-invoice-sheet.d.ts.map +1 -0
- package/dist/admin/booking-invoice-sheet.js +101 -0
- package/dist/admin/booking-journey-host.d.ts +24 -0
- package/dist/admin/booking-journey-host.d.ts.map +1 -0
- package/dist/admin/booking-journey-host.js +278 -0
- package/dist/admin/bookings-host.d.ts +26 -0
- package/dist/admin/bookings-host.d.ts.map +1 -0
- package/dist/admin/bookings-host.js +18 -0
- package/dist/admin/bookings-list-skeleton.d.ts +10 -0
- package/dist/admin/bookings-list-skeleton.d.ts.map +1 -0
- package/dist/admin/bookings-list-skeleton.js +25 -0
- package/dist/admin/index.d.ts +273 -0
- package/dist/admin/index.d.ts.map +1 -0
- package/dist/admin/index.js +331 -0
- package/dist/admin/journey-billing-duplicate-warning.d.ts +3 -0
- package/dist/admin/journey-billing-duplicate-warning.d.ts.map +1 -0
- package/dist/admin/journey-billing-duplicate-warning.js +26 -0
- package/dist/admin/journey-departure-picker.d.ts +7 -0
- package/dist/admin/journey-departure-picker.d.ts.map +1 -0
- package/dist/admin/journey-departure-picker.js +100 -0
- package/dist/admin/journey-units-picker.d.ts +11 -0
- package/dist/admin/journey-units-picker.d.ts.map +1 -0
- package/dist/admin/journey-units-picker.js +60 -0
- package/dist/admin/journey-voucher-picker.d.ts +3 -0
- package/dist/admin/journey-voucher-picker.d.ts.map +1 -0
- package/dist/admin/journey-voucher-picker.js +71 -0
- package/dist/admin/pages/booking-compose-page.d.ts +9 -0
- package/dist/admin/pages/booking-compose-page.d.ts.map +1 -0
- package/dist/admin/pages/booking-compose-page.js +17 -0
- package/dist/admin/pages/booking-detail-page.d.ts +11 -0
- package/dist/admin/pages/booking-detail-page.d.ts.map +1 -0
- package/dist/admin/pages/booking-detail-page.js +14 -0
- package/dist/admin/pages/booking-journey-page.d.ts +12 -0
- package/dist/admin/pages/booking-journey-page.d.ts.map +1 -0
- package/dist/admin/pages/booking-journey-page.js +26 -0
- package/dist/admin/pages/booking-new-page.d.ts +15 -0
- package/dist/admin/pages/booking-new-page.d.ts.map +1 -0
- package/dist/admin/pages/booking-new-page.js +50 -0
- package/dist/admin/pages/bookings-index-page.d.ts +20 -0
- package/dist/admin/pages/bookings-index-page.d.ts.map +1 -0
- package/dist/admin/pages/bookings-index-page.js +18 -0
- package/dist/admin/person-bookings-widget.d.ts +13 -0
- package/dist/admin/person-bookings-widget.d.ts.map +1 -0
- package/dist/admin/person-bookings-widget.js +48 -0
- package/dist/admin/slots.d.ts +31 -0
- package/dist/admin/slots.d.ts.map +1 -0
- package/dist/admin/slots.js +30 -0
- package/dist/admin/use-booking-action-ledger-events.d.ts +15 -0
- package/dist/admin/use-booking-action-ledger-events.d.ts.map +1 -0
- package/dist/admin/use-booking-action-ledger-events.js +66 -0
- package/dist/client.d.ts +14 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +59 -0
- package/dist/components/booking-activity-timeline.d.ts +32 -0
- package/dist/components/booking-activity-timeline.d.ts.map +1 -0
- package/dist/components/booking-activity-timeline.js +147 -0
- package/dist/components/booking-billing-dialog.d.ts +16 -0
- package/dist/components/booking-billing-dialog.d.ts.map +1 -0
- package/dist/components/booking-billing-dialog.js +315 -0
- package/dist/components/booking-cancellation-dialog.d.ts +18 -0
- package/dist/components/booking-cancellation-dialog.d.ts.map +1 -0
- package/dist/components/booking-cancellation-dialog.js +79 -0
- package/dist/components/booking-combobox.d.ts +13 -0
- package/dist/components/booking-combobox.d.ts.map +1 -0
- package/dist/components/booking-combobox.js +44 -0
- package/dist/components/booking-create-form-utils.d.ts +56 -0
- package/dist/components/booking-create-form-utils.d.ts.map +1 -0
- package/dist/components/booking-create-form-utils.js +216 -0
- package/dist/components/booking-create-page.d.ts +14 -0
- package/dist/components/booking-create-page.d.ts.map +1 -0
- package/dist/components/booking-create-page.js +11 -0
- package/dist/components/booking-create-preview-card.d.ts +26 -0
- package/dist/components/booking-create-preview-card.d.ts.map +1 -0
- package/dist/components/booking-create-preview-card.js +107 -0
- package/dist/components/booking-create-product-extras-picker.d.ts +18 -0
- package/dist/components/booking-create-product-extras-picker.d.ts.map +1 -0
- package/dist/components/booking-create-product-extras-picker.js +82 -0
- package/dist/components/booking-create-sheet.d.ts +34 -0
- package/dist/components/booking-create-sheet.d.ts.map +1 -0
- package/dist/components/booking-create-sheet.js +811 -0
- package/dist/components/booking-create-utils.d.ts +66 -0
- package/dist/components/booking-create-utils.d.ts.map +1 -0
- package/dist/components/booking-create-utils.js +185 -0
- package/dist/components/booking-detail-page.d.ts +126 -0
- package/dist/components/booking-detail-page.d.ts.map +1 -0
- package/dist/components/booking-detail-page.js +264 -0
- package/dist/components/booking-dialog.d.ts +28 -0
- package/dist/components/booking-dialog.d.ts.map +1 -0
- package/dist/components/booking-dialog.js +130 -0
- package/dist/components/booking-document-dialog.d.ts +8 -0
- package/dist/components/booking-document-dialog.d.ts.map +1 -0
- package/dist/components/booking-document-dialog.js +83 -0
- package/dist/components/booking-document-list.d.ts +5 -0
- package/dist/components/booking-document-list.d.ts.map +1 -0
- package/dist/components/booking-document-list.js +43 -0
- package/dist/components/booking-group-link-dialog.d.ts +10 -0
- package/dist/components/booking-group-link-dialog.d.ts.map +1 -0
- package/dist/components/booking-group-link-dialog.js +79 -0
- package/dist/components/booking-group-section.d.ts +27 -0
- package/dist/components/booking-group-section.d.ts.map +1 -0
- package/dist/components/booking-group-section.js +51 -0
- package/dist/components/booking-guarantee-dialog.d.ts +10 -0
- package/dist/components/booking-guarantee-dialog.d.ts.map +1 -0
- package/dist/components/booking-guarantee-dialog.js +123 -0
- package/dist/components/booking-guarantee-list.d.ts +5 -0
- package/dist/components/booking-guarantee-list.d.ts.map +1 -0
- package/dist/components/booking-guarantee-list.js +86 -0
- package/dist/components/booking-item-dialog.d.ts +10 -0
- package/dist/components/booking-item-dialog.d.ts.map +1 -0
- package/dist/components/booking-item-dialog.js +155 -0
- package/dist/components/booking-item-list.d.ts +12 -0
- package/dist/components/booking-item-list.d.ts.map +1 -0
- package/dist/components/booking-item-list.js +191 -0
- package/dist/components/booking-item-travelers.d.ts +6 -0
- package/dist/components/booking-item-travelers.d.ts.map +1 -0
- package/dist/components/booking-item-travelers.js +57 -0
- package/dist/components/booking-list-filters.d.ts +43 -0
- package/dist/components/booking-list-filters.d.ts.map +1 -0
- package/dist/components/booking-list-filters.js +192 -0
- package/dist/components/booking-list.d.ts +50 -0
- package/dist/components/booking-list.d.ts.map +1 -0
- package/dist/components/booking-list.js +352 -0
- package/dist/components/booking-note-dialog.d.ts +16 -0
- package/dist/components/booking-note-dialog.d.ts.map +1 -0
- package/dist/components/booking-note-dialog.js +41 -0
- package/dist/components/booking-notes.d.ts +5 -0
- package/dist/components/booking-notes.d.ts.map +1 -0
- package/dist/components/booking-notes.js +45 -0
- package/dist/components/booking-payment-reconciliation-banner.d.ts +5 -0
- package/dist/components/booking-payment-reconciliation-banner.d.ts.map +1 -0
- package/dist/components/booking-payment-reconciliation-banner.js +91 -0
- package/dist/components/booking-payment-schedule-dialog.d.ts +10 -0
- package/dist/components/booking-payment-schedule-dialog.d.ts.map +1 -0
- package/dist/components/booking-payment-schedule-dialog.js +117 -0
- package/dist/components/booking-payment-schedule-list.d.ts +10 -0
- package/dist/components/booking-payment-schedule-list.d.ts.map +1 -0
- package/dist/components/booking-payment-schedule-list.js +217 -0
- package/dist/components/booking-payments-summary.d.ts +83 -0
- package/dist/components/booking-payments-summary.d.ts.map +1 -0
- package/dist/components/booking-payments-summary.js +176 -0
- package/dist/components/booking-quick-view-sheet.d.ts +14 -0
- package/dist/components/booking-quick-view-sheet.d.ts.map +1 -0
- package/dist/components/booking-quick-view-sheet.js +283 -0
- package/dist/components/bookings-page.d.ts +19 -0
- package/dist/components/bookings-page.d.ts.map +1 -0
- package/dist/components/bookings-page.js +9 -0
- package/dist/components/file-dropzone.d.ts +25 -0
- package/dist/components/file-dropzone.d.ts.map +1 -0
- package/dist/components/file-dropzone.js +102 -0
- package/dist/components/icon-action-button.d.ts +18 -0
- package/dist/components/icon-action-button.d.ts.map +1 -0
- package/dist/components/icon-action-button.js +13 -0
- package/dist/components/option-units-stepper-section.d.ts +111 -0
- package/dist/components/option-units-stepper-section.d.ts.map +1 -0
- package/dist/components/option-units-stepper-section.js +276 -0
- package/dist/components/payment-schedule-section.d.ts +91 -0
- package/dist/components/payment-schedule-section.d.ts.map +1 -0
- package/dist/components/payment-schedule-section.js +206 -0
- package/dist/components/person-picker-section.d.ts +71 -0
- package/dist/components/person-picker-section.d.ts.map +1 -0
- package/dist/components/person-picker-section.js +160 -0
- package/dist/components/price-breakdown-section.d.ts +83 -0
- package/dist/components/price-breakdown-section.d.ts.map +1 -0
- package/dist/components/price-breakdown-section.js +278 -0
- package/dist/components/product-picker-section.d.ts +29 -0
- package/dist/components/product-picker-section.d.ts.map +1 -0
- package/dist/components/product-picker-section.js +74 -0
- package/dist/components/shared-room-section.d.ts +40 -0
- package/dist/components/shared-room-section.d.ts.map +1 -0
- package/dist/components/shared-room-section.js +99 -0
- package/dist/components/status-badge.d.ts +24 -0
- package/dist/components/status-badge.d.ts.map +1 -0
- package/dist/components/status-badge.js +65 -0
- package/dist/components/status-change-dialog.d.ts +10 -0
- package/dist/components/status-change-dialog.d.ts.map +1 -0
- package/dist/components/status-change-dialog.js +57 -0
- package/dist/components/supplier-status-dialog.d.ts +10 -0
- package/dist/components/supplier-status-dialog.d.ts.map +1 -0
- package/dist/components/supplier-status-dialog.js +98 -0
- package/dist/components/supplier-status-list.d.ts +5 -0
- package/dist/components/supplier-status-list.d.ts.map +1 -0
- package/dist/components/supplier-status-list.js +115 -0
- package/dist/components/traveler-category-buttons.d.ts +26 -0
- package/dist/components/traveler-category-buttons.d.ts.map +1 -0
- package/dist/components/traveler-category-buttons.js +35 -0
- package/dist/components/traveler-dialog.d.ts +10 -0
- package/dist/components/traveler-dialog.d.ts.map +1 -0
- package/dist/components/traveler-dialog.js +256 -0
- package/dist/components/traveler-list.d.ts +6 -0
- package/dist/components/traveler-list.d.ts.map +1 -0
- package/dist/components/traveler-list.js +295 -0
- package/dist/components/travelers-section-controls.d.ts +52 -0
- package/dist/components/travelers-section-controls.d.ts.map +1 -0
- package/dist/components/travelers-section-controls.js +206 -0
- package/dist/components/travelers-section.d.ts +159 -0
- package/dist/components/travelers-section.d.ts.map +1 -0
- package/dist/components/travelers-section.js +355 -0
- package/dist/components/voucher-picker-section.d.ts +50 -0
- package/dist/components/voucher-picker-section.d.ts.map +1 -0
- package/dist/components/voucher-picker-section.js +79 -0
- package/dist/extras/client.d.ts +14 -0
- package/dist/extras/client.d.ts.map +1 -0
- package/dist/extras/client.js +58 -0
- package/dist/extras/components/extra-catalog-card.d.ts +13 -0
- package/dist/extras/components/extra-catalog-card.d.ts.map +1 -0
- package/dist/extras/components/extra-catalog-card.js +52 -0
- package/dist/extras/components/product-combobox.d.ts +9 -0
- package/dist/extras/components/product-combobox.d.ts.map +1 -0
- package/dist/extras/components/product-combobox.js +46 -0
- package/dist/extras/components/slot-extras-manifest-panel.d.ts +6 -0
- package/dist/extras/components/slot-extras-manifest-panel.d.ts.map +1 -0
- package/dist/extras/components/slot-extras-manifest-panel.js +108 -0
- package/dist/extras/hooks/index.d.ts +5 -0
- package/dist/extras/hooks/index.d.ts.map +1 -0
- package/dist/extras/hooks/index.js +4 -0
- package/dist/extras/hooks/use-product-extra.d.ts +24 -0
- package/dist/extras/hooks/use-product-extra.d.ts.map +1 -0
- package/dist/extras/hooks/use-product-extra.js +12 -0
- package/dist/extras/hooks/use-product-extras.d.ts +30 -0
- package/dist/extras/hooks/use-product-extras.d.ts.map +1 -0
- package/dist/extras/hooks/use-product-extras.js +12 -0
- package/dist/extras/hooks/use-slot-extra-manifest-mutation.d.ts +48 -0
- package/dist/extras/hooks/use-slot-extra-manifest-mutation.d.ts.map +1 -0
- package/dist/extras/hooks/use-slot-extra-manifest-mutation.js +26 -0
- package/dist/extras/hooks/use-slot-extra-manifest.d.ts +68 -0
- package/dist/extras/hooks/use-slot-extra-manifest.d.ts.map +1 -0
- package/dist/extras/hooks/use-slot-extra-manifest.js +11 -0
- package/dist/extras/i18n/en.d.ts +52 -0
- package/dist/extras/i18n/en.d.ts.map +1 -0
- package/dist/extras/i18n/en.js +51 -0
- package/dist/extras/i18n/index.d.ts +5 -0
- package/dist/extras/i18n/index.d.ts.map +1 -0
- package/dist/extras/i18n/index.js +3 -0
- package/dist/extras/i18n/messages.d.ts +37 -0
- package/dist/extras/i18n/messages.d.ts.map +1 -0
- package/dist/extras/i18n/messages.js +1 -0
- package/dist/extras/i18n/provider.d.ts +126 -0
- package/dist/extras/i18n/provider.d.ts.map +1 -0
- package/dist/extras/i18n/provider.js +44 -0
- package/dist/extras/i18n/ro.d.ts +52 -0
- package/dist/extras/i18n/ro.d.ts.map +1 -0
- package/dist/extras/i18n/ro.js +51 -0
- package/dist/extras/index.d.ts +7 -0
- package/dist/extras/index.d.ts.map +1 -0
- package/dist/extras/index.js +6 -0
- package/dist/extras/provider.d.ts +2 -0
- package/dist/extras/provider.d.ts.map +1 -0
- package/dist/extras/provider.js +1 -0
- package/dist/extras/query-keys.d.ts +16 -0
- package/dist/extras/query-keys.d.ts.map +1 -0
- package/dist/extras/query-keys.js +8 -0
- package/dist/extras/query-options.d.ts +455 -0
- package/dist/extras/query-options.d.ts.map +1 -0
- package/dist/extras/query-options.js +44 -0
- package/dist/extras/schemas.d.ts +416 -0
- package/dist/extras/schemas.d.ts.map +1 -0
- package/dist/extras/schemas.js +89 -0
- package/dist/extras/ui.d.ts +4 -0
- package/dist/extras/ui.d.ts.map +1 -0
- package/dist/extras/ui.js +3 -0
- package/dist/extras.d.ts +10 -0
- package/dist/extras.d.ts.map +1 -0
- package/dist/extras.js +9 -0
- package/dist/hooks/index.d.ts +36 -0
- package/dist/hooks/index.d.ts.map +1 -0
- package/dist/hooks/index.js +35 -0
- package/dist/hooks/use-booking-action-ledger.d.ts +63 -0
- package/dist/hooks/use-booking-action-ledger.d.ts.map +1 -0
- package/dist/hooks/use-booking-action-ledger.js +34 -0
- package/dist/hooks/use-booking-activity.d.ts +17 -0
- package/dist/hooks/use-booking-activity.d.ts.map +1 -0
- package/dist/hooks/use-booking-activity.js +12 -0
- package/dist/hooks/use-booking-cancel-mutation.d.ts +69 -0
- package/dist/hooks/use-booking-cancel-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-cancel-mutation.js +24 -0
- package/dist/hooks/use-booking-contract-generation.d.ts +31 -0
- package/dist/hooks/use-booking-contract-generation.d.ts.map +1 -0
- package/dist/hooks/use-booking-contract-generation.js +36 -0
- package/dist/hooks/use-booking-convert-mutation.d.ts +81 -0
- package/dist/hooks/use-booking-convert-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-convert-mutation.js +24 -0
- package/dist/hooks/use-booking-create-mutation.d.ts +337 -0
- package/dist/hooks/use-booking-create-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-create-mutation.js +43 -0
- package/dist/hooks/use-booking-documents.d.ts +41 -0
- package/dist/hooks/use-booking-documents.d.ts.map +1 -0
- package/dist/hooks/use-booking-documents.js +46 -0
- package/dist/hooks/use-booking-dual-create-mutation.d.ts +338 -0
- package/dist/hooks/use-booking-dual-create-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-dual-create-mutation.js +45 -0
- package/dist/hooks/use-booking-group-for-booking.d.ts +24 -0
- package/dist/hooks/use-booking-group-for-booking.d.ts.map +1 -0
- package/dist/hooks/use-booking-group-for-booking.js +12 -0
- package/dist/hooks/use-booking-group-member-mutation.d.ts +27 -0
- package/dist/hooks/use-booking-group-member-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-group-member-mutation.js +38 -0
- package/dist/hooks/use-booking-group-mutation.d.ts +40 -0
- package/dist/hooks/use-booking-group-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-group-mutation.js +32 -0
- package/dist/hooks/use-booking-group.d.ts +85 -0
- package/dist/hooks/use-booking-group.d.ts.map +1 -0
- package/dist/hooks/use-booking-group.js +12 -0
- package/dist/hooks/use-booking-groups.d.ts +21 -0
- package/dist/hooks/use-booking-groups.d.ts.map +1 -0
- package/dist/hooks/use-booking-groups.js +12 -0
- package/dist/hooks/use-booking-item-mutation.d.ts +101 -0
- package/dist/hooks/use-booking-item-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-item-mutation.js +42 -0
- package/dist/hooks/use-booking-item-travelers.d.ts +32 -0
- package/dist/hooks/use-booking-item-travelers.d.ts.map +1 -0
- package/dist/hooks/use-booking-item-travelers.js +48 -0
- package/dist/hooks/use-booking-items.d.ts +36 -0
- package/dist/hooks/use-booking-items.d.ts.map +1 -0
- package/dist/hooks/use-booking-items.js +12 -0
- package/dist/hooks/use-booking-mutation.d.ts +158 -0
- package/dist/hooks/use-booking-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-mutation.js +39 -0
- package/dist/hooks/use-booking-note-mutation.d.ts +39 -0
- package/dist/hooks/use-booking-note-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-note-mutation.js +44 -0
- package/dist/hooks/use-booking-notes.d.ts +15 -0
- package/dist/hooks/use-booking-notes.d.ts.map +1 -0
- package/dist/hooks/use-booking-notes.js +12 -0
- package/dist/hooks/use-booking-primary-product.d.ts +28 -0
- package/dist/hooks/use-booking-primary-product.d.ts.map +1 -0
- package/dist/hooks/use-booking-primary-product.js +20 -0
- package/dist/hooks/use-booking-status-mutation.d.ts +156 -0
- package/dist/hooks/use-booking-status-mutation.d.ts.map +1 -0
- package/dist/hooks/use-booking-status-mutation.js +54 -0
- package/dist/hooks/use-booking-tax-preview.d.ts +29 -0
- package/dist/hooks/use-booking-tax-preview.d.ts.map +1 -0
- package/dist/hooks/use-booking-tax-preview.js +21 -0
- package/dist/hooks/use-booking.d.ts +67 -0
- package/dist/hooks/use-booking.d.ts.map +1 -0
- package/dist/hooks/use-booking.js +12 -0
- package/dist/hooks/use-bookings.d.ts +71 -0
- package/dist/hooks/use-bookings.d.ts.map +1 -0
- package/dist/hooks/use-bookings.js +12 -0
- package/dist/hooks/use-pricing-preview.d.ts +61 -0
- package/dist/hooks/use-pricing-preview.d.ts.map +1 -0
- package/dist/hooks/use-pricing-preview.js +18 -0
- package/dist/hooks/use-public-booking-session-flow-mutation.d.ts +148 -0
- package/dist/hooks/use-public-booking-session-flow-mutation.d.ts.map +1 -0
- package/dist/hooks/use-public-booking-session-flow-mutation.js +35 -0
- package/dist/hooks/use-public-booking-session-state.d.ts +16 -0
- package/dist/hooks/use-public-booking-session-state.d.ts.map +1 -0
- package/dist/hooks/use-public-booking-session-state.js +12 -0
- package/dist/hooks/use-public-booking-session.d.ts +101 -0
- package/dist/hooks/use-public-booking-session.d.ts.map +1 -0
- package/dist/hooks/use-public-booking-session.js +12 -0
- package/dist/hooks/use-reveal-traveler.d.ts +54 -0
- package/dist/hooks/use-reveal-traveler.d.ts.map +1 -0
- package/dist/hooks/use-reveal-traveler.js +18 -0
- package/dist/hooks/use-sharing-groups.d.ts +41 -0
- package/dist/hooks/use-sharing-groups.d.ts.map +1 -0
- package/dist/hooks/use-sharing-groups.js +20 -0
- package/dist/hooks/use-supplier-status-mutation.d.ts +46 -0
- package/dist/hooks/use-supplier-status-mutation.d.ts.map +1 -0
- package/dist/hooks/use-supplier-status-mutation.js +39 -0
- package/dist/hooks/use-supplier-statuses.d.ts +20 -0
- package/dist/hooks/use-supplier-statuses.d.ts.map +1 -0
- package/dist/hooks/use-supplier-statuses.js +12 -0
- package/dist/hooks/use-traveler-mutation.d.ts +55 -0
- package/dist/hooks/use-traveler-mutation.d.ts.map +1 -0
- package/dist/hooks/use-traveler-mutation.js +42 -0
- package/dist/hooks/use-traveler-with-travel-details-mutation.d.ts +120 -0
- package/dist/hooks/use-traveler-with-travel-details-mutation.d.ts.map +1 -0
- package/dist/hooks/use-traveler-with-travel-details-mutation.js +43 -0
- package/dist/hooks/use-travelers.d.ts +23 -0
- package/dist/hooks/use-travelers.d.ts.map +1 -0
- package/dist/hooks/use-travelers.js +12 -0
- package/dist/i18n/en-base.d.ts +295 -0
- package/dist/i18n/en-base.d.ts.map +1 -0
- package/dist/i18n/en-base.js +294 -0
- package/dist/i18n/en-create-list.d.ts +327 -0
- package/dist/i18n/en-create-list.d.ts.map +1 -0
- package/dist/i18n/en-create-list.js +326 -0
- package/dist/i18n/en-journey.d.ts +229 -0
- package/dist/i18n/en-journey.d.ts.map +1 -0
- package/dist/i18n/en-journey.js +228 -0
- package/dist/i18n/en-operations.d.ts +382 -0
- package/dist/i18n/en-operations.d.ts.map +1 -0
- package/dist/i18n/en-operations.js +381 -0
- package/dist/i18n/en-sections.d.ts +360 -0
- package/dist/i18n/en-sections.d.ts.map +1 -0
- package/dist/i18n/en-sections.js +359 -0
- package/dist/i18n/en.d.ts +1581 -0
- package/dist/i18n/en.d.ts.map +1 -0
- package/dist/i18n/en.js +12 -0
- package/dist/i18n/index.d.ts +5 -0
- package/dist/i18n/index.d.ts.map +1 -0
- package/dist/i18n/index.js +3 -0
- package/dist/i18n/messages-base.d.ts +251 -0
- package/dist/i18n/messages-base.d.ts.map +1 -0
- package/dist/i18n/messages-base.js +1 -0
- package/dist/i18n/messages-create-list.d.ts +310 -0
- package/dist/i18n/messages-create-list.d.ts.map +1 -0
- package/dist/i18n/messages-create-list.js +1 -0
- package/dist/i18n/messages-journey.d.ts +198 -0
- package/dist/i18n/messages-journey.d.ts.map +1 -0
- package/dist/i18n/messages-journey.js +1 -0
- package/dist/i18n/messages-operations.d.ts +362 -0
- package/dist/i18n/messages-operations.d.ts.map +1 -0
- package/dist/i18n/messages-operations.js +1 -0
- package/dist/i18n/messages-sections.d.ts +312 -0
- package/dist/i18n/messages-sections.d.ts.map +1 -0
- package/dist/i18n/messages-sections.js +1 -0
- package/dist/i18n/messages.d.ts +7 -0
- package/dist/i18n/messages.d.ts.map +1 -0
- package/dist/i18n/messages.js +1 -0
- package/dist/i18n/provider.d.ts +3185 -0
- package/dist/i18n/provider.d.ts.map +1 -0
- package/dist/i18n/provider.js +45 -0
- package/dist/i18n/ro-base.d.ts +295 -0
- package/dist/i18n/ro-base.d.ts.map +1 -0
- package/dist/i18n/ro-base.js +294 -0
- package/dist/i18n/ro-create-list.d.ts +327 -0
- package/dist/i18n/ro-create-list.d.ts.map +1 -0
- package/dist/i18n/ro-create-list.js +326 -0
- package/dist/i18n/ro-journey.d.ts +229 -0
- package/dist/i18n/ro-journey.d.ts.map +1 -0
- package/dist/i18n/ro-journey.js +228 -0
- package/dist/i18n/ro-operations.d.ts +382 -0
- package/dist/i18n/ro-operations.d.ts.map +1 -0
- package/dist/i18n/ro-operations.js +381 -0
- package/dist/i18n/ro-sections.d.ts +360 -0
- package/dist/i18n/ro-sections.d.ts.map +1 -0
- package/dist/i18n/ro-sections.js +359 -0
- package/dist/i18n/ro.d.ts +1581 -0
- package/dist/i18n/ro.d.ts.map +1 -0
- package/dist/i18n/ro.js +12 -0
- package/dist/index.d.ts +9 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +8 -0
- package/dist/journey/components/booking-journey-rules.d.ts +48 -0
- package/dist/journey/components/booking-journey-rules.d.ts.map +1 -0
- package/dist/journey/components/booking-journey-rules.js +235 -0
- package/dist/journey/components/booking-journey.d.ts +3 -0
- package/dist/journey/components/booking-journey.d.ts.map +1 -0
- package/dist/journey/components/booking-journey.js +368 -0
- package/dist/journey/components/configure-step-skeleton.d.ts +8 -0
- package/dist/journey/components/configure-step-skeleton.d.ts.map +1 -0
- package/dist/journey/components/configure-step-skeleton.js +11 -0
- package/dist/journey/components/contract-preview-dialog.d.ts +47 -0
- package/dist/journey/components/contract-preview-dialog.d.ts.map +1 -0
- package/dist/journey/components/contract-preview-dialog.js +124 -0
- package/dist/journey/components/journey-steps/accommodation-step.d.ts +3 -0
- package/dist/journey/components/journey-steps/accommodation-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/accommodation-step.js +71 -0
- package/dist/journey/components/journey-steps/addons-step.d.ts +3 -0
- package/dist/journey/components/journey-steps/addons-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/addons-step.js +40 -0
- package/dist/journey/components/journey-steps/billing-step.d.ts +8 -0
- package/dist/journey/components/journey-steps/billing-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/billing-step.js +78 -0
- package/dist/journey/components/journey-steps/configure-steps.d.ts +28 -0
- package/dist/journey/components/journey-steps/configure-steps.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/configure-steps.js +232 -0
- package/dist/journey/components/journey-steps/documents-step.d.ts +11 -0
- package/dist/journey/components/journey-steps/documents-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/documents-step.js +36 -0
- package/dist/journey/components/journey-steps/payment-step.d.ts +29 -0
- package/dist/journey/components/journey-steps/payment-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/payment-step.js +225 -0
- package/dist/journey/components/journey-steps/review-step.d.ts +27 -0
- package/dist/journey/components/journey-steps/review-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/review-step.js +18 -0
- package/dist/journey/components/journey-steps/shared.d.ts +75 -0
- package/dist/journey/components/journey-steps/shared.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/shared.js +108 -0
- package/dist/journey/components/journey-steps/travelers-step.d.ts +7 -0
- package/dist/journey/components/journey-steps/travelers-step.d.ts.map +1 -0
- package/dist/journey/components/journey-steps/travelers-step.js +201 -0
- package/dist/journey/components/journey-steps.d.ts +21 -0
- package/dist/journey/components/journey-steps.d.ts.map +1 -0
- package/dist/journey/components/journey-steps.js +20 -0
- package/dist/journey/components/side-panel.d.ts +17 -0
- package/dist/journey/components/side-panel.d.ts.map +1 -0
- package/dist/journey/components/side-panel.js +245 -0
- package/dist/journey/components/stacked-journey.d.ts +30 -0
- package/dist/journey/components/stacked-journey.d.ts.map +1 -0
- package/dist/journey/components/stacked-journey.js +50 -0
- package/dist/journey/components/step-header.d.ts +7 -0
- package/dist/journey/components/step-header.d.ts.map +1 -0
- package/dist/journey/components/step-header.js +12 -0
- package/dist/journey/index.d.ts +18 -0
- package/dist/journey/index.d.ts.map +1 -0
- package/dist/journey/index.js +17 -0
- package/dist/journey/lib/draft-state.d.ts +35 -0
- package/dist/journey/lib/draft-state.d.ts.map +1 -0
- package/dist/journey/lib/draft-state.js +57 -0
- package/dist/journey/lib/pax-band-dependencies.d.ts +27 -0
- package/dist/journey/lib/pax-band-dependencies.d.ts.map +1 -0
- package/dist/journey/lib/pax-band-dependencies.js +50 -0
- package/dist/journey/lib/payment-schedule.d.ts +19 -0
- package/dist/journey/lib/payment-schedule.d.ts.map +1 -0
- package/dist/journey/lib/payment-schedule.js +90 -0
- package/dist/journey/types.d.ts +403 -0
- package/dist/journey/types.d.ts.map +1 -0
- package/dist/journey/types.js +19 -0
- package/dist/provider.d.ts +2 -0
- package/dist/provider.d.ts.map +1 -0
- package/dist/provider.js +1 -0
- package/dist/query-keys.d.ts +74 -0
- package/dist/query-keys.d.ts.map +1 -0
- package/dist/query-keys.js +26 -0
- package/dist/query-options.d.ts +2534 -0
- package/dist/query-options.d.ts.map +1 -0
- package/dist/query-options.js +233 -0
- package/dist/requirements/client.d.ts +14 -0
- package/dist/requirements/client.d.ts.map +1 -0
- package/dist/requirements/client.js +59 -0
- package/dist/requirements/components/booking-requirements-contact-tab.d.ts +8 -0
- package/dist/requirements/components/booking-requirements-contact-tab.d.ts.map +1 -0
- package/dist/requirements/components/booking-requirements-contact-tab.js +8 -0
- package/dist/requirements/components/booking-requirements-questions-tab.d.ts +14 -0
- package/dist/requirements/components/booking-requirements-questions-tab.d.ts.map +1 -0
- package/dist/requirements/components/booking-requirements-questions-tab.js +17 -0
- package/dist/requirements/constants.d.ts +114 -0
- package/dist/requirements/constants.d.ts.map +1 -0
- package/dist/requirements/constants.js +45 -0
- package/dist/requirements/hooks/index.d.ts +6 -0
- package/dist/requirements/hooks/index.d.ts.map +1 -0
- package/dist/requirements/hooks/index.js +6 -0
- package/dist/requirements/hooks/use-booking-questions.d.ts +24 -0
- package/dist/requirements/hooks/use-booking-questions.d.ts.map +1 -0
- package/dist/requirements/hooks/use-booking-questions.js +9 -0
- package/dist/requirements/hooks/use-contact-requirements.d.ts +22 -0
- package/dist/requirements/hooks/use-contact-requirements.d.ts.map +1 -0
- package/dist/requirements/hooks/use-contact-requirements.js +9 -0
- package/dist/requirements/hooks/use-products.d.ts +16 -0
- package/dist/requirements/hooks/use-products.d.ts.map +1 -0
- package/dist/requirements/hooks/use-products.js +9 -0
- package/dist/requirements/hooks/use-question-options.d.ts +19 -0
- package/dist/requirements/hooks/use-question-options.d.ts.map +1 -0
- package/dist/requirements/hooks/use-question-options.js +9 -0
- package/dist/requirements/hooks/use-transport-requirements.d.ts +30 -0
- package/dist/requirements/hooks/use-transport-requirements.d.ts.map +1 -0
- package/dist/requirements/hooks/use-transport-requirements.js +9 -0
- package/dist/requirements/i18n/en.d.ts +94 -0
- package/dist/requirements/i18n/en.d.ts.map +1 -0
- package/dist/requirements/i18n/en.js +93 -0
- package/dist/requirements/i18n/index.d.ts +5 -0
- package/dist/requirements/i18n/index.d.ts.map +1 -0
- package/dist/requirements/i18n/index.js +3 -0
- package/dist/requirements/i18n/messages.d.ts +59 -0
- package/dist/requirements/i18n/messages.d.ts.map +1 -0
- package/dist/requirements/i18n/messages.js +1 -0
- package/dist/requirements/i18n/provider.d.ts +210 -0
- package/dist/requirements/i18n/provider.d.ts.map +1 -0
- package/dist/requirements/i18n/provider.js +44 -0
- package/dist/requirements/i18n/ro.d.ts +94 -0
- package/dist/requirements/i18n/ro.d.ts.map +1 -0
- package/dist/requirements/i18n/ro.js +93 -0
- package/dist/requirements/index.d.ts +9 -0
- package/dist/requirements/index.d.ts.map +1 -0
- package/dist/requirements/index.js +8 -0
- package/dist/requirements/provider.d.ts +2 -0
- package/dist/requirements/provider.d.ts.map +1 -0
- package/dist/requirements/provider.js +1 -0
- package/dist/requirements/query-keys.d.ts +33 -0
- package/dist/requirements/query-keys.d.ts.map +1 -0
- package/dist/requirements/query-keys.js +13 -0
- package/dist/requirements/query-options.d.ts +371 -0
- package/dist/requirements/query-options.d.ts.map +1 -0
- package/dist/requirements/query-options.js +80 -0
- package/dist/requirements/schemas.d.ts +320 -0
- package/dist/requirements/schemas.d.ts.map +1 -0
- package/dist/requirements/schemas.js +121 -0
- package/dist/requirements/ui.d.ts +4 -0
- package/dist/requirements/ui.d.ts.map +1 -0
- package/dist/requirements/ui.js +3 -0
- package/dist/requirements/utils.d.ts +2 -0
- package/dist/requirements/utils.d.ts.map +1 -0
- package/dist/requirements/utils.js +4 -0
- package/dist/schemas.d.ts +2070 -0
- package/dist/schemas.d.ts.map +1 -0
- package/dist/schemas.js +507 -0
- package/dist/status-presentation.d.ts +34 -0
- package/dist/status-presentation.d.ts.map +1 -0
- package/dist/status-presentation.js +38 -0
- package/dist/ui.d.ts +43 -0
- package/dist/ui.d.ts.map +1 -0
- package/dist/ui.js +42 -0
- package/package.json +256 -0
- package/src/requirements/styles.css +1 -0
- package/src/styles.css +13 -0
|
@@ -0,0 +1,264 @@
|
|
|
1
|
+
// agent-quality: file-size exception -- owner: bookings-react; existing UI surface stays co-located until a dedicated split preserves behavior and tests.
|
|
2
|
+
"use client";
|
|
3
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
4
|
+
import { useOrganization, usePerson } from "@voyant-travel/relationships-react";
|
|
5
|
+
import { AlertDialog, AlertDialogAction, AlertDialogCancel, AlertDialogContent, AlertDialogDescription, AlertDialogFooter, AlertDialogHeader, AlertDialogTitle, Badge, Button, Card, CardContent, cn, } from "@voyant-travel/ui/components";
|
|
6
|
+
import { Collapsible, CollapsibleContent, CollapsibleTrigger, } from "@voyant-travel/ui/components/collapsible";
|
|
7
|
+
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@voyant-travel/ui/components/tabs";
|
|
8
|
+
import { Tooltip, TooltipContent, TooltipTrigger } from "@voyant-travel/ui/components/tooltip";
|
|
9
|
+
import { Ban, ChevronDown, ChevronRight, CreditCard, Info, Mail, MapPin, Pencil, Phone, Plus, RefreshCw, Trash2, } from "lucide-react";
|
|
10
|
+
import { Fragment, useState } from "react";
|
|
11
|
+
import { formatMessage, useBookingsUiI18nOrDefault, useBookingsUiMessagesOrDefault, } from "../i18n/index.js";
|
|
12
|
+
import { useBooking, useBookingItems, useBookingMutation } from "../index.js";
|
|
13
|
+
import { BookingActivityTimeline } from "./booking-activity-timeline.js";
|
|
14
|
+
import { BookingBillingDialog } from "./booking-billing-dialog.js";
|
|
15
|
+
import { BookingCancellationDialog } from "./booking-cancellation-dialog.js";
|
|
16
|
+
import { BookingDialog } from "./booking-dialog.js";
|
|
17
|
+
import { BookingGroupSection } from "./booking-group-section.js";
|
|
18
|
+
import { BookingGuaranteeList } from "./booking-guarantee-list.js";
|
|
19
|
+
import { BookingItemList } from "./booking-item-list.js";
|
|
20
|
+
import { BookingNotes } from "./booking-notes.js";
|
|
21
|
+
import { BookingPaymentScheduleList } from "./booking-payment-schedule-list.js";
|
|
22
|
+
import { BookingPaymentsSummary, } from "./booking-payments-summary.js";
|
|
23
|
+
import { StatusBadge } from "./status-badge.js";
|
|
24
|
+
import { StatusChangeDialog } from "./status-change-dialog.js";
|
|
25
|
+
import { SupplierStatusList } from "./supplier-status-list.js";
|
|
26
|
+
import { TravelerList } from "./traveler-list.js";
|
|
27
|
+
export function BookingDetailPage({ id, className, locale, hideBreadcrumb, onBack, onRecordPayment, recordPaymentDisabledReason, addScheduleDisabledReason, paidAmountCents, onItemResourceOpen, onPersonOpen, onOrganizationOpen, onInvoiceOpen, onViewPayment, onEditPayment, onDeletePayment, activeTab, onTabChange, slots, }) {
|
|
28
|
+
const i18n = useBookingsUiI18nOrDefault();
|
|
29
|
+
const messages = useBookingsUiMessagesOrDefault();
|
|
30
|
+
const detailMessages = messages.bookingDetailPage;
|
|
31
|
+
const resolvedLocale = locale ?? i18n.locale;
|
|
32
|
+
const [editOpen, setEditOpen] = useState(false);
|
|
33
|
+
const [statusDialogOpen, setStatusDialogOpen] = useState(false);
|
|
34
|
+
const [cancelDialogOpen, setCancelDialogOpen] = useState(false);
|
|
35
|
+
const [deleteDialogOpen, setDeleteDialogOpen] = useState(false);
|
|
36
|
+
const { data: bookingData, isPending } = useBooking(id);
|
|
37
|
+
const { remove } = useBookingMutation();
|
|
38
|
+
const headerPersonId = bookingData?.data?.personId ?? null;
|
|
39
|
+
const headerOrganizationId = bookingData?.data?.organizationId ?? null;
|
|
40
|
+
const headerPerson = usePerson(headerPersonId ?? undefined, {
|
|
41
|
+
enabled: Boolean(headerPersonId),
|
|
42
|
+
}).data;
|
|
43
|
+
const headerOrganization = useOrganization(headerOrganizationId ?? undefined, {
|
|
44
|
+
enabled: Boolean(headerOrganizationId) && !headerPersonId,
|
|
45
|
+
}).data;
|
|
46
|
+
// Pull booking items so the subtitle can link to the primary product
|
|
47
|
+
// + availability slot. We pick the first item (or the one flagged
|
|
48
|
+
// `isPrimary` when present) — most bookings only have one product.
|
|
49
|
+
const headerItems = useBookingItems(id).data?.data ?? [];
|
|
50
|
+
const primaryItem = headerItems.find((item) => item.isPrimary) ??
|
|
51
|
+
headerItems[0] ??
|
|
52
|
+
null;
|
|
53
|
+
if (isPending) {
|
|
54
|
+
return (_jsx("div", { className: cn("flex items-center justify-center py-12", className), children: _jsx("p", { className: "text-sm text-muted-foreground", children: messages.common.loading }) }));
|
|
55
|
+
}
|
|
56
|
+
const booking = bookingData?.data;
|
|
57
|
+
if (!booking) {
|
|
58
|
+
return (_jsxs("div", { className: cn("flex flex-col items-center justify-center gap-4 py-12", className), children: [_jsx("p", { className: "text-muted-foreground", children: detailMessages.notFound }), onBack ? (_jsx(Button, { variant: "outline", onClick: onBack, children: detailMessages.backToBookings })) : null] }));
|
|
59
|
+
}
|
|
60
|
+
const canCancel = ["draft", "on_hold", "confirmed", "in_progress"].includes(booking.status);
|
|
61
|
+
const sellHint = booking.priceOverride?.isManual
|
|
62
|
+
? `${detailMessages.summaryPriceOverride}: ${booking.priceOverride.reason}`
|
|
63
|
+
: undefined;
|
|
64
|
+
const billingPersonName = [booking.contactFirstName, booking.contactLastName].filter(Boolean).join(" ") ||
|
|
65
|
+
(headerPerson
|
|
66
|
+
? [headerPerson.firstName, headerPerson.lastName].filter(Boolean).join(" ")
|
|
67
|
+
: "") ||
|
|
68
|
+
headerOrganization?.name ||
|
|
69
|
+
"";
|
|
70
|
+
const billingHref = headerPersonId
|
|
71
|
+
? () => onPersonOpen?.(headerPersonId)
|
|
72
|
+
: headerOrganizationId
|
|
73
|
+
? () => onOrganizationOpen?.(headerOrganizationId)
|
|
74
|
+
: null;
|
|
75
|
+
const billingClickable = billingHref && (headerPersonId ? Boolean(onPersonOpen) : Boolean(onOrganizationOpen));
|
|
76
|
+
const headerDateRange = booking.startDate
|
|
77
|
+
? `${formatDate(booking.startDate, resolvedLocale, detailMessages.noValue)} - ${formatDate(booking.endDate, resolvedLocale, detailMessages.noValue)}`
|
|
78
|
+
: null;
|
|
79
|
+
const headerPax = booking.pax != null ? `${booking.pax} PAX` : null;
|
|
80
|
+
const headerProductTitle = primaryItem?.productNameSnapshot ?? primaryItem?.title ?? null;
|
|
81
|
+
const headerProductId = primaryItem?.productId ?? null;
|
|
82
|
+
const headerSlotId = primaryItem?.availabilitySlotId ?? null;
|
|
83
|
+
const headerSubtitleParts = [
|
|
84
|
+
billingPersonName ? (billingClickable ? (_jsx("button", { type: "button", onClick: billingHref, className: "hover:text-foreground hover:underline", children: billingPersonName }, "billing")) : (_jsx("span", { children: billingPersonName }, "billing"))) : null,
|
|
85
|
+
headerProductTitle ? (headerProductId && onItemResourceOpen ? (_jsx("button", { type: "button", onClick: () => onItemResourceOpen("product", headerProductId), className: "inline-block max-w-[18rem] truncate align-bottom hover:text-foreground hover:underline", title: headerProductTitle, children: headerProductTitle }, "product")) : (_jsx("span", { className: "inline-block max-w-[18rem] truncate align-bottom", title: headerProductTitle, children: headerProductTitle }, "product"))) : null,
|
|
86
|
+
headerDateRange ? (headerSlotId && onItemResourceOpen ? (_jsx("button", { type: "button", onClick: () => onItemResourceOpen("availabilitySlot", headerSlotId), className: "hover:text-foreground hover:underline", children: headerDateRange }, "dates")) : (_jsx("span", { children: headerDateRange }, "dates"))) : null,
|
|
87
|
+
headerPax ? _jsx("span", { children: headerPax }, "pax") : null,
|
|
88
|
+
].filter(Boolean);
|
|
89
|
+
return (_jsxs("div", { "data-slot": "booking-detail-page", className: cn("flex flex-col gap-6 p-6", className), children: [hideBreadcrumb ? null : (_jsxs("div", { className: "flex items-center gap-1.5 text-sm text-muted-foreground", children: [onBack ? (_jsx("button", { type: "button", onClick: onBack, className: "transition-colors hover:text-foreground", children: detailMessages.breadcrumbBookings })) : (_jsx("span", { children: detailMessages.breadcrumbBookings })), _jsx(ChevronRight, { className: "h-3.5 w-3.5", "aria-hidden": "true" }), _jsx("span", { className: "font-normal text-foreground", children: booking.bookingNumber })] })), _jsxs("div", { className: "flex items-start justify-between", children: [_jsxs("div", { className: "flex flex-col gap-1", children: [_jsxs("div", { className: "flex items-center gap-3", children: [_jsx("h1", { className: "text-2xl font-bold tracking-tight", children: booking.bookingNumber }), _jsx(StatusBadge, { status: booking.status, children: getBookingStatusLabel(booking.status, messages.common.bookingStatusLabels) })] }), headerSubtitleParts.length > 0 ? (_jsx("div", { className: "flex flex-wrap items-center gap-1.5 text-sm text-muted-foreground", children: headerSubtitleParts.map((part, idx) => (_jsxs(Fragment, { children: [idx > 0 ? _jsx("span", { className: "text-muted-foreground/60", children: "/" }) : null, part] }, idx))) })) : null] }), _jsxs("div", { className: "flex items-center gap-2", children: [_jsxs(Button, { variant: "outline", size: "sm", onClick: () => setEditOpen(true), children: [_jsx(Pencil, { className: "mr-1.5 h-3.5 w-3.5", "aria-hidden": "true" }), detailMessages.editAction] }), _jsxs(Button, { variant: "outline", size: "sm", onClick: () => setStatusDialogOpen(true), children: [_jsx(RefreshCw, { className: "mr-1.5 h-3.5 w-3.5", "aria-hidden": "true" }), detailMessages.changeStatusAction] }), canCancel ? (_jsxs(Button, { variant: "outline", size: "sm", onClick: () => setCancelDialogOpen(true), children: [_jsx(Ban, { className: "mr-1.5 h-3.5 w-3.5", "aria-hidden": "true" }), detailMessages.cancelBookingAction] })) : null, _jsxs(Button, { variant: "outline", size: "sm", className: "text-destructive hover:bg-destructive/10 hover:text-destructive", disabled: remove.isPending, onClick: () => setDeleteDialogOpen(true), children: [_jsx(Trash2, { className: "mr-1.5 h-3.5 w-3.5", "aria-hidden": "true" }), detailMessages.deleteAction] })] })] }), slots?.header?.(booking), _jsxs("div", { className: "grid grid-cols-2 gap-4 sm:grid-cols-4", children: [_jsx(StatCard, { label: detailMessages.summaryTotal, hint: sellHint, children: formatAmount(booking.sellAmountCents, booking.sellCurrency, resolvedLocale, detailMessages.noValue) }), _jsx(StatCard, { label: detailMessages.summaryPaid, badge: paidAmountCents != null && booking.sellAmountCents
|
|
90
|
+
? renderPercentBadge(Math.round((paidAmountCents / booking.sellAmountCents) * 100), paidBadgeClass)
|
|
91
|
+
: null, children: paidAmountCents != null
|
|
92
|
+
? formatAmount(paidAmountCents, booking.sellCurrency, resolvedLocale, detailMessages.noValue)
|
|
93
|
+
: detailMessages.noValue }), _jsx(StatCard, { label: detailMessages.summaryCostMargin, badge: booking.marginPercent != null
|
|
94
|
+
? renderPercentBadge(booking.marginPercent, marginBadgeClass)
|
|
95
|
+
: null, children: formatAmount(booking.costAmountCents, booking.sellCurrency, resolvedLocale, detailMessages.noValue) }), _jsx(StatCard, { label: detailMessages.summaryCreated, children: formatDate(booking.createdAt, resolvedLocale, detailMessages.noValue) })] }), slots?.afterSummary?.(booking), _jsxs(Tabs, { defaultValue: "items", value: activeTab, onValueChange: (value) => onTabChange?.(String(value)), children: [_jsxs(TabsList, { className: "w-full justify-start", children: [_jsx(TabsTrigger, { value: "items", children: detailMessages.tabOverview }), _jsx(TabsTrigger, { value: "travelers", children: detailMessages.tabTravelers }), _jsx(TabsTrigger, { value: "finance", children: detailMessages.tabFinance }), slots?.invoicesTab ? (_jsx(TabsTrigger, { value: "invoices", children: slots.invoicesTab.label ?? detailMessages.tabInvoices })) : null, _jsx(TabsTrigger, { value: "documents", children: detailMessages.tabDocuments }), _jsx(TabsTrigger, { value: "suppliers", children: detailMessages.tabSuppliers }), _jsx(TabsTrigger, { value: "activity", children: detailMessages.tabActivity }), _jsx(TabsTrigger, { value: "metadata", children: detailMessages.tabMetadata })] }), _jsxs(TabsContent, { value: "items", className: "mt-4 flex flex-col gap-6", children: [slots?.overviewStart?.(booking), _jsx(BookingItemList, { bookingId: id, onResourceOpen: onItemResourceOpen }), _jsx(BookingGroupSection, { bookingId: id }), visibleInternalNotes(booking.internalNotes) ? (_jsx(Card, { children: _jsxs(CardContent, { className: "py-5", children: [_jsx("p", { className: "mb-1 text-xs font-medium text-muted-foreground", children: detailMessages.internalNotesLabel }), _jsx("p", { className: "whitespace-pre-wrap text-sm", children: visibleInternalNotes(booking.internalNotes) })] }) })) : null, slots?.overviewEnd?.(booking)] }), _jsxs(TabsContent, { value: "travelers", className: "mt-4 flex flex-col gap-6", children: [slots?.travelersStart?.(booking), _jsx(BookingBillingContextCard, { booking: booking, onPersonOpen: onPersonOpen, onOrganizationOpen: onOrganizationOpen }), _jsx(TravelerList, { bookingId: id, autoReveal: true })] }), _jsxs(TabsContent, { value: "finance", className: "mt-4 flex flex-col gap-6", children: [_jsx(BookingPaymentScheduleList, { bookingId: id, addScheduleDisabledReason: addScheduleDisabledReason ?? null }), slots?.paymentsContent ? (renderDetailSlot(slots.paymentsContent, booking)) : (_jsx(BookingPaymentsSummary, { bookingId: id, variant: "admin", onViewPayment: onViewPayment, onInvoiceOpen: onInvoiceOpen, onEditPayment: onEditPayment, onDeletePayment: onDeletePayment, headerAction: onRecordPayment ? (_jsx(RecordPaymentHeaderButton, { label: detailMessages.recordPaymentAction, disabledReason: recordPaymentDisabledReason ?? null, onClick: () => onRecordPayment(booking) })) : null })), slots?.financeStart?.(booking), _jsxs(Collapsible, { children: [_jsxs(CollapsibleTrigger, { className: "group flex w-full items-center justify-between rounded-md border bg-background px-4 py-3 text-sm font-semibold hover:bg-muted/30", children: [messages.bookingGuaranteeList.title, _jsx(ChevronDown, { className: "h-4 w-4 transition-transform group-data-panel-open:rotate-180" })] }), _jsx(CollapsibleContent, { className: "mt-3", children: _jsx(BookingGuaranteeList, { bookingId: id }) })] }), slots?.financeEnd?.(booking)] }), slots?.invoicesTab ? (_jsx(TabsContent, { value: "invoices", className: "mt-4 flex flex-col gap-6", children: renderDetailSlot(slots.invoicesTab.content, booking) })) : null, _jsx(TabsContent, { value: "documents", className: "mt-4 flex flex-col gap-4", children: slots?.documents ? (slots.documents(booking)) : (_jsx("p", { className: "rounded-md border border-dashed p-4 text-sm text-muted-foreground", children: detailMessages.documentsSlotEmpty })) }), _jsx(TabsContent, { value: "suppliers", className: "mt-4", children: _jsx(SupplierStatusList, { bookingId: id }) }), _jsxs(TabsContent, { value: "activity", className: "mt-4 flex flex-col gap-6", children: [_jsx(BookingNotes, { bookingId: id }), _jsx(BookingActivityTimeline, { bookingId: id, additionalEvents: slots?.activityExtraEvents, footer: slots?.activityTimelineFooter }), slots?.activityEnd?.(booking)] }), _jsx(TabsContent, { value: "metadata", className: "mt-4", children: _jsx(BookingMetadataList, { booking: booking, messages: detailMessages.metadataSection, statusLabel: getBookingStatusLabel(booking.status, messages.common.bookingStatusLabels), formatDateTime: i18n.formatDateTime, noValue: detailMessages.noValue }) })] }), _jsx(BookingDialog, { open: editOpen, onOpenChange: setEditOpen, booking: booking }), _jsx(StatusChangeDialog, { open: statusDialogOpen, onOpenChange: setStatusDialogOpen, bookingId: id, currentStatus: booking.status }), _jsx(BookingCancellationDialog, { open: cancelDialogOpen, onOpenChange: setCancelDialogOpen, booking: booking }), _jsx(AlertDialog, { open: deleteDialogOpen, onOpenChange: (next) => {
|
|
96
|
+
if (!next && !remove.isPending)
|
|
97
|
+
setDeleteDialogOpen(false);
|
|
98
|
+
}, children: _jsxs(AlertDialogContent, { size: "sm", children: [_jsxs(AlertDialogHeader, { children: [_jsx(AlertDialogTitle, { children: detailMessages.deleteConfirm }), _jsx(AlertDialogDescription, { children: booking.bookingNumber
|
|
99
|
+
? formatMessage(detailMessages.deleteConfirmDescription, {
|
|
100
|
+
number: booking.bookingNumber,
|
|
101
|
+
})
|
|
102
|
+
: detailMessages.deleteConfirmDescriptionFallback })] }), _jsxs(AlertDialogFooter, { children: [_jsx(AlertDialogCancel, { disabled: remove.isPending, children: detailMessages.deleteCancel }), _jsx(AlertDialogAction, { variant: "destructive", disabled: remove.isPending, onClick: async () => {
|
|
103
|
+
await remove.mutateAsync(id);
|
|
104
|
+
setDeleteDialogOpen(false);
|
|
105
|
+
onBack?.();
|
|
106
|
+
}, children: detailMessages.deleteConfirmAction })] })] }) })] }));
|
|
107
|
+
}
|
|
108
|
+
function RecordPaymentHeaderButton({ label, disabledReason, onClick, }) {
|
|
109
|
+
if (disabledReason) {
|
|
110
|
+
return (_jsxs(Tooltip, { children: [_jsx(TooltipTrigger, { render: _jsx("span", { tabIndex: 0, className: "inline-block" }), children: _jsxs(Button, { variant: "outline", size: "sm", disabled: true, className: "pointer-events-none", children: [_jsx(Plus, { className: "mr-2 h-4 w-4" }), label] }) }), _jsx(TooltipContent, { children: disabledReason })] }));
|
|
111
|
+
}
|
|
112
|
+
return (_jsxs(Button, { variant: "outline", size: "sm", onClick: onClick, children: [_jsx(Plus, { className: "mr-2 h-4 w-4" }), label] }));
|
|
113
|
+
}
|
|
114
|
+
function renderDetailSlot(content, booking) {
|
|
115
|
+
return typeof content === "function" ? content(booking) : content;
|
|
116
|
+
}
|
|
117
|
+
/**
|
|
118
|
+
* Billing/contact card for the Travelers tab. Snapshot fields on the
|
|
119
|
+
* booking row are the source of truth (they capture contact info at
|
|
120
|
+
* the time of booking). When they're empty — typically for bookings
|
|
121
|
+
* created via flows that don't snapshot — fall back to the linked CRM
|
|
122
|
+
* person / organization so the operator still sees who they're
|
|
123
|
+
* billing.
|
|
124
|
+
*/
|
|
125
|
+
export function BookingBillingContextCard({ booking, onPersonOpen, onOrganizationOpen, }) {
|
|
126
|
+
const messages = useBookingsUiMessagesOrDefault().bookingDetailPage;
|
|
127
|
+
const [editOpen, setEditOpen] = useState(false);
|
|
128
|
+
const person = usePerson(booking.personId ?? undefined, {
|
|
129
|
+
enabled: Boolean(booking.personId),
|
|
130
|
+
}).data;
|
|
131
|
+
const organization = useOrganization(booking.organizationId ?? undefined, {
|
|
132
|
+
enabled: Boolean(booking.organizationId) && !booking.personId,
|
|
133
|
+
}).data;
|
|
134
|
+
const payerName = [booking.contactFirstName, booking.contactLastName].filter(Boolean).join(" ") ||
|
|
135
|
+
(person ? [person.firstName, person.lastName].filter(Boolean).join(" ") : "") ||
|
|
136
|
+
organization?.name ||
|
|
137
|
+
"";
|
|
138
|
+
const email = booking.contactEmail ?? person?.email ?? null;
|
|
139
|
+
const phone = booking.contactPhone ?? person?.phone ?? null;
|
|
140
|
+
const taxId = booking.contactTaxId ?? organization?.taxId ?? null;
|
|
141
|
+
const address = [
|
|
142
|
+
booking.contactAddressLine1,
|
|
143
|
+
booking.contactAddressLine2,
|
|
144
|
+
booking.contactCity,
|
|
145
|
+
booking.contactRegion,
|
|
146
|
+
booking.contactPostalCode,
|
|
147
|
+
booking.contactCountry,
|
|
148
|
+
]
|
|
149
|
+
.filter(Boolean)
|
|
150
|
+
.join(", ");
|
|
151
|
+
return (_jsxs("div", { "data-slot": "booking-billing-context", className: "flex flex-col gap-3", children: [_jsxs("div", { className: "flex items-center justify-between", children: [_jsxs("h2", { className: "flex items-center gap-2 text-base font-semibold", children: [_jsx(CreditCard, { className: "h-4 w-4", "aria-hidden": "true" }), messages.billingPayer] }), _jsxs(Button, { variant: "outline", size: "sm", onClick: () => setEditOpen(true), children: [_jsx(Pencil, { className: "mr-1 h-3.5 w-3.5", "aria-hidden": "true" }), messages.editAction] })] }), _jsxs("div", { className: "flex flex-col gap-4 rounded-md border p-4", children: [_jsxs("div", { className: "grid gap-4 md:grid-cols-4", children: [_jsx(BillingField, { label: messages.billingPayer, value: payerName ? (booking.personId && onPersonOpen ? (_jsx("button", { type: "button", onClick: () => onPersonOpen(booking.personId), className: "text-left text-primary hover:underline", children: payerName })) : booking.organizationId && !booking.personId && onOrganizationOpen ? (_jsx("button", { type: "button", onClick: () => onOrganizationOpen(booking.organizationId), className: "text-left text-primary hover:underline", children: payerName })) : (payerName)) : (messages.noValue) }), _jsx(BillingField, { label: messages.billingTaxId, value: taxId ?? messages.noValue }), _jsx(BillingField, { label: messages.billingEmail, value: email ?? messages.noValue, icon: _jsx(Mail, { className: "h-3.5 w-3.5", "aria-hidden": "true" }) }), _jsx(BillingField, { label: messages.billingPhone, value: phone ?? messages.noValue, icon: _jsx(Phone, { className: "h-3.5 w-3.5", "aria-hidden": "true" }) })] }), _jsx(BillingField, { label: messages.billingAddress, value: address || messages.noValue, icon: _jsx(MapPin, { className: "h-3.5 w-3.5", "aria-hidden": "true" }) })] }), _jsx(BookingBillingDialog, { open: editOpen, onOpenChange: setEditOpen, booking: booking })] }));
|
|
152
|
+
}
|
|
153
|
+
function StatCard({ label, children, hint, badge, }) {
|
|
154
|
+
return (_jsx(Card, { children: _jsxs(CardContent, { className: "flex flex-col gap-1", children: [_jsx("div", { className: "text-xs font-medium text-muted-foreground", children: label }), _jsxs("div", { className: "flex flex-wrap items-center gap-2", children: [_jsx("div", { className: "text-xl font-semibold tabular-nums leading-none", children: children }), badge] }), hint ? _jsx("div", { className: "text-xs text-muted-foreground", children: hint }) : null] }) }));
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Definition-list metadata panel for the Metadata tab. Surfaces booking
|
|
158
|
+
* fields that don't fit any other tab but are still useful for support
|
|
159
|
+
* / debugging: raw id, booking number, status, communication language,
|
|
160
|
+
* created/updated timestamps. Label-left, value-right rows matching the
|
|
161
|
+
* supplier-status / documents / notes tab style.
|
|
162
|
+
*/
|
|
163
|
+
function BookingMetadataList({ booking, messages, statusLabel, formatDateTime, noValue, }) {
|
|
164
|
+
const rows = [
|
|
165
|
+
{
|
|
166
|
+
label: messages.bookingId,
|
|
167
|
+
value: _jsx("span", { className: "font-mono text-xs", children: booking.id }),
|
|
168
|
+
},
|
|
169
|
+
{
|
|
170
|
+
label: messages.bookingNumber,
|
|
171
|
+
value: _jsx("span", { className: "font-mono text-xs", children: booking.bookingNumber }),
|
|
172
|
+
},
|
|
173
|
+
{
|
|
174
|
+
label: messages.status,
|
|
175
|
+
value: _jsx(StatusBadge, { status: booking.status, children: statusLabel }),
|
|
176
|
+
},
|
|
177
|
+
{
|
|
178
|
+
label: messages.communicationLanguage,
|
|
179
|
+
value: booking.communicationLanguage ? (_jsx("span", { className: "uppercase", children: booking.communicationLanguage })) : (_jsx("span", { className: "text-muted-foreground", children: noValue })),
|
|
180
|
+
},
|
|
181
|
+
{ label: messages.created, value: formatDateTime(booking.createdAt) },
|
|
182
|
+
{ label: messages.updated, value: formatDateTime(booking.updatedAt) },
|
|
183
|
+
];
|
|
184
|
+
return (_jsxs("div", { "data-slot": "booking-metadata", className: "flex flex-col gap-3", children: [_jsxs("h2", { className: "flex items-center gap-2 text-base font-semibold", children: [_jsx(Info, { className: "h-4 w-4 text-muted-foreground" }), messages.title] }), _jsx("div", { className: "overflow-hidden rounded-md border bg-background", children: _jsx("dl", { className: "divide-y", children: rows.map((row) => (_jsxs("div", { className: "flex items-center justify-between gap-4 px-4 py-3 text-sm", children: [_jsx("dt", { className: "text-muted-foreground", children: row.label }), _jsx("dd", { className: "text-right", children: row.value })] }, row.label))) }) })] }));
|
|
185
|
+
}
|
|
186
|
+
/**
|
|
187
|
+
* Traffic-light badge for a percentage value. Color thresholds are
|
|
188
|
+
* passed in by the caller (Paid uses 0 → red, 0..100 → yellow, 100 →
|
|
189
|
+
* green; Margin uses <0 → red, 0..10 → yellow, >10 → green).
|
|
190
|
+
*/
|
|
191
|
+
function renderPercentBadge(percent, classFor) {
|
|
192
|
+
return (_jsxs(Badge, { variant: "outline", className: cn("border-transparent", classFor(percent)), children: [percent, "%"] }));
|
|
193
|
+
}
|
|
194
|
+
function paidBadgeClass(percent) {
|
|
195
|
+
if (percent <= 0)
|
|
196
|
+
return "bg-red-500/10 text-red-600 dark:text-red-400";
|
|
197
|
+
if (percent >= 100)
|
|
198
|
+
return "bg-green-500/10 text-green-600 dark:text-green-400";
|
|
199
|
+
return "bg-yellow-500/10 text-yellow-700 dark:text-yellow-400";
|
|
200
|
+
}
|
|
201
|
+
function marginBadgeClass(percent) {
|
|
202
|
+
if (percent < 0)
|
|
203
|
+
return "bg-red-500/10 text-red-600 dark:text-red-400";
|
|
204
|
+
if (percent > 10)
|
|
205
|
+
return "bg-green-500/10 text-green-600 dark:text-green-400";
|
|
206
|
+
return "bg-yellow-500/10 text-yellow-700 dark:text-yellow-400";
|
|
207
|
+
}
|
|
208
|
+
function BillingField({ label, value, icon, }) {
|
|
209
|
+
return (_jsxs("div", { className: "min-w-0", children: [_jsxs("div", { className: "mb-1 flex items-center gap-1.5 text-xs font-medium text-muted-foreground", children: [icon, label] }), _jsx("div", { className: "truncate text-sm font-medium", children: value })] }));
|
|
210
|
+
}
|
|
211
|
+
function getBookingStatusLabel(status, labels) {
|
|
212
|
+
return labels[status] ?? status;
|
|
213
|
+
}
|
|
214
|
+
function formatAmount(cents, currency, locale, empty) {
|
|
215
|
+
if (cents == null)
|
|
216
|
+
return empty;
|
|
217
|
+
const amount = cents / 100;
|
|
218
|
+
const amountText = new Intl.NumberFormat(locale, {
|
|
219
|
+
maximumFractionDigits: 0,
|
|
220
|
+
}).format(amount);
|
|
221
|
+
// RON's "symbol" is just the ISO code itself, so a `<symbol> <amount> <code>`
|
|
222
|
+
// layout would print "RON 1.185 RON" — collapse it back to "1.185 RON".
|
|
223
|
+
if (currency.toUpperCase() === "RON") {
|
|
224
|
+
return `${amountText} ${currency}`;
|
|
225
|
+
}
|
|
226
|
+
// Extract the locale's native symbol so we can always render
|
|
227
|
+
// `<symbol> <amount> <code>` regardless of where Intl would otherwise
|
|
228
|
+
// place the symbol for this locale (e.g. Romanian puts it after).
|
|
229
|
+
const parts = new Intl.NumberFormat(locale, {
|
|
230
|
+
style: "currency",
|
|
231
|
+
currency,
|
|
232
|
+
currencyDisplay: "narrowSymbol",
|
|
233
|
+
maximumFractionDigits: 0,
|
|
234
|
+
}).formatToParts(amount);
|
|
235
|
+
const symbol = parts.find((p) => p.type === "currency")?.value ?? currency;
|
|
236
|
+
return `${symbol} ${amountText} ${currency}`;
|
|
237
|
+
}
|
|
238
|
+
function formatDate(iso, locale, empty) {
|
|
239
|
+
if (!iso)
|
|
240
|
+
return empty;
|
|
241
|
+
return new Date(iso).toLocaleDateString(locale, {
|
|
242
|
+
month: "short",
|
|
243
|
+
day: "numeric",
|
|
244
|
+
year: "numeric",
|
|
245
|
+
});
|
|
246
|
+
}
|
|
247
|
+
/**
|
|
248
|
+
* Strip internal markers (`__contract_acceptance__:`,
|
|
249
|
+
* `__payment_policy_source__:`) before rendering — those are
|
|
250
|
+
* server-side relays, not human-readable notes. Their canonical home
|
|
251
|
+
* is on the contract signature row / schedule history; surfacing them
|
|
252
|
+
* here just confuses operators with raw JSON.
|
|
253
|
+
*/
|
|
254
|
+
function visibleInternalNotes(notes) {
|
|
255
|
+
if (!notes)
|
|
256
|
+
return null;
|
|
257
|
+
const visible = notes
|
|
258
|
+
.split("\n")
|
|
259
|
+
.filter((line) => !/^__[a-z0-9_]+__:/i.test(line))
|
|
260
|
+
.join("\n")
|
|
261
|
+
.replace(/\n{3,}/g, "\n\n")
|
|
262
|
+
.trim();
|
|
263
|
+
return visible.length > 0 ? visible : null;
|
|
264
|
+
}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { type BookingRecord } from "../index.js";
|
|
2
|
+
export interface BookingDialogProps {
|
|
3
|
+
open: boolean;
|
|
4
|
+
onOpenChange: (open: boolean) => void;
|
|
5
|
+
booking?: BookingRecord;
|
|
6
|
+
onSuccess?: (booking: BookingRecord) => void;
|
|
7
|
+
/**
|
|
8
|
+
* Pre-seeds the product picker in create mode. Useful when opened from
|
|
9
|
+
* a product detail page. Ignored when editing an existing booking.
|
|
10
|
+
*/
|
|
11
|
+
defaultProductId?: string;
|
|
12
|
+
/**
|
|
13
|
+
* Pre-seeds and locks the departure picker in create mode. Useful when opened from
|
|
14
|
+
* a slot allocation page. Ignored when editing an existing booking.
|
|
15
|
+
*/
|
|
16
|
+
defaultSlotId?: string;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Single booking dialog that handles both create and edit:
|
|
20
|
+
* - Create (no `booking` prop): renders the rich product → option → person
|
|
21
|
+
* picker flow via `BookingCreateSheet`, so the draft booking inherits
|
|
22
|
+
* pricing, dates, and currency from the catalogue instead of being
|
|
23
|
+
* hand-entered.
|
|
24
|
+
* - Edit (with `booking` prop): renders the flat form below that patches
|
|
25
|
+
* the existing row's metadata (status, amounts, dates, notes).
|
|
26
|
+
*/
|
|
27
|
+
export declare function BookingDialog({ open, onOpenChange, booking, onSuccess, defaultProductId, defaultSlotId, }: BookingDialogProps): import("react/jsx-runtime").JSX.Element;
|
|
28
|
+
//# sourceMappingURL=booking-dialog.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"booking-dialog.d.ts","sourceRoot":"","sources":["../../src/components/booking-dialog.tsx"],"names":[],"mappings":"AA4BA,OAAO,EAAE,KAAK,aAAa,EAAsB,MAAM,aAAa,CAAA;AA8BpE,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,OAAO,CAAA;IACb,YAAY,EAAE,CAAC,IAAI,EAAE,OAAO,KAAK,IAAI,CAAA;IACrC,OAAO,CAAC,EAAE,aAAa,CAAA;IACvB,SAAS,CAAC,EAAE,CAAC,OAAO,EAAE,aAAa,KAAK,IAAI,CAAA;IAC5C;;;OAGG;IACH,gBAAgB,CAAC,EAAE,MAAM,CAAA;IACzB;;;OAGG;IACH,aAAa,CAAC,EAAE,MAAM,CAAA;CACvB;AAWD;;;;;;;;GAQG;AACH,wBAAgB,aAAa,CAAC,EAC5B,IAAI,EACJ,YAAY,EACZ,OAAO,EACP,SAAS,EACT,gBAAgB,EAChB,aAAa,GACd,EAAE,kBAAkB,2CAqBpB"}
|
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
|
+
import { Button, Dialog, DialogBody, DialogContent, DialogFooter, DialogHeader, DialogTitle, Input, Label, Select, SelectContent, SelectItem, SelectTrigger, SelectValue, Textarea, } from "@voyant-travel/ui/components";
|
|
4
|
+
import { CurrencyCombobox } from "@voyant-travel/ui/components/currency-combobox";
|
|
5
|
+
import { CurrencyInput } from "@voyant-travel/ui/components/currency-input";
|
|
6
|
+
import { DateRangePicker } from "@voyant-travel/ui/components/date-picker";
|
|
7
|
+
import { zodResolver } from "@voyant-travel/ui/lib/zod-resolver";
|
|
8
|
+
import { Loader2 } from "lucide-react";
|
|
9
|
+
import { useEffect } from "react";
|
|
10
|
+
import { useForm } from "react-hook-form";
|
|
11
|
+
import { z } from "zod/v4";
|
|
12
|
+
import { useBookingsUiMessagesOrDefault } from "../i18n/provider.js";
|
|
13
|
+
import { useBookingMutation } from "../index.js";
|
|
14
|
+
import { BookingCreateSheet } from "./booking-create-sheet.js";
|
|
15
|
+
function createBookingFormSchema(messages) {
|
|
16
|
+
return z.object({
|
|
17
|
+
bookingNumber: z.string().min(1, messages.bookingDialog.validation.bookingNumberRequired),
|
|
18
|
+
status: z.enum([
|
|
19
|
+
"draft",
|
|
20
|
+
"on_hold",
|
|
21
|
+
"awaiting_payment",
|
|
22
|
+
"confirmed",
|
|
23
|
+
"in_progress",
|
|
24
|
+
"completed",
|
|
25
|
+
"expired",
|
|
26
|
+
"cancelled",
|
|
27
|
+
]),
|
|
28
|
+
sellCurrency: z.string().min(3).max(3, messages.bookingDialog.validation.sellCurrencyInvalid),
|
|
29
|
+
sellAmountCents: z.coerce.number().int().min(0).optional().or(z.literal("")).nullable(),
|
|
30
|
+
costAmountCents: z.coerce.number().int().min(0).optional().or(z.literal("")).nullable(),
|
|
31
|
+
startDate: z.string().optional().nullable(),
|
|
32
|
+
endDate: z.string().optional().nullable(),
|
|
33
|
+
pax: z.coerce.number().int().positive().optional().or(z.literal("")).nullable(),
|
|
34
|
+
internalNotes: z.string().optional().nullable(),
|
|
35
|
+
});
|
|
36
|
+
}
|
|
37
|
+
const BOOKING_STATUS_VALUES = [
|
|
38
|
+
"draft",
|
|
39
|
+
"confirmed",
|
|
40
|
+
"in_progress",
|
|
41
|
+
"completed",
|
|
42
|
+
"cancelled",
|
|
43
|
+
];
|
|
44
|
+
const DEFAULT_CURRENCY = "EUR"; // i18n-literal-ok ISO default currency
|
|
45
|
+
/**
|
|
46
|
+
* Single booking dialog that handles both create and edit:
|
|
47
|
+
* - Create (no `booking` prop): renders the rich product → option → person
|
|
48
|
+
* picker flow via `BookingCreateSheet`, so the draft booking inherits
|
|
49
|
+
* pricing, dates, and currency from the catalogue instead of being
|
|
50
|
+
* hand-entered.
|
|
51
|
+
* - Edit (with `booking` prop): renders the flat form below that patches
|
|
52
|
+
* the existing row's metadata (status, amounts, dates, notes).
|
|
53
|
+
*/
|
|
54
|
+
export function BookingDialog({ open, onOpenChange, booking, onSuccess, defaultProductId, defaultSlotId, }) {
|
|
55
|
+
if (!booking) {
|
|
56
|
+
return (_jsx(BookingCreateSheet, { open: open, onOpenChange: onOpenChange, defaultProductId: defaultProductId, defaultSlotId: defaultSlotId, onCreated: onSuccess }));
|
|
57
|
+
}
|
|
58
|
+
return (_jsx(BookingEditDialog, { open: open, onOpenChange: onOpenChange, booking: booking, onSuccess: onSuccess }));
|
|
59
|
+
}
|
|
60
|
+
function BookingEditDialog({ open, onOpenChange, booking, onSuccess }) {
|
|
61
|
+
const { update } = useBookingMutation();
|
|
62
|
+
const messages = useBookingsUiMessagesOrDefault();
|
|
63
|
+
const bookingFormSchema = createBookingFormSchema(messages);
|
|
64
|
+
const form = useForm({
|
|
65
|
+
resolver: zodResolver(bookingFormSchema),
|
|
66
|
+
defaultValues: {
|
|
67
|
+
bookingNumber: "",
|
|
68
|
+
status: "draft",
|
|
69
|
+
sellCurrency: DEFAULT_CURRENCY,
|
|
70
|
+
sellAmountCents: "",
|
|
71
|
+
costAmountCents: "",
|
|
72
|
+
startDate: "",
|
|
73
|
+
endDate: "",
|
|
74
|
+
pax: "",
|
|
75
|
+
internalNotes: "",
|
|
76
|
+
},
|
|
77
|
+
});
|
|
78
|
+
useEffect(() => {
|
|
79
|
+
if (!open)
|
|
80
|
+
return;
|
|
81
|
+
form.reset({
|
|
82
|
+
bookingNumber: booking.bookingNumber,
|
|
83
|
+
status: booking.status === "on_hold" || booking.status === "expired" ? "draft" : booking.status,
|
|
84
|
+
sellCurrency: booking.sellCurrency,
|
|
85
|
+
sellAmountCents: booking.sellAmountCents ?? "",
|
|
86
|
+
costAmountCents: booking.costAmountCents ?? "",
|
|
87
|
+
startDate: booking.startDate ?? "",
|
|
88
|
+
endDate: booking.endDate ?? "",
|
|
89
|
+
pax: booking.pax ?? "",
|
|
90
|
+
internalNotes: booking.internalNotes ?? "",
|
|
91
|
+
});
|
|
92
|
+
}, [booking, form, open]);
|
|
93
|
+
const onSubmit = async (values) => {
|
|
94
|
+
const payload = {
|
|
95
|
+
bookingNumber: values.bookingNumber,
|
|
96
|
+
status: values.status,
|
|
97
|
+
sellCurrency: values.sellCurrency,
|
|
98
|
+
sellAmountCents: typeof values.sellAmountCents === "number" ? values.sellAmountCents : null,
|
|
99
|
+
costAmountCents: typeof values.costAmountCents === "number" ? values.costAmountCents : null,
|
|
100
|
+
startDate: values.startDate || null,
|
|
101
|
+
endDate: values.endDate || null,
|
|
102
|
+
pax: values.pax && typeof values.pax === "number" ? values.pax : null,
|
|
103
|
+
internalNotes: values.internalNotes || null,
|
|
104
|
+
};
|
|
105
|
+
const saved = await update.mutateAsync({ id: booking.id, input: payload });
|
|
106
|
+
onOpenChange(false);
|
|
107
|
+
onSuccess?.(saved);
|
|
108
|
+
};
|
|
109
|
+
const isSubmitting = update.isPending;
|
|
110
|
+
return (_jsx(Dialog, { open: open, onOpenChange: onOpenChange, children: _jsxs(DialogContent, { size: "lg", children: [_jsx(DialogHeader, { children: _jsx(DialogTitle, { children: messages.bookingDialog.editTitle }) }), _jsxs("form", { onSubmit: form.handleSubmit(onSubmit), className: "flex flex-1 flex-col overflow-hidden", children: [_jsxs(DialogBody, { className: "grid gap-4", children: [_jsxs("div", { className: "grid grid-cols-2 gap-4", children: [_jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.bookingNumber }), _jsx(Input, { ...form.register("bookingNumber"), placeholder: messages.bookingDialog.placeholders.bookingNumber }), form.formState.errors.bookingNumber && (_jsx("p", { className: "text-xs text-destructive", children: form.formState.errors.bookingNumber.message }))] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.status }), _jsxs(Select, { value: form.watch("status"), onValueChange: (value) => form.setValue("status", value), children: [_jsx(SelectTrigger, { className: "w-full", children: _jsx(SelectValue, {}) }), _jsx(SelectContent, { children: BOOKING_STATUS_VALUES.map((status) => (_jsx(SelectItem, { value: status, children: messages.common.bookingStatusLabels[status] }, status))) })] })] })] }), _jsxs("div", { className: "grid grid-cols-2 gap-4", children: [_jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.sellCurrency }), _jsx(CurrencyCombobox, { value: form.watch("sellCurrency") || null, onChange: (next) => form.setValue("sellCurrency", next ?? DEFAULT_CURRENCY, {
|
|
111
|
+
shouldValidate: true,
|
|
112
|
+
shouldDirty: true,
|
|
113
|
+
}) })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.travelDates }), _jsx(DateRangePicker, { value: {
|
|
114
|
+
from: form.watch("startDate") || null,
|
|
115
|
+
to: form.watch("endDate") || null,
|
|
116
|
+
}, onChange: (nextValue) => {
|
|
117
|
+
form.setValue("startDate", nextValue?.from ?? "", { shouldDirty: true });
|
|
118
|
+
form.setValue("endDate", nextValue?.to ?? "", { shouldDirty: true });
|
|
119
|
+
}, placeholder: messages.bookingDialog.placeholders.travelDates, className: "w-full" })] })] }), _jsxs("div", { className: "grid grid-cols-3 gap-4", children: [_jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.sellAmountCents }), _jsx(CurrencyInput, { value: typeof form.watch("sellAmountCents") === "number"
|
|
120
|
+
? form.watch("sellAmountCents")
|
|
121
|
+
: null, onChange: (next) => form.setValue("sellAmountCents", next, {
|
|
122
|
+
shouldDirty: true,
|
|
123
|
+
shouldValidate: true,
|
|
124
|
+
}), currency: form.watch("sellCurrency") })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.costAmountCents }), _jsx(CurrencyInput, { value: typeof form.watch("costAmountCents") === "number"
|
|
125
|
+
? form.watch("costAmountCents")
|
|
126
|
+
: null, onChange: (next) => form.setValue("costAmountCents", next, {
|
|
127
|
+
shouldDirty: true,
|
|
128
|
+
shouldValidate: true,
|
|
129
|
+
}), currency: form.watch("sellCurrency") })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.pax }), _jsx(Input, { ...form.register("pax"), type: "number", min: "1", placeholder: messages.bookingDialog.placeholders.pax })] })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDialog.fields.internalNotes }), _jsx(Textarea, { ...form.register("internalNotes"), placeholder: messages.bookingDialog.placeholders.internalNotes })] })] }), _jsxs(DialogFooter, { children: [_jsx(Button, { type: "button", variant: "ghost", size: "sm", onClick: () => onOpenChange(false), children: messages.common.cancel }), _jsxs(Button, { type: "submit", size: "sm", disabled: isSubmitting, children: [isSubmitting && _jsx(Loader2, { className: "mr-2 h-4 w-4 animate-spin" }), messages.common.saveChanges] })] })] })] }) }));
|
|
130
|
+
}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
export interface BookingDocumentDialogProps {
|
|
2
|
+
open: boolean;
|
|
3
|
+
onOpenChange: (open: boolean) => void;
|
|
4
|
+
bookingId: string;
|
|
5
|
+
onSuccess?: () => void;
|
|
6
|
+
}
|
|
7
|
+
export declare function BookingDocumentDialog({ open, onOpenChange, bookingId, onSuccess, }: BookingDocumentDialogProps): import("react/jsx-runtime").JSX.Element;
|
|
8
|
+
//# sourceMappingURL=booking-document-dialog.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"booking-document-dialog.d.ts","sourceRoot":"","sources":["../../src/components/booking-document-dialog.tsx"],"names":[],"mappings":"AAkDA,MAAM,WAAW,0BAA0B;IACzC,IAAI,EAAE,OAAO,CAAA;IACb,YAAY,EAAE,CAAC,IAAI,EAAE,OAAO,KAAK,IAAI,CAAA;IACrC,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,CAAC,EAAE,MAAM,IAAI,CAAA;CACvB;AAED,wBAAgB,qBAAqB,CAAC,EACpC,IAAI,EACJ,YAAY,EACZ,SAAS,EACT,SAAS,GACV,EAAE,0BAA0B,2CAyK5B"}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
|
+
import { Button, Dialog, DialogBody, DialogContent, DialogFooter, DialogHeader, DialogTitle, Label, Select, SelectContent, SelectItem, SelectTrigger, SelectValue, Textarea, } from "@voyant-travel/ui/components";
|
|
4
|
+
import { DatePicker } from "@voyant-travel/ui/components/date-picker";
|
|
5
|
+
import { zodResolver } from "@voyant-travel/ui/lib/zod-resolver";
|
|
6
|
+
import { Loader2 } from "lucide-react";
|
|
7
|
+
import { useEffect, useMemo } from "react";
|
|
8
|
+
import { useForm } from "react-hook-form";
|
|
9
|
+
import { z } from "zod/v4";
|
|
10
|
+
import { useBookingsUiMessagesOrDefault } from "../i18n/provider.js";
|
|
11
|
+
import { useBookingTravelerDocumentMutation, useTravelers } from "../index.js";
|
|
12
|
+
import { FileDropzone } from "./file-dropzone.js";
|
|
13
|
+
const documentTypes = ["visa", "insurance", "health", "passport_copy", "other"];
|
|
14
|
+
const UNASSIGNED = "__unassigned__";
|
|
15
|
+
function createDocumentFormSchema(messages) {
|
|
16
|
+
return z.object({
|
|
17
|
+
type: z.enum(documentTypes).default("other"),
|
|
18
|
+
fileName: z
|
|
19
|
+
.string()
|
|
20
|
+
.min(1, messages.bookingDocumentDialog.validation.fileNameRequired)
|
|
21
|
+
.max(500),
|
|
22
|
+
fileUrl: z.string().url(messages.bookingDocumentDialog.validation.fileUrlInvalid),
|
|
23
|
+
travelerId: z.string().optional().nullable(),
|
|
24
|
+
expiresAt: z.string().optional().nullable(),
|
|
25
|
+
notes: z.string().optional().nullable(),
|
|
26
|
+
});
|
|
27
|
+
}
|
|
28
|
+
export function BookingDocumentDialog({ open, onOpenChange, bookingId, onSuccess, }) {
|
|
29
|
+
const { create } = useBookingTravelerDocumentMutation(bookingId);
|
|
30
|
+
const { data: travelersData } = useTravelers(bookingId);
|
|
31
|
+
const travelers = travelersData?.data ?? [];
|
|
32
|
+
const messages = useBookingsUiMessagesOrDefault();
|
|
33
|
+
const documentFormSchema = createDocumentFormSchema(messages);
|
|
34
|
+
const typeItems = useMemo(() => documentTypes.map((t) => ({
|
|
35
|
+
value: t,
|
|
36
|
+
label: messages.bookingDocumentDialog.documentTypeLabels[t],
|
|
37
|
+
})), [messages.bookingDocumentDialog.documentTypeLabels]);
|
|
38
|
+
const travelerItems = useMemo(() => [
|
|
39
|
+
{
|
|
40
|
+
value: UNASSIGNED,
|
|
41
|
+
label: messages.bookingDocumentDialog.placeholders.travelerUnassigned,
|
|
42
|
+
},
|
|
43
|
+
...travelers.map((t) => ({
|
|
44
|
+
value: t.id,
|
|
45
|
+
label: `${t.firstName} ${t.lastName}`,
|
|
46
|
+
})),
|
|
47
|
+
], [travelers, messages.bookingDocumentDialog.placeholders.travelerUnassigned]);
|
|
48
|
+
const form = useForm({
|
|
49
|
+
resolver: zodResolver(documentFormSchema),
|
|
50
|
+
defaultValues: {
|
|
51
|
+
type: "other",
|
|
52
|
+
fileName: "",
|
|
53
|
+
fileUrl: "",
|
|
54
|
+
travelerId: UNASSIGNED,
|
|
55
|
+
expiresAt: "",
|
|
56
|
+
notes: "",
|
|
57
|
+
},
|
|
58
|
+
});
|
|
59
|
+
useEffect(() => {
|
|
60
|
+
if (open) {
|
|
61
|
+
form.reset();
|
|
62
|
+
}
|
|
63
|
+
}, [form, open]);
|
|
64
|
+
const onSubmit = async (values) => {
|
|
65
|
+
await create.mutateAsync({
|
|
66
|
+
type: values.type,
|
|
67
|
+
fileName: values.fileName,
|
|
68
|
+
fileUrl: values.fileUrl,
|
|
69
|
+
travelerId: values.travelerId && values.travelerId !== UNASSIGNED ? values.travelerId : null,
|
|
70
|
+
expiresAt: values.expiresAt || null,
|
|
71
|
+
notes: values.notes || null,
|
|
72
|
+
});
|
|
73
|
+
onOpenChange(false);
|
|
74
|
+
onSuccess?.();
|
|
75
|
+
};
|
|
76
|
+
return (_jsx(Dialog, { open: open, onOpenChange: onOpenChange, children: _jsxs(DialogContent, { size: "lg", children: [_jsx(DialogHeader, { children: _jsx(DialogTitle, { children: messages.bookingDocumentDialog.title }) }), _jsxs("form", { onSubmit: form.handleSubmit(onSubmit), className: "flex flex-1 flex-col overflow-hidden", children: [_jsxs(DialogBody, { className: "grid gap-4", children: [_jsxs("div", { className: "grid grid-cols-2 gap-4", children: [_jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDocumentDialog.fields.type }), _jsxs(Select, { items: typeItems, value: form.watch("type"), onValueChange: (v) => form.setValue("type", (v ?? "other")), children: [_jsx(SelectTrigger, { className: "w-full", children: _jsx(SelectValue, {}) }), _jsx(SelectContent, { children: documentTypes.map((t) => (_jsx(SelectItem, { value: t, children: messages.bookingDocumentDialog.documentTypeLabels[t] }, t))) })] })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDocumentDialog.fields.traveler }), _jsxs(Select, { items: travelerItems, value: form.watch("travelerId") ?? UNASSIGNED, onValueChange: (v) => form.setValue("travelerId", v ?? UNASSIGNED), children: [_jsx(SelectTrigger, { className: "w-full", children: _jsx(SelectValue, {}) }), _jsxs(SelectContent, { children: [_jsx(SelectItem, { value: UNASSIGNED, children: messages.bookingDocumentDialog.placeholders.travelerUnassigned }), travelers.map((traveler) => (_jsxs(SelectItem, { value: traveler.id, children: [traveler.firstName, " ", traveler.lastName] }, traveler.id)))] })] })] })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDocumentDialog.fields.file }), _jsx(FileDropzone, { accept: "application/pdf,image/*", maxSize: 10 * 1024 * 1024, onUploaded: (upload) => {
|
|
77
|
+
form.setValue("fileUrl", upload.url, { shouldValidate: true });
|
|
78
|
+
form.setValue("fileName", upload.name, { shouldValidate: true });
|
|
79
|
+
}, helperText: messages.bookingDocumentDialog.placeholders.helperText }), form.formState.errors.fileUrl && (_jsx("p", { className: "text-xs text-destructive", children: form.formState.errors.fileUrl.message }))] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDocumentDialog.fields.expiresAt }), _jsx(DatePicker, { value: form.watch("expiresAt") || null, onChange: (next) => form.setValue("expiresAt", next ?? "", {
|
|
80
|
+
shouldValidate: true,
|
|
81
|
+
shouldDirty: true,
|
|
82
|
+
}), placeholder: messages.bookingDocumentDialog.placeholders.expiresAt, className: "w-full" })] }), _jsxs("div", { className: "flex flex-col gap-2", children: [_jsx(Label, { children: messages.bookingDocumentDialog.fields.notes }), _jsx(Textarea, { ...form.register("notes"), placeholder: messages.bookingDocumentDialog.placeholders.notes })] })] }), _jsxs(DialogFooter, { children: [_jsx(Button, { type: "button", variant: "ghost", size: "sm", onClick: () => onOpenChange(false), children: messages.common.cancel }), _jsxs(Button, { type: "submit", size: "sm", disabled: create.isPending, children: [create.isPending && _jsx(Loader2, { className: "mr-2 h-4 w-4 animate-spin" }), messages.bookingDocumentDialog.actions.addDocument] })] })] })] }) }));
|
|
83
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"booking-document-list.d.ts","sourceRoot":"","sources":["../../src/components/booking-document-list.tsx"],"names":[],"mappings":"AA6BA,MAAM,WAAW,wBAAwB;IACvC,SAAS,EAAE,MAAM,CAAA;CAClB;AAED,wBAAgB,mBAAmB,CAAC,EAAE,SAAS,EAAE,EAAE,wBAAwB,2CAqH1E"}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
|
|
3
|
+
import { Badge, Button, Card, CardContent, CardHeader, CardTitle, } from "@voyant-travel/ui/components";
|
|
4
|
+
import { ExternalLink, FileText, Plus, Trash2 } from "lucide-react";
|
|
5
|
+
import * as React from "react";
|
|
6
|
+
import { useBookingsUiI18nOrDefault, useBookingsUiMessagesOrDefault } from "../i18n/provider.js";
|
|
7
|
+
import { useBookingTravelerDocumentMutation, useBookingTravelerDocuments, useTravelers, } from "../index.js";
|
|
8
|
+
import { BookingDocumentDialog } from "./booking-document-dialog.js";
|
|
9
|
+
const typeVariant = {
|
|
10
|
+
visa: "default",
|
|
11
|
+
insurance: "secondary",
|
|
12
|
+
health: "secondary",
|
|
13
|
+
passport_copy: "outline",
|
|
14
|
+
other: "outline",
|
|
15
|
+
};
|
|
16
|
+
export function BookingDocumentList({ bookingId }) {
|
|
17
|
+
const [dialogOpen, setDialogOpen] = React.useState(false);
|
|
18
|
+
const { data } = useBookingTravelerDocuments(bookingId);
|
|
19
|
+
const { data: travelersData } = useTravelers(bookingId);
|
|
20
|
+
const { remove } = useBookingTravelerDocumentMutation(bookingId);
|
|
21
|
+
const { formatDate } = useBookingsUiI18nOrDefault();
|
|
22
|
+
const messages = useBookingsUiMessagesOrDefault();
|
|
23
|
+
const documents = data?.data ?? [];
|
|
24
|
+
const travelers = travelersData?.data ?? [];
|
|
25
|
+
const travelerMap = new Map();
|
|
26
|
+
for (const traveler of travelers) {
|
|
27
|
+
travelerMap.set(traveler.id, traveler);
|
|
28
|
+
}
|
|
29
|
+
return (_jsxs(Card, { "data-slot": "booking-document-list", children: [_jsxs(CardHeader, { className: "flex flex-row items-center justify-between", children: [_jsxs(CardTitle, { className: "flex items-center gap-2", children: [_jsx(FileText, { className: "h-4 w-4" }), messages.bookingDocumentList.title] }), _jsxs(Button, { size: "sm", onClick: () => setDialogOpen(true), children: [_jsx(Plus, { className: "mr-2 h-4 w-4" }), messages.bookingDocumentList.addDocument] })] }), _jsx(CardContent, { children: documents.length === 0 ? (_jsx("p", { className: "py-4 text-center text-sm text-muted-foreground", children: messages.bookingDocumentList.empty })) : (_jsx("div", { className: "rounded border bg-background", children: _jsxs("table", { className: "w-full text-sm", children: [_jsx("thead", { children: _jsxs("tr", { className: "border-b text-muted-foreground", children: [_jsx("th", { className: "p-2 text-left font-medium", children: messages.bookingDocumentList.columns.type }), _jsx("th", { className: "p-2 text-left font-medium", children: messages.bookingDocumentList.columns.file }), _jsx("th", { className: "p-2 text-left font-medium", children: messages.bookingDocumentList.columns.traveler }), _jsx("th", { className: "p-2 text-left font-medium", children: messages.bookingDocumentList.columns.expires }), _jsx("th", { className: "p-2 text-left font-medium", children: messages.bookingDocumentList.columns.notes }), _jsx("th", { className: "w-20 p-2" })] }) }), _jsx("tbody", { children: documents.map((doc) => {
|
|
30
|
+
const traveler = doc.travelerId ? travelerMap.get(doc.travelerId) : undefined;
|
|
31
|
+
return (_jsxs("tr", { className: "border-b last:border-b-0", children: [_jsx("td", { className: "p-2", children: _jsx(Badge, { variant: typeVariant[doc.type] ?? "outline", children: messages.bookingDocumentDialog.documentTypeLabels[doc.type] }) }), _jsx("td", { className: "p-2", children: _jsxs("a", { href: doc.fileUrl, target: "_blank", rel: "noopener noreferrer", className: "inline-flex items-center gap-1 hover:underline", children: [doc.fileName, _jsx(ExternalLink, { className: "h-3 w-3 text-muted-foreground" })] }) }), _jsx("td", { className: "p-2", children: traveler
|
|
32
|
+
? `${traveler.firstName} ${traveler.lastName}`
|
|
33
|
+
: doc.travelerId
|
|
34
|
+
? doc.travelerId
|
|
35
|
+
: messages.bookingDocumentList.values.travelerUnavailable }), _jsx("td", { className: "p-2", children: doc.expiresAt
|
|
36
|
+
? formatDate(doc.expiresAt)
|
|
37
|
+
: messages.bookingDocumentList.values.expiresUnavailable }), _jsx("td", { className: "max-w-[200px] truncate p-2 text-muted-foreground", children: doc.notes ?? messages.bookingDocumentList.values.notesUnavailable }), _jsx("td", { className: "p-2", children: _jsx("button", { type: "button", onClick: () => {
|
|
38
|
+
if (confirm(messages.bookingDocumentList.actions.deleteConfirm)) {
|
|
39
|
+
remove.mutate(doc.id);
|
|
40
|
+
}
|
|
41
|
+
}, className: "text-muted-foreground hover:text-destructive", children: _jsx(Trash2, { className: "h-3.5 w-3.5" }) }) })] }, doc.id));
|
|
42
|
+
}) })] }) })) }), _jsx(BookingDocumentDialog, { open: dialogOpen, onOpenChange: setDialogOpen, bookingId: bookingId })] }));
|
|
43
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export interface BookingGroupLinkDialogProps {
|
|
2
|
+
open: boolean;
|
|
3
|
+
onOpenChange: (open: boolean) => void;
|
|
4
|
+
bookingId: string;
|
|
5
|
+
productId?: string | null;
|
|
6
|
+
optionUnitId?: string | null;
|
|
7
|
+
onLinked?: (groupId: string) => void;
|
|
8
|
+
}
|
|
9
|
+
export declare function BookingGroupLinkDialog({ open, onOpenChange, bookingId, productId, optionUnitId, onLinked, }: BookingGroupLinkDialogProps): import("react/jsx-runtime").JSX.Element;
|
|
10
|
+
//# sourceMappingURL=booking-group-link-dialog.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"booking-group-link-dialog.d.ts","sourceRoot":"","sources":["../../src/components/booking-group-link-dialog.tsx"],"names":[],"mappings":"AA2BA,MAAM,WAAW,2BAA2B;IAC1C,IAAI,EAAE,OAAO,CAAA;IACb,YAAY,EAAE,CAAC,IAAI,EAAE,OAAO,KAAK,IAAI,CAAA;IACrC,SAAS,EAAE,MAAM,CAAA;IACjB,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IACzB,YAAY,CAAC,EAAE,MAAM,GAAG,IAAI,CAAA;IAC5B,QAAQ,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAA;CACrC;AAMD,wBAAgB,sBAAsB,CAAC,EACrC,IAAI,EACJ,YAAY,EACZ,SAAS,EACT,SAAS,EACT,YAAY,EACZ,QAAQ,GACT,EAAE,2BAA2B,2CAqL7B"}
|