@cratis/components 0.1.9
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/.storybook/main.ts +24 -0
- package/CommandDialog/CommandDialog.stories.tsx +25 -0
- package/CommandDialog/CommandDialog.tsx +161 -0
- package/CommandDialog/index.ts +4 -0
- package/CommandForm/CommandForm.stories.tsx +24 -0
- package/CommandForm/CommandForm.tsx +266 -0
- package/CommandForm/CommandFormField.tsx +27 -0
- package/CommandForm/CommandFormFields.tsx +142 -0
- package/CommandForm/DatePickerField.tsx +57 -0
- package/CommandForm/DropdownField.tsx +65 -0
- package/CommandForm/InputTextField.tsx +62 -0
- package/CommandForm/SliderField.tsx +68 -0
- package/CommandForm/index.ts +10 -0
- package/Common/ErrorBoundary.stories.tsx +10 -0
- package/Common/ErrorBoundary.tsx +41 -0
- package/Common/FormElement.stories.tsx +10 -0
- package/Common/FormElement.tsx +20 -0
- package/Common/Page.stories.tsx +10 -0
- package/Common/Page.tsx +21 -0
- package/Common/index.ts +6 -0
- package/DataPage/DataPage.stories.tsx +10 -0
- package/DataPage/DataPage.tsx +191 -0
- package/DataPage/index.ts +4 -0
- package/DataTables/DataTableForObservableQuery.stories.tsx +10 -0
- package/DataTables/DataTableForObservableQuery.tsx +97 -0
- package/DataTables/DataTableForQuery.stories.tsx +10 -0
- package/DataTables/DataTableForQuery.tsx +97 -0
- package/DataTables/index.ts +5 -0
- package/Dialogs/BusyIndicatorDialog.stories.tsx +26 -0
- package/Dialogs/BusyIndicatorDialog.tsx +26 -0
- package/Dialogs/ConfirmationDialog.stories.tsx +36 -0
- package/Dialogs/ConfirmationDialog.tsx +75 -0
- package/Dialogs/index.ts +5 -0
- package/Dropdown/Dropdown.tsx +23 -0
- package/Dropdown/index.ts +4 -0
- package/PivotViewer/PivotViewer.stories.tsx +24 -0
- package/PivotViewer/PivotViewer.tsx +791 -0
- package/PivotViewer/components/AxisLabels.tsx +69 -0
- package/PivotViewer/components/DetailPanel.tsx +108 -0
- package/PivotViewer/components/FilterPanel.tsx +189 -0
- package/PivotViewer/components/FilterPanelContainer.tsx +10 -0
- package/PivotViewer/components/PivotCanvas.tsx +660 -0
- package/PivotViewer/components/PivotViewerMain.tsx +229 -0
- package/PivotViewer/components/RangeHistogramFilter.tsx +220 -0
- package/PivotViewer/components/Spinner.tsx +21 -0
- package/PivotViewer/components/Toolbar.tsx +130 -0
- package/PivotViewer/components/ToolbarContainer.tsx +10 -0
- package/PivotViewer/components/index.ts +12 -0
- package/PivotViewer/components/pivot/animation.ts +108 -0
- package/PivotViewer/components/pivot/buckets.ts +152 -0
- package/PivotViewer/components/pivot/colorResolver.ts +67 -0
- package/PivotViewer/components/pivot/constants.ts +46 -0
- package/PivotViewer/components/pivot/sprites.ts +265 -0
- package/PivotViewer/components/pivot/visibility.ts +319 -0
- package/PivotViewer/constants.ts +9 -0
- package/PivotViewer/engine/layout.ts +149 -0
- package/PivotViewer/engine/pivot.worker.ts +86 -0
- package/PivotViewer/engine/store.ts +437 -0
- package/PivotViewer/engine/types.ts +255 -0
- package/PivotViewer/hooks/index.ts +13 -0
- package/PivotViewer/hooks/useContainerDimensions.ts +45 -0
- package/PivotViewer/hooks/useDimensionState.ts +53 -0
- package/PivotViewer/hooks/useFilterOptions.ts +36 -0
- package/PivotViewer/hooks/useFilterPanelDrag.ts +49 -0
- package/PivotViewer/hooks/useFilterState.ts +106 -0
- package/PivotViewer/hooks/useFilteredData.ts +119 -0
- package/PivotViewer/hooks/usePanning.ts +163 -0
- package/PivotViewer/hooks/usePivotEngine.ts +252 -0
- package/PivotViewer/hooks/useSelectedItem.ts +402 -0
- package/PivotViewer/hooks/useWheelZoom.ts +114 -0
- package/PivotViewer/hooks/useZoomState.ts +34 -0
- package/PivotViewer/index.ts +7 -0
- package/PivotViewer/types.ts +59 -0
- package/PivotViewer/utils/animations.ts +249 -0
- package/PivotViewer/utils/constants.ts +20 -0
- package/PivotViewer/utils/index.ts +6 -0
- package/PivotViewer/utils/selection.ts +292 -0
- package/PivotViewer/utils/utils.ts +259 -0
- package/README.md +1 -0
- package/TimeMachine/EventsView.stories.tsx +10 -0
- package/TimeMachine/EventsView.tsx +119 -0
- package/TimeMachine/Properties.stories.tsx +10 -0
- package/TimeMachine/Properties.tsx +98 -0
- package/TimeMachine/ReadModelView.stories.tsx +10 -0
- package/TimeMachine/ReadModelView.tsx +143 -0
- package/TimeMachine/TimeMachine.stories.tsx +10 -0
- package/TimeMachine/TimeMachine.tsx +244 -0
- package/TimeMachine/index.ts +8 -0
- package/TimeMachine/types.ts +23 -0
- package/dist/cjs/CommandDialog/CommandDialog.js +71 -0
- package/dist/cjs/CommandDialog/CommandDialog.js.map +1 -0
- package/dist/cjs/CommandDialog/index.js +9 -0
- package/dist/cjs/CommandDialog/index.js.map +1 -0
- package/dist/cjs/CommandForm/CommandForm.js +179 -0
- package/dist/cjs/CommandForm/CommandForm.js.map +1 -0
- package/dist/cjs/CommandForm/CommandFormField.js +11 -0
- package/dist/cjs/CommandForm/CommandFormField.js.map +1 -0
- package/dist/cjs/CommandForm/CommandFormFields.js +67 -0
- package/dist/cjs/CommandForm/CommandFormFields.js.map +1 -0
- package/dist/cjs/CommandForm/DatePickerField.js +31 -0
- package/dist/cjs/CommandForm/DatePickerField.js.map +1 -0
- package/dist/cjs/CommandForm/DropdownField.js +31 -0
- package/dist/cjs/CommandForm/DropdownField.js.map +1 -0
- package/dist/cjs/CommandForm/InputTextField.js +32 -0
- package/dist/cjs/CommandForm/InputTextField.js.map +1 -0
- package/dist/cjs/CommandForm/SliderField.js +34 -0
- package/dist/cjs/CommandForm/SliderField.js.map +1 -0
- package/dist/cjs/CommandForm/index.js +23 -0
- package/dist/cjs/CommandForm/index.js.map +1 -0
- package/dist/cjs/Common/Page.js +10 -0
- package/dist/cjs/Common/Page.js.map +1 -0
- package/dist/cjs/DataPage/DataPage.js +64 -0
- package/dist/cjs/DataPage/DataPage.js.map +1 -0
- package/dist/cjs/DataPage/index.js +11 -0
- package/dist/cjs/DataPage/index.js.map +1 -0
- package/dist/cjs/DataTables/DataTableForObservableQuery.js +17 -0
- package/dist/cjs/DataTables/DataTableForObservableQuery.js.map +1 -0
- package/dist/cjs/DataTables/DataTableForQuery.js +17 -0
- package/dist/cjs/DataTables/DataTableForQuery.js.map +1 -0
- package/dist/cjs/DataTables/index.js +10 -0
- package/dist/cjs/DataTables/index.js.map +1 -0
- package/dist/cjs/Dialogs/BusyIndicatorDialog.js +13 -0
- package/dist/cjs/Dialogs/BusyIndicatorDialog.js.map +1 -0
- package/dist/cjs/Dialogs/ConfirmationDialog.js +33 -0
- package/dist/cjs/Dialogs/ConfirmationDialog.js.map +1 -0
- package/dist/cjs/Dialogs/index.js +10 -0
- package/dist/cjs/Dialogs/index.js.map +1 -0
- package/dist/cjs/Dropdown/Dropdown.js +15 -0
- package/dist/cjs/Dropdown/Dropdown.js.map +1 -0
- package/dist/cjs/Dropdown/index.js +8 -0
- package/dist/cjs/Dropdown/index.js.map +1 -0
- package/dist/cjs/PivotViewer/PivotViewer.js +525 -0
- package/dist/cjs/PivotViewer/PivotViewer.js.map +1 -0
- package/dist/cjs/PivotViewer/components/AxisLabels.js +27 -0
- package/dist/cjs/PivotViewer/components/AxisLabels.js.map +1 -0
- package/dist/cjs/PivotViewer/components/DetailPanel.js +35 -0
- package/dist/cjs/PivotViewer/components/DetailPanel.js.map +1 -0
- package/dist/cjs/PivotViewer/components/FilterPanel.js +59 -0
- package/dist/cjs/PivotViewer/components/FilterPanel.js.map +1 -0
- package/dist/cjs/PivotViewer/components/FilterPanelContainer.js +11 -0
- package/dist/cjs/PivotViewer/components/FilterPanelContainer.js.map +1 -0
- package/dist/cjs/PivotViewer/components/PivotCanvas.js +394 -0
- package/dist/cjs/PivotViewer/components/PivotCanvas.js.map +1 -0
- package/dist/cjs/PivotViewer/components/PivotViewerMain.js +81 -0
- package/dist/cjs/PivotViewer/components/PivotViewerMain.js.map +1 -0
- package/dist/cjs/PivotViewer/components/RangeHistogramFilter.js +124 -0
- package/dist/cjs/PivotViewer/components/RangeHistogramFilter.js.map +1 -0
- package/dist/cjs/PivotViewer/components/Spinner.js +11 -0
- package/dist/cjs/PivotViewer/components/Spinner.js.map +1 -0
- package/dist/cjs/PivotViewer/components/Toolbar.js +12 -0
- package/dist/cjs/PivotViewer/components/Toolbar.js.map +1 -0
- package/dist/cjs/PivotViewer/components/ToolbarContainer.js +11 -0
- package/dist/cjs/PivotViewer/components/ToolbarContainer.js.map +1 -0
- package/dist/cjs/PivotViewer/components/pivot/animation.js +82 -0
- package/dist/cjs/PivotViewer/components/pivot/animation.js.map +1 -0
- package/dist/cjs/PivotViewer/components/pivot/buckets.js +124 -0
- package/dist/cjs/PivotViewer/components/pivot/buckets.js.map +1 -0
- package/dist/cjs/PivotViewer/components/pivot/colorResolver.js +64 -0
- package/dist/cjs/PivotViewer/components/pivot/colorResolver.js.map +1 -0
- package/dist/cjs/PivotViewer/components/pivot/constants.js +19 -0
- package/dist/cjs/PivotViewer/components/pivot/constants.js.map +1 -0
- package/dist/cjs/PivotViewer/components/pivot/sprites.js +227 -0
- package/dist/cjs/PivotViewer/components/pivot/sprites.js.map +1 -0
- package/dist/cjs/PivotViewer/components/pivot/visibility.js +223 -0
- package/dist/cjs/PivotViewer/components/pivot/visibility.js.map +1 -0
- package/dist/cjs/PivotViewer/constants.js +16 -0
- package/dist/cjs/PivotViewer/constants.js.map +1 -0
- package/dist/cjs/PivotViewer/engine/layout.js +95 -0
- package/dist/cjs/PivotViewer/engine/layout.js.map +1 -0
- package/dist/cjs/PivotViewer/engine/store.js +336 -0
- package/dist/cjs/PivotViewer/engine/store.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/useContainerDimensions.js +30 -0
- package/dist/cjs/PivotViewer/hooks/useContainerDimensions.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/useDimensionState.js +43 -0
- package/dist/cjs/PivotViewer/hooks/useDimensionState.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/useFilterOptions.js +24 -0
- package/dist/cjs/PivotViewer/hooks/useFilterOptions.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/useFilterState.js +96 -0
- package/dist/cjs/PivotViewer/hooks/useFilterState.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/usePanning.js +120 -0
- package/dist/cjs/PivotViewer/hooks/usePanning.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/usePivotEngine.js +183 -0
- package/dist/cjs/PivotViewer/hooks/usePivotEngine.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/useWheelZoom.js +93 -0
- package/dist/cjs/PivotViewer/hooks/useWheelZoom.js.map +1 -0
- package/dist/cjs/PivotViewer/hooks/useZoomState.js +31 -0
- package/dist/cjs/PivotViewer/hooks/useZoomState.js.map +1 -0
- package/dist/cjs/PivotViewer/index.js +9 -0
- package/dist/cjs/PivotViewer/index.js.map +1 -0
- package/dist/cjs/PivotViewer/utils/animations.js +144 -0
- package/dist/cjs/PivotViewer/utils/animations.js.map +1 -0
- package/dist/cjs/PivotViewer/utils/constants.js +12 -0
- package/dist/cjs/PivotViewer/utils/constants.js.map +1 -0
- package/dist/cjs/PivotViewer/utils/selection.js +136 -0
- package/dist/cjs/PivotViewer/utils/selection.js.map +1 -0
- package/dist/cjs/PivotViewer/utils/utils.js +150 -0
- package/dist/cjs/PivotViewer/utils/utils.js.map +1 -0
- package/dist/cjs/TimeMachine/EventsView.js +57 -0
- package/dist/cjs/TimeMachine/EventsView.js.map +1 -0
- package/dist/cjs/TimeMachine/Properties.js +58 -0
- package/dist/cjs/TimeMachine/Properties.js.map +1 -0
- package/dist/cjs/TimeMachine/ReadModelView.js +40 -0
- package/dist/cjs/TimeMachine/ReadModelView.js.map +1 -0
- package/dist/cjs/TimeMachine/TimeMachine.js +98 -0
- package/dist/cjs/TimeMachine/TimeMachine.js.map +1 -0
- package/dist/cjs/TimeMachine/index.js +14 -0
- package/dist/cjs/TimeMachine/index.js.map +1 -0
- package/dist/cjs/index.js +22 -0
- package/dist/cjs/index.js.map +1 -0
- package/dist/cjs/useOverlayZIndex.js +24 -0
- package/dist/cjs/useOverlayZIndex.js.map +1 -0
- package/dist/esm/CommandDialog/CommandDialog.d.ts +45 -0
- package/dist/esm/CommandDialog/CommandDialog.d.ts.map +1 -0
- package/dist/esm/CommandDialog/CommandDialog.js +68 -0
- package/dist/esm/CommandDialog/CommandDialog.js.map +1 -0
- package/dist/esm/CommandDialog/CommandDialog.stories.d.ts +7 -0
- package/dist/esm/CommandDialog/CommandDialog.stories.d.ts.map +1 -0
- package/dist/esm/CommandDialog/CommandDialog.stories.js +12 -0
- package/dist/esm/CommandDialog/CommandDialog.stories.js.map +1 -0
- package/dist/esm/CommandDialog/index.d.ts +2 -0
- package/dist/esm/CommandDialog/index.d.ts.map +1 -0
- package/dist/esm/CommandDialog/index.js +2 -0
- package/dist/esm/CommandDialog/index.js.map +1 -0
- package/dist/esm/CommandForm/CommandForm.d.ts +49 -0
- package/dist/esm/CommandForm/CommandForm.d.ts.map +1 -0
- package/dist/esm/CommandForm/CommandForm.js +174 -0
- package/dist/esm/CommandForm/CommandForm.js.map +1 -0
- package/dist/esm/CommandForm/CommandForm.stories.d.ts +7 -0
- package/dist/esm/CommandForm/CommandForm.stories.d.ts.map +1 -0
- package/dist/esm/CommandForm/CommandForm.stories.js +12 -0
- package/dist/esm/CommandForm/CommandForm.stories.js.map +1 -0
- package/dist/esm/CommandForm/CommandFormField.d.ts +18 -0
- package/dist/esm/CommandForm/CommandFormField.d.ts.map +1 -0
- package/dist/esm/CommandForm/CommandFormField.js +9 -0
- package/dist/esm/CommandForm/CommandFormField.js.map +1 -0
- package/dist/esm/CommandForm/CommandFormFields.d.ts +11 -0
- package/dist/esm/CommandForm/CommandFormFields.d.ts.map +1 -0
- package/dist/esm/CommandForm/CommandFormFields.js +65 -0
- package/dist/esm/CommandForm/CommandFormFields.js.map +1 -0
- package/dist/esm/CommandForm/DatePickerField.d.ts +20 -0
- package/dist/esm/CommandForm/DatePickerField.d.ts.map +1 -0
- package/dist/esm/CommandForm/DatePickerField.js +29 -0
- package/dist/esm/CommandForm/DatePickerField.js.map +1 -0
- package/dist/esm/CommandForm/DropdownField.d.ts +24 -0
- package/dist/esm/CommandForm/DropdownField.d.ts.map +1 -0
- package/dist/esm/CommandForm/DropdownField.js +29 -0
- package/dist/esm/CommandForm/DropdownField.js.map +1 -0
- package/dist/esm/CommandForm/InputTextField.d.ts +20 -0
- package/dist/esm/CommandForm/InputTextField.d.ts.map +1 -0
- package/dist/esm/CommandForm/InputTextField.js +30 -0
- package/dist/esm/CommandForm/InputTextField.js.map +1 -0
- package/dist/esm/CommandForm/SliderField.d.ts +23 -0
- package/dist/esm/CommandForm/SliderField.d.ts.map +1 -0
- package/dist/esm/CommandForm/SliderField.js +32 -0
- package/dist/esm/CommandForm/SliderField.js.map +1 -0
- package/dist/esm/CommandForm/index.d.ts +8 -0
- package/dist/esm/CommandForm/index.d.ts.map +1 -0
- package/dist/esm/CommandForm/index.js +8 -0
- package/dist/esm/CommandForm/index.js.map +1 -0
- package/dist/esm/Common/ErrorBoundary.d.ts +16 -0
- package/dist/esm/Common/ErrorBoundary.d.ts.map +1 -0
- package/dist/esm/Common/ErrorBoundary.js +21 -0
- package/dist/esm/Common/ErrorBoundary.js.map +1 -0
- package/dist/esm/Common/ErrorBoundary.stories.d.ts +8 -0
- package/dist/esm/Common/ErrorBoundary.stories.d.ts.map +1 -0
- package/dist/esm/Common/ErrorBoundary.stories.js +6 -0
- package/dist/esm/Common/ErrorBoundary.stories.js.map +1 -0
- package/dist/esm/Common/FormElement.d.ts +6 -0
- package/dist/esm/Common/FormElement.d.ts.map +1 -0
- package/dist/esm/Common/FormElement.js +5 -0
- package/dist/esm/Common/FormElement.js.map +1 -0
- package/dist/esm/Common/FormElement.stories.d.ts +8 -0
- package/dist/esm/Common/FormElement.stories.d.ts.map +1 -0
- package/dist/esm/Common/FormElement.stories.js +6 -0
- package/dist/esm/Common/FormElement.stories.js.map +1 -0
- package/dist/esm/Common/Page.d.ts +8 -0
- package/dist/esm/Common/Page.d.ts.map +1 -0
- package/dist/esm/Common/Page.js +8 -0
- package/dist/esm/Common/Page.js.map +1 -0
- package/dist/esm/Common/Page.stories.d.ts +8 -0
- package/dist/esm/Common/Page.stories.d.ts.map +1 -0
- package/dist/esm/Common/Page.stories.js +6 -0
- package/dist/esm/Common/Page.stories.js.map +1 -0
- package/dist/esm/Common/index.d.ts +4 -0
- package/dist/esm/Common/index.d.ts.map +1 -0
- package/dist/esm/Common/index.js +4 -0
- package/dist/esm/Common/index.js.map +1 -0
- package/dist/esm/DataPage/DataPage.d.ts +41 -0
- package/dist/esm/DataPage/DataPage.d.ts.map +1 -0
- package/dist/esm/DataPage/DataPage.js +59 -0
- package/dist/esm/DataPage/DataPage.js.map +1 -0
- package/dist/esm/DataPage/DataPage.stories.d.ts +8 -0
- package/dist/esm/DataPage/DataPage.stories.d.ts.map +1 -0
- package/dist/esm/DataPage/DataPage.stories.js +6 -0
- package/dist/esm/DataPage/DataPage.stories.js.map +1 -0
- package/dist/esm/DataPage/index.d.ts +2 -0
- package/dist/esm/DataPage/index.d.ts.map +1 -0
- package/dist/esm/DataPage/index.js +2 -0
- package/dist/esm/DataPage/index.js.map +1 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.d.ts +17 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.d.ts.map +1 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.js +15 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.js.map +1 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.stories.d.ts +8 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.stories.d.ts.map +1 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.stories.js +6 -0
- package/dist/esm/DataTables/DataTableForObservableQuery.stories.js.map +1 -0
- package/dist/esm/DataTables/DataTableForQuery.d.ts +17 -0
- package/dist/esm/DataTables/DataTableForQuery.d.ts.map +1 -0
- package/dist/esm/DataTables/DataTableForQuery.js +15 -0
- package/dist/esm/DataTables/DataTableForQuery.js.map +1 -0
- package/dist/esm/DataTables/DataTableForQuery.stories.d.ts +8 -0
- package/dist/esm/DataTables/DataTableForQuery.stories.d.ts.map +1 -0
- package/dist/esm/DataTables/DataTableForQuery.stories.js +6 -0
- package/dist/esm/DataTables/DataTableForQuery.stories.js.map +1 -0
- package/dist/esm/DataTables/index.d.ts +3 -0
- package/dist/esm/DataTables/index.d.ts.map +1 -0
- package/dist/esm/DataTables/index.js +3 -0
- package/dist/esm/DataTables/index.js.map +1 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.d.ts +3 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.d.ts.map +1 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.js +11 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.js.map +1 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.stories.d.ts +7 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.stories.d.ts.map +1 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.stories.js +15 -0
- package/dist/esm/Dialogs/BusyIndicatorDialog.stories.js.map +1 -0
- package/dist/esm/Dialogs/ConfirmationDialog.d.ts +2 -0
- package/dist/esm/Dialogs/ConfirmationDialog.d.ts.map +1 -0
- package/dist/esm/Dialogs/ConfirmationDialog.js +31 -0
- package/dist/esm/Dialogs/ConfirmationDialog.js.map +1 -0
- package/dist/esm/Dialogs/ConfirmationDialog.stories.d.ts +7 -0
- package/dist/esm/Dialogs/ConfirmationDialog.stories.d.ts.map +1 -0
- package/dist/esm/Dialogs/ConfirmationDialog.stories.js +20 -0
- package/dist/esm/Dialogs/ConfirmationDialog.stories.js.map +1 -0
- package/dist/esm/Dialogs/index.d.ts +3 -0
- package/dist/esm/Dialogs/index.d.ts.map +1 -0
- package/dist/esm/Dialogs/index.js +3 -0
- package/dist/esm/Dialogs/index.js.map +1 -0
- package/dist/esm/Dropdown/Dropdown.d.ts +5 -0
- package/dist/esm/Dropdown/Dropdown.d.ts.map +1 -0
- package/dist/esm/Dropdown/Dropdown.js +13 -0
- package/dist/esm/Dropdown/Dropdown.js.map +1 -0
- package/dist/esm/Dropdown/index.d.ts +2 -0
- package/dist/esm/Dropdown/index.d.ts.map +1 -0
- package/dist/esm/Dropdown/index.js +2 -0
- package/dist/esm/Dropdown/index.js.map +1 -0
- package/dist/esm/PivotViewer/PivotViewer.d.ts +4 -0
- package/dist/esm/PivotViewer/PivotViewer.d.ts.map +1 -0
- package/dist/esm/PivotViewer/PivotViewer.js +523 -0
- package/dist/esm/PivotViewer/PivotViewer.js.map +1 -0
- package/dist/esm/PivotViewer/PivotViewer.stories.d.ts +7 -0
- package/dist/esm/PivotViewer/PivotViewer.stories.d.ts.map +1 -0
- package/dist/esm/PivotViewer/PivotViewer.stories.js +12 -0
- package/dist/esm/PivotViewer/PivotViewer.stories.js.map +1 -0
- package/dist/esm/PivotViewer/components/AxisLabels.d.ts +13 -0
- package/dist/esm/PivotViewer/components/AxisLabels.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/AxisLabels.js +25 -0
- package/dist/esm/PivotViewer/components/AxisLabels.js.map +1 -0
- package/dist/esm/PivotViewer/components/DetailPanel.d.ts +6 -0
- package/dist/esm/PivotViewer/components/DetailPanel.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/DetailPanel.js +33 -0
- package/dist/esm/PivotViewer/components/DetailPanel.js.map +1 -0
- package/dist/esm/PivotViewer/components/FilterPanel.d.ts +27 -0
- package/dist/esm/PivotViewer/components/FilterPanel.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/FilterPanel.js +57 -0
- package/dist/esm/PivotViewer/components/FilterPanel.js.map +1 -0
- package/dist/esm/PivotViewer/components/FilterPanelContainer.d.ts +4 -0
- package/dist/esm/PivotViewer/components/FilterPanelContainer.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/FilterPanelContainer.js +9 -0
- package/dist/esm/PivotViewer/components/FilterPanelContainer.js.map +1 -0
- package/dist/esm/PivotViewer/components/PivotCanvas.d.ts +29 -0
- package/dist/esm/PivotViewer/components/PivotCanvas.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/PivotCanvas.js +373 -0
- package/dist/esm/PivotViewer/components/PivotCanvas.js.map +1 -0
- package/dist/esm/PivotViewer/components/PivotViewerMain.d.ts +43 -0
- package/dist/esm/PivotViewer/components/PivotViewerMain.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/PivotViewerMain.js +79 -0
- package/dist/esm/PivotViewer/components/PivotViewerMain.js.map +1 -0
- package/dist/esm/PivotViewer/components/RangeHistogramFilter.d.ts +11 -0
- package/dist/esm/PivotViewer/components/RangeHistogramFilter.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/RangeHistogramFilter.js +122 -0
- package/dist/esm/PivotViewer/components/RangeHistogramFilter.js.map +1 -0
- package/dist/esm/PivotViewer/components/Spinner.d.ts +3 -0
- package/dist/esm/PivotViewer/components/Spinner.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/Spinner.js +9 -0
- package/dist/esm/PivotViewer/components/Spinner.js.map +1 -0
- package/dist/esm/PivotViewer/components/Toolbar.d.ts +21 -0
- package/dist/esm/PivotViewer/components/Toolbar.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/Toolbar.js +10 -0
- package/dist/esm/PivotViewer/components/Toolbar.js.map +1 -0
- package/dist/esm/PivotViewer/components/ToolbarContainer.d.ts +4 -0
- package/dist/esm/PivotViewer/components/ToolbarContainer.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/ToolbarContainer.js +9 -0
- package/dist/esm/PivotViewer/components/ToolbarContainer.js.map +1 -0
- package/dist/esm/PivotViewer/components/index.d.ts +9 -0
- package/dist/esm/PivotViewer/components/index.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/index.js +8 -0
- package/dist/esm/PivotViewer/components/index.js.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/animation.d.ts +29 -0
- package/dist/esm/PivotViewer/components/pivot/animation.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/animation.js +79 -0
- package/dist/esm/PivotViewer/components/pivot/animation.js.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/buckets.d.ts +6 -0
- package/dist/esm/PivotViewer/components/pivot/buckets.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/buckets.js +102 -0
- package/dist/esm/PivotViewer/components/pivot/buckets.js.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/colorResolver.d.ts +4 -0
- package/dist/esm/PivotViewer/components/pivot/colorResolver.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/colorResolver.js +61 -0
- package/dist/esm/PivotViewer/components/pivot/colorResolver.js.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/constants.d.ts +38 -0
- package/dist/esm/PivotViewer/components/pivot/constants.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/constants.js +14 -0
- package/dist/esm/PivotViewer/components/pivot/constants.js.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/sprites.d.ts +6 -0
- package/dist/esm/PivotViewer/components/pivot/sprites.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/sprites.js +203 -0
- package/dist/esm/PivotViewer/components/pivot/sprites.js.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/visibility.d.ts +26 -0
- package/dist/esm/PivotViewer/components/pivot/visibility.d.ts.map +1 -0
- package/dist/esm/PivotViewer/components/pivot/visibility.js +221 -0
- package/dist/esm/PivotViewer/components/pivot/visibility.js.map +1 -0
- package/dist/esm/PivotViewer/constants.d.ts +7 -0
- package/dist/esm/PivotViewer/constants.d.ts.map +1 -0
- package/dist/esm/PivotViewer/constants.js +9 -0
- package/dist/esm/PivotViewer/constants.js.map +1 -0
- package/dist/esm/PivotViewer/engine/layout.d.ts +3 -0
- package/dist/esm/PivotViewer/engine/layout.d.ts.map +1 -0
- package/dist/esm/PivotViewer/engine/layout.js +93 -0
- package/dist/esm/PivotViewer/engine/layout.js.map +1 -0
- package/dist/esm/PivotViewer/engine/pivot.worker.d.ts +2 -0
- package/dist/esm/PivotViewer/engine/pivot.worker.d.ts.map +1 -0
- package/dist/esm/PivotViewer/engine/pivot.worker.js +58 -0
- package/dist/esm/PivotViewer/engine/pivot.worker.js.map +1 -0
- package/dist/esm/PivotViewer/engine/store.d.ts +9 -0
- package/dist/esm/PivotViewer/engine/store.d.ts.map +1 -0
- package/dist/esm/PivotViewer/engine/store.js +328 -0
- package/dist/esm/PivotViewer/engine/store.js.map +1 -0
- package/dist/esm/PivotViewer/engine/types.d.ts +125 -0
- package/dist/esm/PivotViewer/engine/types.d.ts.map +1 -0
- package/dist/esm/PivotViewer/engine/types.js +2 -0
- package/dist/esm/PivotViewer/engine/types.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/index.d.ts +11 -0
- package/dist/esm/PivotViewer/hooks/index.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/index.js +11 -0
- package/dist/esm/PivotViewer/hooks/index.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useContainerDimensions.d.ts +5 -0
- package/dist/esm/PivotViewer/hooks/useContainerDimensions.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useContainerDimensions.js +28 -0
- package/dist/esm/PivotViewer/hooks/useContainerDimensions.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useDimensionState.d.ts +11 -0
- package/dist/esm/PivotViewer/hooks/useDimensionState.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useDimensionState.js +41 -0
- package/dist/esm/PivotViewer/hooks/useDimensionState.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilterOptions.d.ts +12 -0
- package/dist/esm/PivotViewer/hooks/useFilterOptions.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilterOptions.js +22 -0
- package/dist/esm/PivotViewer/hooks/useFilterOptions.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilterPanelDrag.d.ts +9 -0
- package/dist/esm/PivotViewer/hooks/useFilterPanelDrag.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilterPanelDrag.js +42 -0
- package/dist/esm/PivotViewer/hooks/useFilterPanelDrag.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilterState.d.ts +11 -0
- package/dist/esm/PivotViewer/hooks/useFilterState.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilterState.js +94 -0
- package/dist/esm/PivotViewer/hooks/useFilterState.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilteredData.d.ts +9 -0
- package/dist/esm/PivotViewer/hooks/useFilteredData.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useFilteredData.js +85 -0
- package/dist/esm/PivotViewer/hooks/useFilteredData.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/usePanning.d.ts +10 -0
- package/dist/esm/PivotViewer/hooks/usePanning.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/usePanning.js +118 -0
- package/dist/esm/PivotViewer/hooks/usePanning.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/usePivotEngine.d.ts +14 -0
- package/dist/esm/PivotViewer/hooks/usePivotEngine.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/usePivotEngine.js +180 -0
- package/dist/esm/PivotViewer/hooks/usePivotEngine.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useSelectedItem.d.ts +8 -0
- package/dist/esm/PivotViewer/hooks/useSelectedItem.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useSelectedItem.js +322 -0
- package/dist/esm/PivotViewer/hooks/useSelectedItem.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useWheelZoom.d.ts +2 -0
- package/dist/esm/PivotViewer/hooks/useWheelZoom.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useWheelZoom.js +91 -0
- package/dist/esm/PivotViewer/hooks/useWheelZoom.js.map +1 -0
- package/dist/esm/PivotViewer/hooks/useZoomState.d.ts +9 -0
- package/dist/esm/PivotViewer/hooks/useZoomState.d.ts.map +1 -0
- package/dist/esm/PivotViewer/hooks/useZoomState.js +29 -0
- package/dist/esm/PivotViewer/hooks/useZoomState.js.map +1 -0
- package/dist/esm/PivotViewer/index.d.ts +4 -0
- package/dist/esm/PivotViewer/index.d.ts.map +1 -0
- package/dist/esm/PivotViewer/index.js +2 -0
- package/dist/esm/PivotViewer/index.js.map +1 -0
- package/dist/esm/PivotViewer/types.d.ts +47 -0
- package/dist/esm/PivotViewer/types.d.ts.map +1 -0
- package/dist/esm/PivotViewer/types.js +2 -0
- package/dist/esm/PivotViewer/types.js.map +1 -0
- package/dist/esm/PivotViewer/utils/animations.d.ts +54 -0
- package/dist/esm/PivotViewer/utils/animations.d.ts.map +1 -0
- package/dist/esm/PivotViewer/utils/animations.js +139 -0
- package/dist/esm/PivotViewer/utils/animations.js.map +1 -0
- package/dist/esm/PivotViewer/utils/constants.d.ts +13 -0
- package/dist/esm/PivotViewer/utils/constants.d.ts.map +1 -0
- package/dist/esm/PivotViewer/utils/constants.js +7 -0
- package/dist/esm/PivotViewer/utils/constants.js.map +1 -0
- package/dist/esm/PivotViewer/utils/index.d.ts +4 -0
- package/dist/esm/PivotViewer/utils/index.d.ts.map +1 -0
- package/dist/esm/PivotViewer/utils/index.js +4 -0
- package/dist/esm/PivotViewer/utils/index.js.map +1 -0
- package/dist/esm/PivotViewer/utils/selection.d.ts +50 -0
- package/dist/esm/PivotViewer/utils/selection.d.ts.map +1 -0
- package/dist/esm/PivotViewer/utils/selection.js +134 -0
- package/dist/esm/PivotViewer/utils/selection.js.map +1 -0
- package/dist/esm/PivotViewer/utils/utils.d.ts +24 -0
- package/dist/esm/PivotViewer/utils/utils.d.ts.map +1 -0
- package/dist/esm/PivotViewer/utils/utils.js +138 -0
- package/dist/esm/PivotViewer/utils/utils.js.map +1 -0
- package/dist/esm/TimeMachine/EventsView.d.ts +9 -0
- package/dist/esm/TimeMachine/EventsView.d.ts.map +1 -0
- package/dist/esm/TimeMachine/EventsView.js +55 -0
- package/dist/esm/TimeMachine/EventsView.js.map +1 -0
- package/dist/esm/TimeMachine/EventsView.stories.d.ts +8 -0
- package/dist/esm/TimeMachine/EventsView.stories.d.ts.map +1 -0
- package/dist/esm/TimeMachine/EventsView.stories.js +6 -0
- package/dist/esm/TimeMachine/EventsView.stories.js.map +1 -0
- package/dist/esm/TimeMachine/Properties.d.ts +9 -0
- package/dist/esm/TimeMachine/Properties.d.ts.map +1 -0
- package/dist/esm/TimeMachine/Properties.js +56 -0
- package/dist/esm/TimeMachine/Properties.js.map +1 -0
- package/dist/esm/TimeMachine/Properties.stories.d.ts +8 -0
- package/dist/esm/TimeMachine/Properties.stories.d.ts.map +1 -0
- package/dist/esm/TimeMachine/Properties.stories.js +6 -0
- package/dist/esm/TimeMachine/Properties.stories.js.map +1 -0
- package/dist/esm/TimeMachine/ReadModelView.d.ts +12 -0
- package/dist/esm/TimeMachine/ReadModelView.d.ts.map +1 -0
- package/dist/esm/TimeMachine/ReadModelView.js +38 -0
- package/dist/esm/TimeMachine/ReadModelView.js.map +1 -0
- package/dist/esm/TimeMachine/ReadModelView.stories.d.ts +8 -0
- package/dist/esm/TimeMachine/ReadModelView.stories.d.ts.map +1 -0
- package/dist/esm/TimeMachine/ReadModelView.stories.js +6 -0
- package/dist/esm/TimeMachine/ReadModelView.stories.js.map +1 -0
- package/dist/esm/TimeMachine/TimeMachine.d.ts +12 -0
- package/dist/esm/TimeMachine/TimeMachine.d.ts.map +1 -0
- package/dist/esm/TimeMachine/TimeMachine.js +93 -0
- package/dist/esm/TimeMachine/TimeMachine.js.map +1 -0
- package/dist/esm/TimeMachine/TimeMachine.stories.d.ts +8 -0
- package/dist/esm/TimeMachine/TimeMachine.stories.d.ts.map +1 -0
- package/dist/esm/TimeMachine/TimeMachine.stories.js +6 -0
- package/dist/esm/TimeMachine/TimeMachine.stories.js.map +1 -0
- package/dist/esm/TimeMachine/index.d.ts +6 -0
- package/dist/esm/TimeMachine/index.d.ts.map +1 -0
- package/dist/esm/TimeMachine/index.js +5 -0
- package/dist/esm/TimeMachine/index.js.map +1 -0
- package/dist/esm/TimeMachine/types.d.ts +19 -0
- package/dist/esm/TimeMachine/types.d.ts.map +1 -0
- package/dist/esm/TimeMachine/types.js +2 -0
- package/dist/esm/TimeMachine/types.js.map +1 -0
- package/dist/esm/index.d.ts +10 -0
- package/dist/esm/index.d.ts.map +1 -0
- package/dist/esm/index.js +17 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/tsconfig.tsbuildinfo +1 -0
- package/dist/esm/useOverlayZIndex.d.ts +2 -0
- package/dist/esm/useOverlayZIndex.d.ts.map +1 -0
- package/dist/esm/useOverlayZIndex.js +22 -0
- package/dist/esm/useOverlayZIndex.js.map +1 -0
- package/dist/esm/vite.config.d.ts +3 -0
- package/dist/esm/vite.config.d.ts.map +1 -0
- package/dist/esm/vite.config.js +76 -0
- package/dist/esm/vite.config.js.map +1 -0
- package/global.d.ts +11 -0
- package/index.ts +22 -0
- package/package.json +160 -0
- package/useOverlayZIndex.ts +32 -0
- package/vite.config.ts +80 -0
|
@@ -0,0 +1,252 @@
|
|
|
1
|
+
// Copyright (c) Cratis. All rights reserved.
|
|
2
|
+
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
|
|
3
|
+
|
|
4
|
+
import { useEffect, useRef, useState, useCallback } from 'react';
|
|
5
|
+
import type {
|
|
6
|
+
PivotStore,
|
|
7
|
+
PivotIndexes,
|
|
8
|
+
FilterSpec,
|
|
9
|
+
FilterResult,
|
|
10
|
+
GroupSpec,
|
|
11
|
+
GroupingResult,
|
|
12
|
+
WorkerInMessage,
|
|
13
|
+
WorkerOutMessage,
|
|
14
|
+
FieldValue,
|
|
15
|
+
} from '../engine/types';
|
|
16
|
+
import { buildStore, buildIndexes, applyFilters, computeGrouping, sortIds } from '../engine/store';
|
|
17
|
+
|
|
18
|
+
export interface UsePivotEngineOptions<TItem extends object> {
|
|
19
|
+
data: TItem[];
|
|
20
|
+
fieldExtractors: Map<string, (item: TItem) => FieldValue>;
|
|
21
|
+
indexFields: string[];
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export interface UsePivotEngineResult {
|
|
25
|
+
ready: boolean;
|
|
26
|
+
applyFilters: (filters: FilterSpec[]) => Promise<FilterResult>;
|
|
27
|
+
computeGrouping: (visibleIds: Uint32Array, groupBy: GroupSpec) => Promise<GroupingResult>;
|
|
28
|
+
sortIds: (visibleIds: Uint32Array, sortBy: string) => Promise<Uint32Array>;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function usePivotEngine<TItem extends object>({
|
|
32
|
+
data,
|
|
33
|
+
fieldExtractors,
|
|
34
|
+
indexFields,
|
|
35
|
+
}: UsePivotEngineOptions<TItem>): UsePivotEngineResult {
|
|
36
|
+
const [ready, setReady] = useState(false);
|
|
37
|
+
const workerRef = useRef<Worker | null>(null);
|
|
38
|
+
const indexesRef = useRef<PivotIndexes | null>(null);
|
|
39
|
+
const fallbackRef = useRef(false);
|
|
40
|
+
const storeRef = useRef<PivotStore | null>(null);
|
|
41
|
+
const pendingCallbacksRef = useRef<Map<string, (result: unknown) => void>>(new Map());
|
|
42
|
+
|
|
43
|
+
useEffect(() => {
|
|
44
|
+
const worker = new Worker(
|
|
45
|
+
new URL('../engine/pivot.worker.ts', import.meta.url),
|
|
46
|
+
{ type: 'module' }
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
workerRef.current = worker;
|
|
50
|
+
|
|
51
|
+
worker.onmessage = (e: MessageEvent<WorkerOutMessage>) => {
|
|
52
|
+
const message = e.data;
|
|
53
|
+
|
|
54
|
+
switch (message.type) {
|
|
55
|
+
case 'indexesReady':
|
|
56
|
+
console.log('[PivotEngine] Indexes ready');
|
|
57
|
+
setReady(true);
|
|
58
|
+
break;
|
|
59
|
+
|
|
60
|
+
case 'filterResult': {
|
|
61
|
+
const callback = pendingCallbacksRef.current.get('filter');
|
|
62
|
+
if (callback) {
|
|
63
|
+
callback(message.result);
|
|
64
|
+
pendingCallbacksRef.current.delete('filter');
|
|
65
|
+
}
|
|
66
|
+
break;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
case 'groupingResult': {
|
|
70
|
+
const callback = pendingCallbacksRef.current.get('grouping');
|
|
71
|
+
if (callback) {
|
|
72
|
+
callback(message.result);
|
|
73
|
+
pendingCallbacksRef.current.delete('grouping');
|
|
74
|
+
}
|
|
75
|
+
break;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
case 'sortResult': {
|
|
79
|
+
const callback = pendingCallbacksRef.current.get('sort');
|
|
80
|
+
if (callback) {
|
|
81
|
+
callback(message.result);
|
|
82
|
+
pendingCallbacksRef.current.delete('sort');
|
|
83
|
+
}
|
|
84
|
+
break;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
worker.onerror = (error) => {
|
|
90
|
+
console.error('[PivotEngine] Worker error:', error);
|
|
91
|
+
// enable synchronous fallback so UI can still function without worker
|
|
92
|
+
fallbackRef.current = true;
|
|
93
|
+
workerRef.current = null;
|
|
94
|
+
// if we already built store/indexes, mark ready so UI can use fallback
|
|
95
|
+
if (storeRef.current) {
|
|
96
|
+
try {
|
|
97
|
+
indexesRef.current = buildIndexes(storeRef.current, indexFields);
|
|
98
|
+
} catch (e) {
|
|
99
|
+
console.error('[PivotEngine] Failed to build indexes in fallback:', e);
|
|
100
|
+
indexesRef.current = null;
|
|
101
|
+
}
|
|
102
|
+
setReady(true);
|
|
103
|
+
}
|
|
104
|
+
};
|
|
105
|
+
|
|
106
|
+
return () => {
|
|
107
|
+
worker.terminate();
|
|
108
|
+
};
|
|
109
|
+
}, []);
|
|
110
|
+
|
|
111
|
+
useEffect(() => {
|
|
112
|
+
if (!workerRef.current) return;
|
|
113
|
+
|
|
114
|
+
console.log('[PivotEngine] Building indexes for', data.length, 'items');
|
|
115
|
+
setReady(false);
|
|
116
|
+
|
|
117
|
+
const store = buildStore(data, fieldExtractors);
|
|
118
|
+
storeRef.current = store;
|
|
119
|
+
|
|
120
|
+
// compute indexes locally for immediate fallback and cache
|
|
121
|
+
try {
|
|
122
|
+
indexesRef.current = buildIndexes(store, indexFields);
|
|
123
|
+
} catch (e) {
|
|
124
|
+
console.error('[PivotEngine] buildIndexes failed:', e);
|
|
125
|
+
indexesRef.current = null;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
if (workerRef.current) {
|
|
129
|
+
const message: WorkerInMessage = {
|
|
130
|
+
type: 'buildIndexes',
|
|
131
|
+
store,
|
|
132
|
+
fields: indexFields,
|
|
133
|
+
};
|
|
134
|
+
|
|
135
|
+
workerRef.current.postMessage(message);
|
|
136
|
+
} else {
|
|
137
|
+
// no worker available, mark ready to allow fallback synchronous usage
|
|
138
|
+
setReady(true);
|
|
139
|
+
}
|
|
140
|
+
}, [data, fieldExtractors, indexFields]);
|
|
141
|
+
|
|
142
|
+
const applyFiltersCallback = useCallback(
|
|
143
|
+
(filters: FilterSpec[]): Promise<FilterResult> => {
|
|
144
|
+
return new Promise((resolve) => {
|
|
145
|
+
// If worker is not available, use synchronous fallback using local indexes
|
|
146
|
+
if (!workerRef.current || fallbackRef.current) {
|
|
147
|
+
try {
|
|
148
|
+
const store = storeRef.current;
|
|
149
|
+
const indexes = indexesRef.current;
|
|
150
|
+
if (store && indexes) {
|
|
151
|
+
const result = applyFilters(store, indexes, filters);
|
|
152
|
+
resolve(result);
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
} catch (e) {
|
|
156
|
+
console.error('[PivotEngine] fallback applyFilters error:', e);
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
// if fallback not possible, return empty result
|
|
160
|
+
resolve({ visibleIds: new Uint32Array(0), count: 0 });
|
|
161
|
+
return;
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
pendingCallbacksRef.current.set('filter', resolve as (result: unknown) => void);
|
|
165
|
+
|
|
166
|
+
const message: WorkerInMessage = {
|
|
167
|
+
type: 'applyFilters',
|
|
168
|
+
filters,
|
|
169
|
+
};
|
|
170
|
+
|
|
171
|
+
workerRef.current.postMessage(message);
|
|
172
|
+
});
|
|
173
|
+
},
|
|
174
|
+
[ready]
|
|
175
|
+
);
|
|
176
|
+
|
|
177
|
+
const computeGroupingCallback = useCallback(
|
|
178
|
+
(visibleIds: Uint32Array, groupBy: GroupSpec): Promise<GroupingResult> => {
|
|
179
|
+
return new Promise((resolve) => {
|
|
180
|
+
// synchronous fallback if worker unavailable
|
|
181
|
+
if (!workerRef.current || fallbackRef.current) {
|
|
182
|
+
try {
|
|
183
|
+
const store = storeRef.current;
|
|
184
|
+
const indexes = indexesRef.current;
|
|
185
|
+
if (store && indexes) {
|
|
186
|
+
const result = computeGrouping(store, indexes, visibleIds, groupBy);
|
|
187
|
+
resolve(result);
|
|
188
|
+
return;
|
|
189
|
+
}
|
|
190
|
+
} catch (e) {
|
|
191
|
+
console.error('[PivotEngine] fallback computeGrouping error:', e);
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
resolve({ groups: [] });
|
|
195
|
+
return;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
pendingCallbacksRef.current.set('grouping', resolve as (result: unknown) => void);
|
|
199
|
+
|
|
200
|
+
const message: WorkerInMessage = {
|
|
201
|
+
type: 'computeGrouping',
|
|
202
|
+
visibleIds,
|
|
203
|
+
groupBy,
|
|
204
|
+
};
|
|
205
|
+
|
|
206
|
+
workerRef.current.postMessage(message);
|
|
207
|
+
});
|
|
208
|
+
},
|
|
209
|
+
[ready]
|
|
210
|
+
);
|
|
211
|
+
|
|
212
|
+
const sortIdsCallback = useCallback(
|
|
213
|
+
(visibleIds: Uint32Array, sortBy: string): Promise<Uint32Array> => {
|
|
214
|
+
return new Promise((resolve) => {
|
|
215
|
+
// synchronous fallback if worker unavailable
|
|
216
|
+
if (!workerRef.current || fallbackRef.current) {
|
|
217
|
+
try {
|
|
218
|
+
const store = storeRef.current;
|
|
219
|
+
if (store) {
|
|
220
|
+
const result = sortIds(store, visibleIds, sortBy);
|
|
221
|
+
resolve(result);
|
|
222
|
+
return;
|
|
223
|
+
}
|
|
224
|
+
} catch (e) {
|
|
225
|
+
console.error('[PivotEngine] fallback sortIds error:', e);
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
resolve(visibleIds);
|
|
229
|
+
return;
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
pendingCallbacksRef.current.set('sort', resolve as (result: unknown) => void);
|
|
233
|
+
|
|
234
|
+
const message: WorkerInMessage = {
|
|
235
|
+
type: 'sort',
|
|
236
|
+
ids: visibleIds,
|
|
237
|
+
sortBy,
|
|
238
|
+
};
|
|
239
|
+
|
|
240
|
+
workerRef.current.postMessage(message);
|
|
241
|
+
});
|
|
242
|
+
},
|
|
243
|
+
[ready]
|
|
244
|
+
);
|
|
245
|
+
|
|
246
|
+
return {
|
|
247
|
+
ready,
|
|
248
|
+
applyFilters: applyFiltersCallback,
|
|
249
|
+
computeGrouping: computeGroupingCallback,
|
|
250
|
+
sortIds: sortIdsCallback,
|
|
251
|
+
};
|
|
252
|
+
}
|
|
@@ -0,0 +1,402 @@
|
|
|
1
|
+
// Copyright (c) Cratis. All rights reserved.
|
|
2
|
+
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
|
|
3
|
+
|
|
4
|
+
import { useState, useRef, useCallback, useEffect } from 'react';
|
|
5
|
+
import { flushSync } from 'react-dom';
|
|
6
|
+
import { ZOOM_MAX } from '../utils/utils';
|
|
7
|
+
|
|
8
|
+
function getOffsetWithin(element: HTMLElement, ancestor: HTMLElement) {
|
|
9
|
+
let current: HTMLElement | null = element;
|
|
10
|
+
let x = 0;
|
|
11
|
+
let y = 0;
|
|
12
|
+
|
|
13
|
+
while (current && current !== ancestor) {
|
|
14
|
+
x += current.offsetLeft;
|
|
15
|
+
y += current.offsetTop;
|
|
16
|
+
current = current.offsetParent as HTMLElement | null;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
if (current !== ancestor) {
|
|
20
|
+
const containerRect = ancestor.getBoundingClientRect();
|
|
21
|
+
const elementRect = element.getBoundingClientRect();
|
|
22
|
+
x = elementRect.left - containerRect.left + ancestor.scrollLeft;
|
|
23
|
+
y = elementRect.top - containerRect.top + ancestor.scrollTop;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
return { x, y };
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function scrollCardToCenter(
|
|
30
|
+
container: HTMLElement,
|
|
31
|
+
card: HTMLElement,
|
|
32
|
+
smooth: boolean = true,
|
|
33
|
+
detailPanelWidth: number = 0
|
|
34
|
+
) {
|
|
35
|
+
const { x, y } = getOffsetWithin(card, container);
|
|
36
|
+
const cardCenterX = x + card.offsetWidth / 2;
|
|
37
|
+
const cardCenterY = y + card.offsetHeight / 2;
|
|
38
|
+
|
|
39
|
+
const availableWidth = container.clientWidth - detailPanelWidth;
|
|
40
|
+
const targetX = availableWidth / 2;
|
|
41
|
+
const targetY = container.clientHeight / 2;
|
|
42
|
+
|
|
43
|
+
const maxScrollLeft = Math.max(0, container.scrollWidth - container.clientWidth);
|
|
44
|
+
const maxScrollTop = Math.max(0, container.scrollHeight - container.clientHeight);
|
|
45
|
+
const targetLeft = Math.min(maxScrollLeft, Math.max(0, cardCenterX - targetX));
|
|
46
|
+
const targetTop = Math.min(maxScrollTop, Math.max(0, cardCenterY - targetY));
|
|
47
|
+
|
|
48
|
+
container.scrollTo({
|
|
49
|
+
left: targetLeft,
|
|
50
|
+
top: targetTop,
|
|
51
|
+
behavior: smooth ? 'smooth' : 'auto'
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
function escapeAttributeSelectorValue(value: string) {
|
|
56
|
+
return value.replace(/\\/g, '\\\\').replace(/"/g, '\\"');
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
export function useSelectedItem<TItem extends object>(
|
|
60
|
+
containerRef: React.RefObject<HTMLDivElement | null>,
|
|
61
|
+
zoomLevel: number,
|
|
62
|
+
setZoomLevel: (zoom: number) => void,
|
|
63
|
+
resolveId: (item: TItem, index: number) => string | number,
|
|
64
|
+
viewMode: 'collection' | 'grouped' = 'grouped'
|
|
65
|
+
) {
|
|
66
|
+
const [selectedItem, setSelectedItem] = useState<TItem | null>(null);
|
|
67
|
+
const [preSelectionState, setPreSelectionState] = useState<{ zoom: number; scrollLeft: number; scrollTop: number } | null>(null);
|
|
68
|
+
const [isZooming, setIsZooming] = useState(false);
|
|
69
|
+
const pendingCenterRaf = useRef<number | null>(null);
|
|
70
|
+
|
|
71
|
+
const focusCardById = useCallback((targetId: string | number, detailPanelWidth: number = 380) => {
|
|
72
|
+
const container = containerRef.current;
|
|
73
|
+
if (!container) {
|
|
74
|
+
return;
|
|
75
|
+
}
|
|
76
|
+
const selector = `[data-card-id="${escapeAttributeSelectorValue(String(targetId))}"]`;
|
|
77
|
+
const latestCard = container.querySelector(selector) as HTMLElement | null;
|
|
78
|
+
if (latestCard) {
|
|
79
|
+
scrollCardToCenter(container, latestCard, true, detailPanelWidth);
|
|
80
|
+
}
|
|
81
|
+
}, [containerRef]);
|
|
82
|
+
|
|
83
|
+
const handleCardClick = useCallback((item: TItem, e: React.MouseEvent) => {
|
|
84
|
+
const container = containerRef.current;
|
|
85
|
+
if (!container) return;
|
|
86
|
+
|
|
87
|
+
const card = (e.target as HTMLElement).closest('.pv-card') as HTMLElement;
|
|
88
|
+
if (!card) return;
|
|
89
|
+
|
|
90
|
+
const itemId = resolveId(item, 0);
|
|
91
|
+
const selectedId = selectedItem ? resolveId(selectedItem, 0) : null;
|
|
92
|
+
|
|
93
|
+
if (selectedId === itemId) {
|
|
94
|
+
if (pendingCenterRaf.current !== null && typeof window !== 'undefined') {
|
|
95
|
+
window.clearTimeout(pendingCenterRaf.current);
|
|
96
|
+
pendingCenterRaf.current = null;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
if (viewMode === 'collection') {
|
|
100
|
+
setSelectedItem(null);
|
|
101
|
+
if (preSelectionState) {
|
|
102
|
+
container.scrollTo({ left: preSelectionState.scrollLeft, top: preSelectionState.scrollTop, behavior: 'smooth' });
|
|
103
|
+
setPreSelectionState(null);
|
|
104
|
+
}
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
if (!preSelectionState || Math.abs(preSelectionState.zoom - zoomLevel) <= 0.001) {
|
|
109
|
+
setSelectedItem(null);
|
|
110
|
+
if (preSelectionState) {
|
|
111
|
+
container.scrollTo({ left: preSelectionState.scrollLeft, top: preSelectionState.scrollTop, behavior: 'smooth' });
|
|
112
|
+
setPreSelectionState(null);
|
|
113
|
+
}
|
|
114
|
+
return;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
setIsZooming(true);
|
|
118
|
+
container.style.scrollBehavior = 'auto';
|
|
119
|
+
|
|
120
|
+
const startZoom = zoomLevel;
|
|
121
|
+
const targetZoom = preSelectionState.zoom;
|
|
122
|
+
const targetScrollLeft = preSelectionState.scrollLeft;
|
|
123
|
+
const targetScrollTop = preSelectionState.scrollTop;
|
|
124
|
+
|
|
125
|
+
const startOffset = getOffsetWithin(card, container);
|
|
126
|
+
const startCardScreenX = startOffset.x + card.offsetWidth / 2 - container.scrollLeft;
|
|
127
|
+
const startCardScreenY = startOffset.y + card.offsetHeight / 2 - container.scrollTop;
|
|
128
|
+
|
|
129
|
+
const duration = 300;
|
|
130
|
+
let startTime: number | null = null;
|
|
131
|
+
|
|
132
|
+
const easeOutCubic = (t: number) => 1 - Math.pow(1 - t, 3);
|
|
133
|
+
|
|
134
|
+
const animate = (timestamp: number) => {
|
|
135
|
+
if (startTime === null) {
|
|
136
|
+
startTime = timestamp;
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
const elapsed = timestamp - startTime;
|
|
140
|
+
const progress = Math.min(1, elapsed / duration);
|
|
141
|
+
const easedProgress = easeOutCubic(progress);
|
|
142
|
+
|
|
143
|
+
const currentZoom = startZoom + (targetZoom - startZoom) * easedProgress;
|
|
144
|
+
|
|
145
|
+
flushSync(() => {
|
|
146
|
+
setZoomLevel(currentZoom);
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
const currentOffset = getOffsetWithin(card, container);
|
|
150
|
+
const currentCardCenterX = currentOffset.x + card.offsetWidth / 2;
|
|
151
|
+
const currentCardCenterY = currentOffset.y + card.offsetHeight / 2;
|
|
152
|
+
|
|
153
|
+
const endCardScreenX = currentCardCenterX - targetScrollLeft;
|
|
154
|
+
const endCardScreenY = currentCardCenterY - targetScrollTop;
|
|
155
|
+
|
|
156
|
+
const desiredScreenX = startCardScreenX + (endCardScreenX - startCardScreenX) * easedProgress;
|
|
157
|
+
const desiredScreenY = startCardScreenY + (endCardScreenY - startCardScreenY) * easedProgress;
|
|
158
|
+
|
|
159
|
+
const neededScrollLeft = currentCardCenterX - desiredScreenX;
|
|
160
|
+
const neededScrollTop = currentCardCenterY - desiredScreenY;
|
|
161
|
+
|
|
162
|
+
const maxScrollLeft = container.scrollWidth - container.clientWidth;
|
|
163
|
+
const maxScrollTop = container.scrollHeight - container.clientHeight;
|
|
164
|
+
container.scrollLeft = Math.max(0, Math.min(neededScrollLeft, maxScrollLeft));
|
|
165
|
+
container.scrollTop = Math.max(0, Math.min(neededScrollTop, maxScrollTop));
|
|
166
|
+
|
|
167
|
+
if (progress < 1) {
|
|
168
|
+
requestAnimationFrame(animate);
|
|
169
|
+
} else {
|
|
170
|
+
container.style.scrollBehavior = '';
|
|
171
|
+
setIsZooming(false);
|
|
172
|
+
setSelectedItem(null);
|
|
173
|
+
setPreSelectionState(null);
|
|
174
|
+
container.scrollLeft = targetScrollLeft;
|
|
175
|
+
container.scrollTop = targetScrollTop;
|
|
176
|
+
}
|
|
177
|
+
};
|
|
178
|
+
|
|
179
|
+
requestAnimationFrame(animate);
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
const isFirstSelection = selectedItem === null;
|
|
184
|
+
|
|
185
|
+
if (isFirstSelection) {
|
|
186
|
+
setPreSelectionState({
|
|
187
|
+
zoom: zoomLevel,
|
|
188
|
+
scrollLeft: container.scrollLeft,
|
|
189
|
+
scrollTop: container.scrollTop,
|
|
190
|
+
});
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
setSelectedItem(item);
|
|
194
|
+
|
|
195
|
+
if (isFirstSelection) {
|
|
196
|
+
if (viewMode === 'collection') {
|
|
197
|
+
focusCardById(itemId, 0);
|
|
198
|
+
} else {
|
|
199
|
+
const zoomMultiplier = 1.15;
|
|
200
|
+
const minZoom = 1.2;
|
|
201
|
+
const targetZoom = Math.min(ZOOM_MAX, Math.max(minZoom, zoomLevel * zoomMultiplier));
|
|
202
|
+
const shouldZoom = Math.abs(targetZoom - zoomLevel) > 0.001;
|
|
203
|
+
|
|
204
|
+
if (shouldZoom) {
|
|
205
|
+
const selector = `[data-card-id="${escapeAttributeSelectorValue(String(itemId))}"]`;
|
|
206
|
+
const cardEl = container.querySelector(selector) as HTMLElement | null;
|
|
207
|
+
|
|
208
|
+
if (cardEl) {
|
|
209
|
+
setIsZooming(true);
|
|
210
|
+
container.style.scrollBehavior = 'auto';
|
|
211
|
+
|
|
212
|
+
const detailPanelWidth = 380;
|
|
213
|
+
const availableWidth = container.clientWidth - detailPanelWidth;
|
|
214
|
+
const targetScreenX = availableWidth / 2;
|
|
215
|
+
const targetScreenY = container.clientHeight / 2;
|
|
216
|
+
|
|
217
|
+
const startZoom = zoomLevel;
|
|
218
|
+
|
|
219
|
+
const startOffset = getOffsetWithin(cardEl, container);
|
|
220
|
+
const startCardScreenX = startOffset.x + cardEl.offsetWidth / 2 - container.scrollLeft;
|
|
221
|
+
const startCardScreenY = startOffset.y + cardEl.offsetHeight / 2 - container.scrollTop;
|
|
222
|
+
|
|
223
|
+
const duration = 300;
|
|
224
|
+
let startTime: number | null = null;
|
|
225
|
+
|
|
226
|
+
const easeOutCubic = (t: number) => 1 - Math.pow(1 - t, 3);
|
|
227
|
+
|
|
228
|
+
const animate = (timestamp: number) => {
|
|
229
|
+
if (startTime === null) {
|
|
230
|
+
startTime = timestamp;
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
const elapsed = timestamp - startTime;
|
|
234
|
+
const progress = Math.min(1, elapsed / duration);
|
|
235
|
+
const easedProgress = easeOutCubic(progress);
|
|
236
|
+
|
|
237
|
+
const currentZoom = startZoom + (targetZoom - startZoom) * easedProgress;
|
|
238
|
+
|
|
239
|
+
flushSync(() => {
|
|
240
|
+
setZoomLevel(currentZoom);
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
const currentOffset = getOffsetWithin(cardEl, container);
|
|
244
|
+
const currentCardCenterX = currentOffset.x + cardEl.offsetWidth / 2;
|
|
245
|
+
const currentCardCenterY = currentOffset.y + cardEl.offsetHeight / 2;
|
|
246
|
+
|
|
247
|
+
const desiredScreenX = startCardScreenX + (targetScreenX - startCardScreenX) * easedProgress;
|
|
248
|
+
const desiredScreenY = startCardScreenY + (targetScreenY - startCardScreenY) * easedProgress;
|
|
249
|
+
|
|
250
|
+
const neededScrollLeft = currentCardCenterX - desiredScreenX;
|
|
251
|
+
const neededScrollTop = currentCardCenterY - desiredScreenY;
|
|
252
|
+
|
|
253
|
+
const maxScrollLeft = container.scrollWidth - container.clientWidth;
|
|
254
|
+
const maxScrollTop = container.scrollHeight - container.clientHeight;
|
|
255
|
+
container.scrollLeft = Math.max(0, Math.min(neededScrollLeft, maxScrollLeft));
|
|
256
|
+
container.scrollTop = Math.max(0, Math.min(neededScrollTop, maxScrollTop));
|
|
257
|
+
|
|
258
|
+
if (progress < 1) {
|
|
259
|
+
requestAnimationFrame(animate);
|
|
260
|
+
} else {
|
|
261
|
+
container.style.scrollBehavior = '';
|
|
262
|
+
setIsZooming(false);
|
|
263
|
+
requestAnimationFrame(() => {
|
|
264
|
+
focusCardById(itemId, 380);
|
|
265
|
+
});
|
|
266
|
+
}
|
|
267
|
+
};
|
|
268
|
+
|
|
269
|
+
requestAnimationFrame(animate);
|
|
270
|
+
}
|
|
271
|
+
} else {
|
|
272
|
+
focusCardById(itemId, 380);
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
} else {
|
|
276
|
+
focusCardById(itemId, viewMode === 'collection' ? 0 : 380);
|
|
277
|
+
}
|
|
278
|
+
}, [selectedItem, zoomLevel, preSelectionState, resolveId, setZoomLevel, containerRef, viewMode]);
|
|
279
|
+
|
|
280
|
+
const closeDetail = useCallback(() => {
|
|
281
|
+
const container = containerRef.current;
|
|
282
|
+
|
|
283
|
+
if (pendingCenterRaf.current !== null && typeof window !== 'undefined') {
|
|
284
|
+
window.clearTimeout(pendingCenterRaf.current);
|
|
285
|
+
pendingCenterRaf.current = null;
|
|
286
|
+
}
|
|
287
|
+
|
|
288
|
+
if (preSelectionState && container && selectedItem) {
|
|
289
|
+
const itemId = resolveId(selectedItem, 0);
|
|
290
|
+
const selector = `[data-card-id="${escapeAttributeSelectorValue(String(itemId))}"]`;
|
|
291
|
+
const cardEl = container.querySelector(selector) as HTMLElement | null;
|
|
292
|
+
|
|
293
|
+
if (viewMode === 'collection') {
|
|
294
|
+
setSelectedItem(null);
|
|
295
|
+
container.scrollTo({ left: preSelectionState.scrollLeft, top: preSelectionState.scrollTop, behavior: 'smooth' });
|
|
296
|
+
setPreSelectionState(null);
|
|
297
|
+
return;
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
if (cardEl && Math.abs(preSelectionState.zoom - zoomLevel) > 0.001) {
|
|
301
|
+
setIsZooming(true);
|
|
302
|
+
container.style.scrollBehavior = 'auto';
|
|
303
|
+
|
|
304
|
+
const startZoom = zoomLevel;
|
|
305
|
+
const targetZoom = preSelectionState.zoom;
|
|
306
|
+
const targetScrollLeft = preSelectionState.scrollLeft;
|
|
307
|
+
const targetScrollTop = preSelectionState.scrollTop;
|
|
308
|
+
|
|
309
|
+
const startOffset = getOffsetWithin(cardEl, container);
|
|
310
|
+
const startCardScreenX = startOffset.x + cardEl.offsetWidth / 2 - container.scrollLeft;
|
|
311
|
+
const startCardScreenY = startOffset.y + cardEl.offsetHeight / 2 - container.scrollTop;
|
|
312
|
+
|
|
313
|
+
const duration = 300;
|
|
314
|
+
let startTime: number | null = null;
|
|
315
|
+
|
|
316
|
+
const easeOutCubic = (t: number) => 1 - Math.pow(1 - t, 3);
|
|
317
|
+
|
|
318
|
+
const animate = (timestamp: number) => {
|
|
319
|
+
if (startTime === null) {
|
|
320
|
+
startTime = timestamp;
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
const elapsed = timestamp - startTime;
|
|
324
|
+
const progress = Math.min(1, elapsed / duration);
|
|
325
|
+
const easedProgress = easeOutCubic(progress);
|
|
326
|
+
|
|
327
|
+
const currentZoom = startZoom + (targetZoom - startZoom) * easedProgress;
|
|
328
|
+
|
|
329
|
+
flushSync(() => {
|
|
330
|
+
setZoomLevel(currentZoom);
|
|
331
|
+
});
|
|
332
|
+
|
|
333
|
+
const currentOffset = getOffsetWithin(cardEl, container);
|
|
334
|
+
const currentCardCenterX = currentOffset.x + cardEl.offsetWidth / 2;
|
|
335
|
+
const currentCardCenterY = currentOffset.y + cardEl.offsetHeight / 2;
|
|
336
|
+
|
|
337
|
+
const endCardScreenX = currentCardCenterX - targetScrollLeft;
|
|
338
|
+
const endCardScreenY = currentCardCenterY - targetScrollTop;
|
|
339
|
+
|
|
340
|
+
const desiredScreenX = startCardScreenX + (endCardScreenX - startCardScreenX) * easedProgress;
|
|
341
|
+
const desiredScreenY = startCardScreenY + (endCardScreenY - startCardScreenY) * easedProgress;
|
|
342
|
+
|
|
343
|
+
const neededScrollLeft = currentCardCenterX - desiredScreenX;
|
|
344
|
+
const neededScrollTop = currentCardCenterY - desiredScreenY;
|
|
345
|
+
|
|
346
|
+
const maxScrollLeft = container.scrollWidth - container.clientWidth;
|
|
347
|
+
const maxScrollTop = container.scrollHeight - container.clientHeight;
|
|
348
|
+
container.scrollLeft = Math.max(0, Math.min(neededScrollLeft, maxScrollLeft));
|
|
349
|
+
container.scrollTop = Math.max(0, Math.min(neededScrollTop, maxScrollTop));
|
|
350
|
+
|
|
351
|
+
if (progress < 1) {
|
|
352
|
+
requestAnimationFrame(animate);
|
|
353
|
+
} else {
|
|
354
|
+
container.style.scrollBehavior = '';
|
|
355
|
+
setIsZooming(false);
|
|
356
|
+
setSelectedItem(null);
|
|
357
|
+
setPreSelectionState(null);
|
|
358
|
+
container.scrollLeft = targetScrollLeft;
|
|
359
|
+
container.scrollTop = targetScrollTop;
|
|
360
|
+
}
|
|
361
|
+
};
|
|
362
|
+
|
|
363
|
+
requestAnimationFrame(animate);
|
|
364
|
+
} else {
|
|
365
|
+
setSelectedItem(null);
|
|
366
|
+
if (preSelectionState) {
|
|
367
|
+
setZoomLevel(preSelectionState.zoom);
|
|
368
|
+
container.scrollTo({ left: preSelectionState.scrollLeft, top: preSelectionState.scrollTop, behavior: 'smooth' });
|
|
369
|
+
setPreSelectionState(null);
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
} else {
|
|
373
|
+
setSelectedItem(null);
|
|
374
|
+
if (preSelectionState) {
|
|
375
|
+
setPreSelectionState(null);
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
}, [preSelectionState, selectedItem, zoomLevel, setZoomLevel, containerRef, resolveId, viewMode]);
|
|
379
|
+
|
|
380
|
+
const clearSelection = useCallback(() => {
|
|
381
|
+
if (selectedItem) {
|
|
382
|
+
closeDetail();
|
|
383
|
+
}
|
|
384
|
+
}, [selectedItem, closeDetail]);
|
|
385
|
+
|
|
386
|
+
useEffect(() => {
|
|
387
|
+
return () => {
|
|
388
|
+
if (typeof window !== 'undefined' && pendingCenterRaf.current !== null) {
|
|
389
|
+
window.clearTimeout(pendingCenterRaf.current);
|
|
390
|
+
pendingCenterRaf.current = null;
|
|
391
|
+
}
|
|
392
|
+
};
|
|
393
|
+
}, []);
|
|
394
|
+
|
|
395
|
+
return {
|
|
396
|
+
selectedItem,
|
|
397
|
+
isZooming,
|
|
398
|
+
handleCardClick,
|
|
399
|
+
closeDetail,
|
|
400
|
+
clearSelection,
|
|
401
|
+
};
|
|
402
|
+
}
|