@rebasepro/core 0.0.1-canary.0
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 +21 -0
- package/README.md +174 -0
- package/dist/app/AppBar.d.ts +12 -0
- package/dist/app/Drawer.d.ts +19 -0
- package/dist/app/Scaffold.d.ts +34 -0
- package/dist/app/index.d.ts +4 -0
- package/dist/app/useApp.d.ts +17 -0
- package/dist/components/AIIcon.d.ts +16 -0
- package/dist/components/AdminModeSyncer.d.ts +17 -0
- package/dist/components/ArrayContainer.d.ts +59 -0
- package/dist/components/CircularProgressCenter.d.ts +11 -0
- package/dist/components/ClearFilterSortButton.d.ts +5 -0
- package/dist/components/ConfirmationDialog.d.ts +9 -0
- package/dist/components/Debug/UIReferenceView.d.ts +1 -0
- package/dist/components/Debug/UIStyleGuide.d.ts +1 -0
- package/dist/components/DeleteEntityDialog.d.ts +12 -0
- package/dist/components/EntityCollectionTable/EntityCollectionRowActions.d.ts +37 -0
- package/dist/components/EntityCollectionTable/EntityCollectionTable.d.ts +27 -0
- package/dist/components/EntityCollectionTable/EntityCollectionTableProps.d.ts +128 -0
- package/dist/components/EntityCollectionTable/PropertyTableCell.d.ts +25 -0
- package/dist/components/EntityCollectionTable/column_utils.d.ts +15 -0
- package/dist/components/EntityCollectionTable/fields/TableMultipleRelationField.d.ts +20 -0
- package/dist/components/EntityCollectionTable/fields/TableReferenceField.d.ts +21 -0
- package/dist/components/EntityCollectionTable/fields/TableRelationField.d.ts +21 -0
- package/dist/components/EntityCollectionTable/fields/TableRelationSelectorField.d.ts +20 -0
- package/dist/components/EntityCollectionTable/fields/TableStorageUpload.d.ts +32 -0
- package/dist/components/EntityCollectionTable/index.d.ts +6 -0
- package/dist/components/EntityCollectionTable/internal/CollectionTableToolbar.d.ts +16 -0
- package/dist/components/EntityCollectionTable/internal/EntityTableCell.d.ts +32 -0
- package/dist/components/EntityCollectionTable/internal/EntityTableCellActions.d.ts +9 -0
- package/dist/components/EntityCollectionTable/internal/common.d.ts +4 -0
- package/dist/components/EntityCollectionTable/internal/popup_field/PopupFormField.d.ts +26 -0
- package/dist/components/EntityCollectionTable/internal/popup_field/useDraggable.d.ts +13 -0
- package/dist/components/EntityCollectionTable/internal/popup_field/useWindowSize.d.ts +6 -0
- package/dist/components/EntityCollectionView/Board.d.ts +2 -0
- package/dist/components/EntityCollectionView/BoardColumn.d.ts +42 -0
- package/dist/components/EntityCollectionView/BoardColumnTitle.d.ts +9 -0
- package/dist/components/EntityCollectionView/BoardSortableList.d.ts +14 -0
- package/dist/components/EntityCollectionView/EntityBoardCard.d.ts +26 -0
- package/dist/components/EntityCollectionView/EntityCard.d.ts +19 -0
- package/dist/components/EntityCollectionView/EntityCollectionBoardView.d.ts +20 -0
- package/dist/components/EntityCollectionView/EntityCollectionCardView.d.ts +31 -0
- package/dist/components/EntityCollectionView/EntityCollectionView.d.ts +54 -0
- package/dist/components/EntityCollectionView/EntityCollectionViewActions.d.ts +14 -0
- package/dist/components/EntityCollectionView/EntityCollectionViewStartActions.d.ts +15 -0
- package/dist/components/EntityCollectionView/FiltersDialog.d.ts +14 -0
- package/dist/components/EntityCollectionView/ViewModeToggle.d.ts +44 -0
- package/dist/components/EntityCollectionView/board_types.d.ts +105 -0
- package/dist/components/EntityCollectionView/useBoardDataController.d.ts +60 -0
- package/dist/components/EntityCollectionView/useSelectionController.d.ts +2 -0
- package/dist/components/EntityCollectionView/utils.d.ts +3 -0
- package/dist/components/EntityJsonPreview.d.ts +3 -0
- package/dist/components/EntityPreview.d.ts +54 -0
- package/dist/components/EntityView.d.ts +11 -0
- package/dist/components/ErrorBoundary.d.ts +11 -0
- package/dist/components/ErrorTooltip.d.ts +2 -0
- package/dist/components/ErrorView.d.ts +21 -0
- package/dist/components/FieldCaption.d.ts +5 -0
- package/dist/components/HomePage/ContentHomePage.d.ts +11 -0
- package/dist/components/HomePage/FavouritesView.d.ts +3 -0
- package/dist/components/HomePage/HomePageDnD.d.ts +77 -0
- package/dist/components/HomePage/NavigationCard.d.ts +10 -0
- package/dist/components/HomePage/NavigationCardBinding.d.ts +18 -0
- package/dist/components/HomePage/NavigationGroup.d.ts +11 -0
- package/dist/components/HomePage/RenameGroupDialog.d.ts +9 -0
- package/dist/components/HomePage/SmallNavigationCard.d.ts +6 -0
- package/dist/components/HomePage/StudioHomePage.d.ts +9 -0
- package/dist/components/HomePage/index.d.ts +5 -0
- package/dist/components/NotFoundPage.d.ts +1 -0
- package/dist/components/PropertyCollectionView.d.ts +22 -0
- package/dist/components/PropertyConfigBadge.d.ts +6 -0
- package/dist/components/PropertyIdCopyTooltip.d.ts +8 -0
- package/dist/components/RebaseLogo.d.ts +7 -0
- package/dist/components/ReferenceTable/EntitySelectionTable.d.ts +58 -0
- package/dist/components/ReferenceWidget.d.ts +29 -0
- package/dist/components/RelationSelector.d.ts +32 -0
- package/dist/components/SearchIconsView.d.ts +5 -0
- package/dist/components/SelectableTable/SelectableTable.d.ts +89 -0
- package/dist/components/SelectableTable/SelectableTableContext.d.ts +4 -0
- package/dist/components/SelectableTable/filters/BooleanFilterField.d.ts +9 -0
- package/dist/components/SelectableTable/filters/DateTimeFilterField.d.ts +12 -0
- package/dist/components/SelectableTable/filters/ReferenceFilterField.d.ts +15 -0
- package/dist/components/SelectableTable/filters/RelationFilterField.d.ts +12 -0
- package/dist/components/SelectableTable/filters/StringNumberFilterField.d.ts +13 -0
- package/dist/components/UnsavedChangesDialog.d.ts +9 -0
- package/dist/components/UserDisplay.d.ts +7 -0
- package/dist/components/UserSettingsView.d.ts +1 -0
- package/dist/components/VirtualTable/VirtualTable.d.ts +11 -0
- package/dist/components/VirtualTable/VirtualTableCell.d.ts +20 -0
- package/dist/components/VirtualTable/VirtualTableHeader.d.ts +29 -0
- package/dist/components/VirtualTable/VirtualTableHeaderRow.d.ts +2 -0
- package/dist/components/VirtualTable/VirtualTableProps.d.ts +239 -0
- package/dist/components/VirtualTable/VirtualTableRow.d.ts +3 -0
- package/dist/components/VirtualTable/fields/VirtualTableDateField.d.ts +12 -0
- package/dist/components/VirtualTable/fields/VirtualTableInput.d.ts +9 -0
- package/dist/components/VirtualTable/fields/VirtualTableNumberInput.d.ts +8 -0
- package/dist/components/VirtualTable/fields/VirtualTableSelect.d.ts +15 -0
- package/dist/components/VirtualTable/fields/VirtualTableSwitch.d.ts +7 -0
- package/dist/components/VirtualTable/fields/VirtualTableUserSelect.d.ts +12 -0
- package/dist/components/VirtualTable/index.d.ts +3 -0
- package/dist/components/VirtualTable/types.d.ts +37 -0
- package/dist/components/admin/RoleChip.d.ts +4 -0
- package/dist/components/admin/RolesView.d.ts +4 -0
- package/dist/components/admin/UsersView.d.ts +4 -0
- package/dist/components/admin/index.d.ts +3 -0
- package/dist/components/common/default_entity_actions.d.ts +4 -0
- package/dist/components/common/index.d.ts +6 -0
- package/dist/components/common/table_height.d.ts +5 -0
- package/dist/components/common/types.d.ts +58 -0
- package/dist/components/common/useColumnsIds.d.ts +6 -0
- package/dist/components/common/useDataSourceTableController.d.ts +44 -0
- package/dist/components/common/useDebouncedCallback.d.ts +1 -0
- package/dist/components/common/useDebouncedData.d.ts +9 -0
- package/dist/components/common/useScrollRestoration.d.ts +14 -0
- package/dist/components/common/useTableSearchHelper.d.ts +11 -0
- package/dist/components/index.d.ts +35 -0
- package/dist/contexts/AdminModeController.d.ts +4 -0
- package/dist/contexts/AnalyticsContext.d.ts +3 -0
- package/dist/contexts/AuthControllerContext.d.ts +3 -0
- package/dist/contexts/BreacrumbsContext.d.ts +8 -0
- package/dist/contexts/CustomizationControllerContext.d.ts +3 -0
- package/dist/contexts/DataSourceContext.d.ts +3 -0
- package/dist/contexts/DialogsProvider.d.ts +4 -0
- package/dist/contexts/EffectiveRoleController.d.ts +4 -0
- package/dist/contexts/InternalUserManagementContext.d.ts +3 -0
- package/dist/contexts/ModeController.d.ts +4 -0
- package/dist/contexts/NavigationContext.d.ts +3 -0
- package/dist/contexts/SideDialogsControllerContext.d.ts +3 -0
- package/dist/contexts/SideEntityControllerContext.d.ts +3 -0
- package/dist/contexts/SnackbarProvider.d.ts +2 -0
- package/dist/contexts/StorageSourceContext.d.ts +3 -0
- package/dist/contexts/UserConfigurationPersistenceContext.d.ts +3 -0
- package/dist/contexts/index.d.ts +15 -0
- package/dist/core/DefaultAppBar.d.ts +29 -0
- package/dist/core/DefaultDrawer.d.ts +29 -0
- package/dist/core/DrawerNavigationGroup.d.ts +45 -0
- package/dist/core/DrawerNavigationItem.d.ts +10 -0
- package/dist/core/EntityEditView.d.ts +47 -0
- package/dist/core/EntityEditViewFormActions.d.ts +2 -0
- package/dist/core/EntitySidePanel.d.ts +10 -0
- package/dist/core/Rebase.d.ts +13 -0
- package/dist/core/RebaseRouter.d.ts +4 -0
- package/dist/core/RebaseRoutes.d.ts +17 -0
- package/dist/core/SideDialogs.d.ts +25 -0
- package/dist/core/field_configs.d.ts +6 -0
- package/dist/core/index.d.ts +9 -0
- package/dist/form/EntityForm.d.ts +7 -0
- package/dist/form/EntityFormActions.d.ts +2 -0
- package/dist/form/PropertyFieldBinding.d.ts +30 -0
- package/dist/form/components/ErrorFocus.d.ts +4 -0
- package/dist/form/components/FieldHelperText.d.ts +12 -0
- package/dist/form/components/FormEntry.d.ts +6 -0
- package/dist/form/components/FormLayout.d.ts +5 -0
- package/dist/form/components/LabelWithIcon.d.ts +14 -0
- package/dist/form/components/LabelWithIconAndTooltip.d.ts +15 -0
- package/dist/form/components/LocalChangesMenu.d.ts +11 -0
- package/dist/form/components/StorageItemPreview.d.ts +13 -0
- package/dist/form/components/StorageUploadProgress.d.ts +10 -0
- package/dist/form/components/index.d.ts +5 -0
- package/dist/form/field_bindings/ArrayCustomShapedFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/ArrayOfReferencesFieldBinding.d.ts +11 -0
- package/dist/form/field_bindings/BlockFieldBinding.d.ts +10 -0
- package/dist/form/field_bindings/DateTimeFieldBinding.d.ts +11 -0
- package/dist/form/field_bindings/KeyValueFieldBinding.d.ts +7 -0
- package/dist/form/field_bindings/MapFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/MarkdownEditorFieldBinding.d.ts +11 -0
- package/dist/form/field_bindings/MultiSelectFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/MultipleRelationFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/ReadOnlyFieldBinding.d.ts +10 -0
- package/dist/form/field_bindings/ReferenceAsStringFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/ReferenceFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/RelationFieldBinding.d.ts +2 -0
- package/dist/form/field_bindings/RepeatFieldBinding.d.ts +10 -0
- package/dist/form/field_bindings/SelectFieldBinding.d.ts +10 -0
- package/dist/form/field_bindings/StorageUploadFieldBinding.d.ts +19 -0
- package/dist/form/field_bindings/SwitchFieldBinding.d.ts +9 -0
- package/dist/form/field_bindings/TextFieldBinding.d.ts +8 -0
- package/dist/form/field_bindings/UserSelectFieldBinding.d.ts +12 -0
- package/dist/form/index.d.ts +20 -0
- package/dist/form/useClearRestoreValue.d.ts +13 -0
- package/dist/form/validation.d.ts +26 -0
- package/dist/hooks/ApiConfigContext.d.ts +24 -0
- package/dist/hooks/data/delete.d.ts +33 -0
- package/dist/hooks/data/save.d.ts +37 -0
- package/dist/hooks/data/useCollectionFetch.d.ts +50 -0
- package/dist/hooks/data/useDataOrder.d.ts +12 -0
- package/dist/hooks/data/useDataSource.d.ts +6 -0
- package/dist/hooks/data/useEntityFetch.d.ts +29 -0
- package/dist/hooks/data/useRelationSelector.d.ts +45 -0
- package/dist/hooks/index.d.ts +41 -0
- package/dist/hooks/navigation/contexts/CMSUrlContext.d.ts +4 -0
- package/dist/hooks/navigation/contexts/CollectionRegistryContext.d.ts +4 -0
- package/dist/hooks/navigation/contexts/NavigationStateContext.d.ts +4 -0
- package/dist/hooks/navigation/contexts/index.d.ts +3 -0
- package/dist/hooks/navigation/useBuildCMSUrlController.d.ts +6 -0
- package/dist/hooks/navigation/useBuildCollectionRegistryController.d.ts +7 -0
- package/dist/hooks/navigation/useBuildNavigationStateController.d.ts +32 -0
- package/dist/hooks/navigation/useNavigationRegistry.d.ts +10 -0
- package/dist/hooks/navigation/useNavigationResolution.d.ts +5 -0
- package/dist/hooks/navigation/useNavigationURLs.d.ts +11 -0
- package/dist/hooks/navigation/useResolvedCollections.d.ts +26 -0
- package/dist/hooks/navigation/useResolvedViews.d.ts +28 -0
- package/dist/hooks/navigation/useTopLevelNavigation.d.ts +26 -0
- package/dist/hooks/navigation/utils.d.ts +12 -0
- package/dist/hooks/useAdminModeController.d.ts +19 -0
- package/dist/hooks/useAnalyticsController.d.ts +5 -0
- package/dist/hooks/useAuthController.d.ts +11 -0
- package/dist/hooks/useBackendStorageSource.d.ts +30 -0
- package/dist/hooks/useBreadcrumbsController.d.ts +42 -0
- package/dist/hooks/useBrowserTitleAndIcon.d.ts +6 -0
- package/dist/hooks/useBuildAdminModeController.d.ts +6 -0
- package/dist/hooks/useBuildEffectiveRoleController.d.ts +8 -0
- package/dist/hooks/useBuildLocalConfigurationPersistence.d.ts +2 -0
- package/dist/hooks/useBuildModeController.d.ts +6 -0
- package/dist/hooks/useBuildNavigationController.d.ts +16 -0
- package/dist/hooks/useClipboard.d.ts +57 -0
- package/dist/hooks/useCollapsedGroups.d.ts +12 -0
- package/dist/hooks/useCustomizationController.d.ts +11 -0
- package/dist/hooks/useDialogsController.d.ts +11 -0
- package/dist/hooks/useEffectiveRoleController.d.ts +7 -0
- package/dist/hooks/useEntitySelectionDialog.d.ts +18 -0
- package/dist/hooks/useInternalUserManagementController.d.ts +12 -0
- package/dist/hooks/useLargeLayout.d.ts +1 -0
- package/dist/hooks/useModeController.d.ts +19 -0
- package/dist/hooks/usePermissions.d.ts +11 -0
- package/dist/hooks/useRebaseContext.d.ts +11 -0
- package/dist/hooks/useResolvedNavigationFrom.d.ts +72 -0
- package/dist/hooks/useSideDialogsController.d.ts +18 -0
- package/dist/hooks/useSideEntityController.d.ts +12 -0
- package/dist/hooks/useSnackbarController.d.ts +20 -0
- package/dist/hooks/useStorageSource.d.ts +6 -0
- package/dist/hooks/useUnsavedChangesDialog.d.ts +12 -0
- package/dist/hooks/useUserConfigurationPersistence.d.ts +8 -0
- package/dist/hooks/useValidateAuthenticator.d.ts +21 -0
- package/dist/index.d.ts +15 -0
- package/dist/index.es.js +34745 -0
- package/dist/index.es.js.map +1 -0
- package/dist/index.umd.js +34751 -0
- package/dist/index.umd.js.map +1 -0
- package/dist/internal/common.d.ts +3 -0
- package/dist/internal/useBuildDataSource.d.ts +12 -0
- package/dist/internal/useBuildSideDialogsController.d.ts +2 -0
- package/dist/internal/useBuildSideEntityController.d.ts +4 -0
- package/dist/internal/useRestoreScroll.d.ts +6 -0
- package/dist/preview/PropertyPreview.d.ts +6 -0
- package/dist/preview/components/ArrayEnumPreview.d.ts +10 -0
- package/dist/preview/components/AsyncPreviewComponent.d.ts +11 -0
- package/dist/preview/components/BooleanPreview.d.ts +10 -0
- package/dist/preview/components/DatePreview.d.ts +17 -0
- package/dist/preview/components/EmptyValue.d.ts +6 -0
- package/dist/preview/components/EnumValuesChip.d.ts +13 -0
- package/dist/preview/components/ImagePreview.d.ts +16 -0
- package/dist/preview/components/ReferencePreview.d.ts +16 -0
- package/dist/preview/components/RelationPreview.d.ts +16 -0
- package/dist/preview/components/StorageThumbnail.d.ts +15 -0
- package/dist/preview/components/UrlComponentPreview.d.ts +13 -0
- package/dist/preview/components/UserPreview.d.ts +8 -0
- package/dist/preview/index.d.ts +24 -0
- package/dist/preview/property_previews/ArrayOfMapsPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayOfReferencesPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayOfRelationsPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayOfStorageComponentsPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayOfStringsPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayOneOfPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayPropertyEnumPreview.d.ts +5 -0
- package/dist/preview/property_previews/ArrayPropertyPreview.d.ts +5 -0
- package/dist/preview/property_previews/MapPropertyPreview.d.ts +8 -0
- package/dist/preview/property_previews/NumberPropertyPreview.d.ts +6 -0
- package/dist/preview/property_previews/SkeletonPropertyComponent.d.ts +13 -0
- package/dist/preview/property_previews/StringPropertyPreview.d.ts +6 -0
- package/dist/preview/util.d.ts +6 -0
- package/dist/routes/CustomCMSRoute.d.ts +4 -0
- package/dist/routes/RebaseRoute.d.ts +1 -0
- package/dist/routes/index.d.ts +2 -0
- package/dist/util/createFormexStub.d.ts +2 -0
- package/dist/util/entity_cache.d.ts +27 -0
- package/dist/util/enums.d.ts +5 -0
- package/dist/util/icon_list.d.ts +5 -0
- package/dist/util/icon_synonyms.d.ts +1853 -0
- package/dist/util/icons.d.ts +16 -0
- package/dist/util/index.d.ts +11 -0
- package/dist/util/previews.d.ts +3 -0
- package/dist/util/property_utils.d.ts +23 -0
- package/dist/util/useDebouncedCallback.d.ts +1 -0
- package/dist/util/useStorageUploadController.d.ts +36 -0
- package/dist/util/useTraceUpdate.d.ts +2 -0
- package/dist/vitePlugin.d.ts +16 -0
- package/package.json +165 -0
- package/src/app/AppBar.tsx +18 -0
- package/src/app/Drawer.tsx +30 -0
- package/src/app/Scaffold.tsx +238 -0
- package/src/app/index.ts +4 -0
- package/src/app/useApp.tsx +36 -0
- package/src/components/AIIcon.tsx +39 -0
- package/src/components/AdminModeSyncer.tsx +47 -0
- package/src/components/ArrayContainer.tsx +549 -0
- package/src/components/CircularProgressCenter.tsx +26 -0
- package/src/components/ClearFilterSortButton.tsx +44 -0
- package/src/components/ConfirmationDialog.tsx +46 -0
- package/src/components/Debug/UIReferenceView.tsx +710 -0
- package/src/components/Debug/UIStyleGuide.tsx +164 -0
- package/src/components/DeleteEntityDialog.tsx +181 -0
- package/src/components/EntityCollectionTable/EntityCollectionRowActions.tsx +225 -0
- package/src/components/EntityCollectionTable/EntityCollectionTable.tsx +383 -0
- package/src/components/EntityCollectionTable/EntityCollectionTableProps.tsx +180 -0
- package/src/components/EntityCollectionTable/PropertyTableCell.tsx +561 -0
- package/src/components/EntityCollectionTable/column_utils.tsx +74 -0
- package/src/components/EntityCollectionTable/fields/TableMultipleRelationField.tsx +122 -0
- package/src/components/EntityCollectionTable/fields/TableReferenceField.tsx +169 -0
- package/src/components/EntityCollectionTable/fields/TableRelationField.tsx +177 -0
- package/src/components/EntityCollectionTable/fields/TableRelationSelectorField.tsx +42 -0
- package/src/components/EntityCollectionTable/fields/TableStorageUpload.tsx +315 -0
- package/src/components/EntityCollectionTable/index.tsx +12 -0
- package/src/components/EntityCollectionTable/internal/CollectionTableToolbar.tsx +90 -0
- package/src/components/EntityCollectionTable/internal/EntityTableCell.tsx +311 -0
- package/src/components/EntityCollectionTable/internal/EntityTableCellActions.tsx +82 -0
- package/src/components/EntityCollectionTable/internal/common.tsx +72 -0
- package/src/components/EntityCollectionTable/internal/popup_field/PopupFormField.tsx +425 -0
- package/src/components/EntityCollectionTable/internal/popup_field/useDraggable.tsx +96 -0
- package/src/components/EntityCollectionTable/internal/popup_field/useWindowSize.tsx +20 -0
- package/src/components/EntityCollectionView/Board.tsx +324 -0
- package/src/components/EntityCollectionView/BoardColumn.tsx +158 -0
- package/src/components/EntityCollectionView/BoardColumnTitle.tsx +45 -0
- package/src/components/EntityCollectionView/BoardSortableList.tsx +172 -0
- package/src/components/EntityCollectionView/EntityBoardCard.tsx +200 -0
- package/src/components/EntityCollectionView/EntityCard.tsx +225 -0
- package/src/components/EntityCollectionView/EntityCollectionBoardView.tsx +746 -0
- package/src/components/EntityCollectionView/EntityCollectionCardView.tsx +254 -0
- package/src/components/EntityCollectionView/EntityCollectionView.tsx +1220 -0
- package/src/components/EntityCollectionView/EntityCollectionViewActions.tsx +142 -0
- package/src/components/EntityCollectionView/EntityCollectionViewStartActions.tsx +131 -0
- package/src/components/EntityCollectionView/FiltersDialog.tsx +249 -0
- package/src/components/EntityCollectionView/ViewModeToggle.tsx +194 -0
- package/src/components/EntityCollectionView/board_types.ts +113 -0
- package/src/components/EntityCollectionView/useBoardDataController.tsx +490 -0
- package/src/components/EntityCollectionView/useSelectionController.tsx +43 -0
- package/src/components/EntityCollectionView/utils.ts +19 -0
- package/src/components/EntityJsonPreview.tsx +66 -0
- package/src/components/EntityPreview.tsx +367 -0
- package/src/components/EntityView.tsx +66 -0
- package/src/components/ErrorBoundary.tsx +40 -0
- package/src/components/ErrorTooltip.tsx +12 -0
- package/src/components/ErrorView.tsx +69 -0
- package/src/components/FieldCaption.tsx +14 -0
- package/src/components/HomePage/ContentHomePage.tsx +634 -0
- package/src/components/HomePage/FavouritesView.tsx +59 -0
- package/src/components/HomePage/HomePageDnD.tsx +702 -0
- package/src/components/HomePage/NavigationCard.tsx +80 -0
- package/src/components/HomePage/NavigationCardBinding.tsx +111 -0
- package/src/components/HomePage/NavigationGroup.tsx +154 -0
- package/src/components/HomePage/RenameGroupDialog.tsx +121 -0
- package/src/components/HomePage/SmallNavigationCard.tsx +45 -0
- package/src/components/HomePage/StudioHomePage.tsx +231 -0
- package/src/components/HomePage/index.tsx +6 -0
- package/src/components/NotFoundPage.tsx +25 -0
- package/src/components/PropertyCollectionView.tsx +333 -0
- package/src/components/PropertyConfigBadge.tsx +27 -0
- package/src/components/PropertyIdCopyTooltip.tsx +47 -0
- package/src/components/RebaseLogo.tsx +29 -0
- package/src/components/ReferenceTable/EntitySelectionTable.tsx +371 -0
- package/src/components/ReferenceWidget.tsx +152 -0
- package/src/components/RelationSelector.tsx +518 -0
- package/src/components/SearchIconsView.tsx +78 -0
- package/src/components/SelectableTable/SelectableTable.tsx +344 -0
- package/src/components/SelectableTable/SelectableTableContext.tsx +6 -0
- package/src/components/SelectableTable/filters/BooleanFilterField.tsx +49 -0
- package/src/components/SelectableTable/filters/DateTimeFilterField.tsx +126 -0
- package/src/components/SelectableTable/filters/ReferenceFilterField.tsx +203 -0
- package/src/components/SelectableTable/filters/RelationFilterField.tsx +138 -0
- package/src/components/SelectableTable/filters/StringNumberFilterField.tsx +217 -0
- package/src/components/UnsavedChangesDialog.tsx +45 -0
- package/src/components/UserDisplay.tsx +55 -0
- package/src/components/UserSettingsView.tsx +220 -0
- package/src/components/VirtualTable/VirtualTable.performance.test.tsx +386 -0
- package/src/components/VirtualTable/VirtualTable.tsx +625 -0
- package/src/components/VirtualTable/VirtualTableCell.tsx +58 -0
- package/src/components/VirtualTable/VirtualTableHeader.tsx +275 -0
- package/src/components/VirtualTable/VirtualTableHeaderRow.tsx +249 -0
- package/src/components/VirtualTable/VirtualTableProps.tsx +298 -0
- package/src/components/VirtualTable/VirtualTableRow.tsx +52 -0
- package/src/components/VirtualTable/fields/VirtualTableDateField.tsx +39 -0
- package/src/components/VirtualTable/fields/VirtualTableInput.tsx +93 -0
- package/src/components/VirtualTable/fields/VirtualTableNumberInput.tsx +83 -0
- package/src/components/VirtualTable/fields/VirtualTableSelect.tsx +133 -0
- package/src/components/VirtualTable/fields/VirtualTableSwitch.tsx +32 -0
- package/src/components/VirtualTable/fields/VirtualTableUserSelect.tsx +111 -0
- package/src/components/VirtualTable/index.tsx +3 -0
- package/src/components/VirtualTable/types.tsx +46 -0
- package/src/components/admin/RoleChip.tsx +23 -0
- package/src/components/admin/RolesView.tsx +408 -0
- package/src/components/admin/UsersView.tsx +353 -0
- package/src/components/admin/index.ts +3 -0
- package/src/components/common/default_entity_actions.tsx +144 -0
- package/src/components/common/index.ts +6 -0
- package/src/components/common/table_height.tsx +21 -0
- package/src/components/common/types.tsx +61 -0
- package/src/components/common/useColumnsIds.tsx +210 -0
- package/src/components/common/useDataSourceTableController.tsx +480 -0
- package/src/components/common/useDebouncedCallback.tsx +20 -0
- package/src/components/common/useDebouncedData.ts +49 -0
- package/src/components/common/useScrollRestoration.tsx +68 -0
- package/src/components/common/useTableSearchHelper.ts +75 -0
- package/src/components/index.tsx +49 -0
- package/src/contexts/AdminModeController.tsx +11 -0
- package/src/contexts/AnalyticsContext.tsx +4 -0
- package/src/contexts/AuthControllerContext.tsx +4 -0
- package/src/contexts/BreacrumbsContext.tsx +45 -0
- package/src/contexts/CustomizationControllerContext.tsx +4 -0
- package/src/contexts/DataSourceContext.tsx +4 -0
- package/src/contexts/DialogsProvider.tsx +53 -0
- package/src/contexts/EffectiveRoleController.tsx +11 -0
- package/src/contexts/InternalUserManagementContext.tsx +4 -0
- package/src/contexts/ModeController.tsx +11 -0
- package/src/contexts/NavigationContext.tsx +4 -0
- package/src/contexts/SideDialogsControllerContext.tsx +4 -0
- package/src/contexts/SideEntityControllerContext.tsx +4 -0
- package/src/contexts/SnackbarProvider.tsx +14 -0
- package/src/contexts/StorageSourceContext.tsx +4 -0
- package/src/contexts/UserConfigurationPersistenceContext.tsx +4 -0
- package/src/contexts/index.ts +15 -0
- package/src/core/DefaultAppBar.tsx +274 -0
- package/src/core/DefaultDrawer.tsx +267 -0
- package/src/core/DrawerNavigationGroup.tsx +117 -0
- package/src/core/DrawerNavigationItem.tsx +65 -0
- package/src/core/EntityEditView.tsx +590 -0
- package/src/core/EntityEditViewFormActions.tsx +343 -0
- package/src/core/EntitySidePanel.tsx +173 -0
- package/src/core/Rebase.tsx +229 -0
- package/src/core/RebaseRouter.tsx +17 -0
- package/src/core/RebaseRoutes.tsx +47 -0
- package/src/core/SideDialogs.tsx +200 -0
- package/src/core/field_configs.tsx +443 -0
- package/src/core/index.tsx +14 -0
- package/src/form/EntityForm.tsx +820 -0
- package/src/form/EntityFormActions.tsx +201 -0
- package/src/form/PropertyFieldBinding.tsx +348 -0
- package/src/form/components/ErrorFocus.tsx +44 -0
- package/src/form/components/FieldHelperText.tsx +45 -0
- package/src/form/components/FormEntry.tsx +22 -0
- package/src/form/components/FormLayout.tsx +16 -0
- package/src/form/components/LabelWithIcon.tsx +43 -0
- package/src/form/components/LabelWithIconAndTooltip.tsx +28 -0
- package/src/form/components/LocalChangesMenu.tsx +144 -0
- package/src/form/components/StorageItemPreview.tsx +79 -0
- package/src/form/components/StorageUploadProgress.tsx +105 -0
- package/src/form/components/index.tsx +5 -0
- package/src/form/field_bindings/ArrayCustomShapedFieldBinding.tsx +105 -0
- package/src/form/field_bindings/ArrayOfReferencesFieldBinding.tsx +163 -0
- package/src/form/field_bindings/BlockFieldBinding.tsx +268 -0
- package/src/form/field_bindings/DateTimeFieldBinding.tsx +70 -0
- package/src/form/field_bindings/KeyValueFieldBinding.tsx +567 -0
- package/src/form/field_bindings/MapFieldBinding.tsx +156 -0
- package/src/form/field_bindings/MarkdownEditorFieldBinding.tsx +146 -0
- package/src/form/field_bindings/MultiSelectFieldBinding.tsx +120 -0
- package/src/form/field_bindings/MultipleRelationFieldBinding.tsx +151 -0
- package/src/form/field_bindings/ReadOnlyFieldBinding.tsx +64 -0
- package/src/form/field_bindings/ReferenceAsStringFieldBinding.tsx +135 -0
- package/src/form/field_bindings/ReferenceFieldBinding.tsx +134 -0
- package/src/form/field_bindings/RelationFieldBinding.tsx +176 -0
- package/src/form/field_bindings/RepeatFieldBinding.tsx +136 -0
- package/src/form/field_bindings/SelectFieldBinding.tsx +107 -0
- package/src/form/field_bindings/StorageUploadFieldBinding.tsx +490 -0
- package/src/form/field_bindings/SwitchFieldBinding.tsx +64 -0
- package/src/form/field_bindings/TextFieldBinding.tsx +153 -0
- package/src/form/field_bindings/UserSelectFieldBinding.tsx +96 -0
- package/src/form/index.tsx +27 -0
- package/src/form/useClearRestoreValue.tsx +35 -0
- package/src/form/validation.ts +532 -0
- package/src/hooks/ApiConfigContext.tsx +40 -0
- package/src/hooks/data/delete.ts +77 -0
- package/src/hooks/data/save.ts +75 -0
- package/src/hooks/data/useCollectionFetch.tsx +148 -0
- package/src/hooks/data/useDataOrder.ts +26 -0
- package/src/hooks/data/useDataSource.tsx +22 -0
- package/src/hooks/data/useEntityFetch.tsx +121 -0
- package/src/hooks/data/useRelationSelector.tsx +205 -0
- package/src/hooks/index.tsx +51 -0
- package/src/hooks/navigation/contexts/CMSUrlContext.tsx +24 -0
- package/src/hooks/navigation/contexts/CollectionRegistryContext.tsx +19 -0
- package/src/hooks/navigation/contexts/NavigationStateContext.tsx +15 -0
- package/src/hooks/navigation/contexts/index.ts +14 -0
- package/src/hooks/navigation/useBuildCMSUrlController.tsx +68 -0
- package/src/hooks/navigation/useBuildCollectionRegistryController.tsx +150 -0
- package/src/hooks/navigation/useBuildNavigationStateController.tsx +135 -0
- package/src/hooks/navigation/useNavigationRegistry.ts +142 -0
- package/src/hooks/navigation/useNavigationResolution.ts +98 -0
- package/src/hooks/navigation/useNavigationURLs.ts +56 -0
- package/src/hooks/navigation/useResolvedCollections.ts +139 -0
- package/src/hooks/navigation/useResolvedViews.tsx +204 -0
- package/src/hooks/navigation/useTopLevelNavigation.ts +265 -0
- package/src/hooks/navigation/utils.ts +177 -0
- package/src/hooks/useAdminModeController.tsx +23 -0
- package/src/hooks/useAnalyticsController.tsx +8 -0
- package/src/hooks/useAuthController.tsx +14 -0
- package/src/hooks/useBackendStorageSource.ts +276 -0
- package/src/hooks/useBreadcrumbsController.tsx +49 -0
- package/src/hooks/useBrowserTitleAndIcon.tsx +25 -0
- package/src/hooks/useBuildAdminModeController.tsx +24 -0
- package/src/hooks/useBuildEffectiveRoleController.tsx +30 -0
- package/src/hooks/useBuildLocalConfigurationPersistence.tsx +65 -0
- package/src/hooks/useBuildModeController.tsx +71 -0
- package/src/hooks/useBuildNavigationController.tsx +341 -0
- package/src/hooks/useClipboard.tsx +158 -0
- package/src/hooks/useCollapsedGroups.ts +72 -0
- package/src/hooks/useCustomizationController.tsx +14 -0
- package/src/hooks/useDialogsController.tsx +14 -0
- package/src/hooks/useEffectiveRoleController.tsx +10 -0
- package/src/hooks/useEntitySelectionDialog.tsx +56 -0
- package/src/hooks/useInternalUserManagementController.tsx +17 -0
- package/src/hooks/useLargeLayout.tsx +65 -0
- package/src/hooks/useModeController.tsx +23 -0
- package/src/hooks/usePermissions.ts +43 -0
- package/src/hooks/useRebaseContext.tsx +86 -0
- package/src/hooks/useResolvedNavigationFrom.tsx +158 -0
- package/src/hooks/useSideDialogsController.tsx +21 -0
- package/src/hooks/useSideEntityController.tsx +15 -0
- package/src/hooks/useSnackbarController.tsx +52 -0
- package/src/hooks/useStorageSource.tsx +14 -0
- package/src/hooks/useUnsavedChangesDialog.tsx +62 -0
- package/src/hooks/useUserConfigurationPersistence.tsx +11 -0
- package/src/hooks/useValidateAuthenticator.tsx +115 -0
- package/src/index.ts +17 -0
- package/src/internal/common.tsx +5 -0
- package/src/internal/useBuildDataSource.ts +380 -0
- package/src/internal/useBuildSideDialogsController.tsx +135 -0
- package/src/internal/useBuildSideEntityController.tsx +309 -0
- package/src/internal/useRestoreScroll.tsx +60 -0
- package/src/preview/PropertyPreview.tsx +308 -0
- package/src/preview/components/ArrayEnumPreview.tsx +38 -0
- package/src/preview/components/AsyncPreviewComponent.tsx +47 -0
- package/src/preview/components/BooleanPreview.tsx +25 -0
- package/src/preview/components/DatePreview.tsx +94 -0
- package/src/preview/components/EmptyValue.tsx +10 -0
- package/src/preview/components/EnumValuesChip.tsx +39 -0
- package/src/preview/components/ImagePreview.tsx +111 -0
- package/src/preview/components/ReferencePreview.tsx +162 -0
- package/src/preview/components/RelationPreview.tsx +153 -0
- package/src/preview/components/StorageThumbnail.tsx +91 -0
- package/src/preview/components/UrlComponentPreview.tsx +113 -0
- package/src/preview/components/UserPreview.tsx +27 -0
- package/src/preview/index.ts +26 -0
- package/src/preview/property_previews/ArrayOfMapsPreview.tsx +72 -0
- package/src/preview/property_previews/ArrayOfReferencesPreview.tsx +43 -0
- package/src/preview/property_previews/ArrayOfRelationsPreview.tsx +52 -0
- package/src/preview/property_previews/ArrayOfStorageComponentsPreview.tsx +49 -0
- package/src/preview/property_previews/ArrayOfStringsPreview.tsx +44 -0
- package/src/preview/property_previews/ArrayOneOfPreview.tsx +67 -0
- package/src/preview/property_previews/ArrayPropertyEnumPreview.tsx +34 -0
- package/src/preview/property_previews/ArrayPropertyPreview.tsx +69 -0
- package/src/preview/property_previews/MapPropertyPreview.tsx +145 -0
- package/src/preview/property_previews/NumberPropertyPreview.tsx +28 -0
- package/src/preview/property_previews/SkeletonPropertyComponent.tsx +275 -0
- package/src/preview/property_previews/StringPropertyPreview.tsx +57 -0
- package/src/preview/util.ts +30 -0
- package/src/routes/CustomCMSRoute.tsx +21 -0
- package/src/routes/RebaseRoute.tsx +255 -0
- package/src/routes/index.ts +2 -0
- package/src/util/createFormexStub.tsx +66 -0
- package/src/util/entity_cache.ts +267 -0
- package/src/util/enums.ts +32 -0
- package/src/util/icon_list.ts +23 -0
- package/src/util/icon_synonyms.ts +1853 -0
- package/src/util/icons.tsx +87 -0
- package/src/util/index.ts +11 -0
- package/src/util/previews.ts +50 -0
- package/src/util/property_utils.tsx +152 -0
- package/src/util/useDebouncedCallback.ts +25 -0
- package/src/util/useStorageUploadController.tsx +342 -0
- package/src/util/useTraceUpdate.tsx +24 -0
- package/src/vitePlugin.ts +47 -0
|
@@ -0,0 +1,220 @@
|
|
|
1
|
+
import React, { useEffect, useState } from "react";
|
|
2
|
+
import {
|
|
3
|
+
Avatar,
|
|
4
|
+
Button,
|
|
5
|
+
CircularProgress,
|
|
6
|
+
Tabs,
|
|
7
|
+
Tab,
|
|
8
|
+
TextField,
|
|
9
|
+
Typography,
|
|
10
|
+
IconButton,
|
|
11
|
+
DeleteIcon,
|
|
12
|
+
} from "@rebasepro/ui";
|
|
13
|
+
import { useAuthController } from "../hooks";
|
|
14
|
+
|
|
15
|
+
interface SessionInfo {
|
|
16
|
+
id: string;
|
|
17
|
+
userAgent?: string;
|
|
18
|
+
ipAddress?: string;
|
|
19
|
+
createdAt: string;
|
|
20
|
+
isCurrentSession?: boolean;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
interface ExtendedAuthController {
|
|
24
|
+
user: { displayName?: string | null; photoURL?: string | null; email?: string | null } | null;
|
|
25
|
+
updateProfile?: (displayName: string, photoURL: string) => Promise<void>;
|
|
26
|
+
fetchSessions?: () => Promise<SessionInfo[]>;
|
|
27
|
+
revokeSession?: (id: string) => Promise<void>;
|
|
28
|
+
revokeAllSessions?: () => Promise<void>;
|
|
29
|
+
signOut: () => Promise<void>;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
export function UserSettingsView() {
|
|
33
|
+
const authController = useAuthController() as unknown as ExtendedAuthController;
|
|
34
|
+
const user = authController.user;
|
|
35
|
+
|
|
36
|
+
const [activeTab, setActiveTab] = useState<"profile" | "sessions">("profile");
|
|
37
|
+
|
|
38
|
+
// Profile state
|
|
39
|
+
const [displayName, setDisplayName] = useState(user?.displayName || "");
|
|
40
|
+
const [photoURL, setPhotoURL] = useState(user?.photoURL || "");
|
|
41
|
+
const [savingProfile, setSavingProfile] = useState(false);
|
|
42
|
+
const [profileError, setProfileError] = useState<string | null>(null);
|
|
43
|
+
|
|
44
|
+
// Sessions state
|
|
45
|
+
const [sessions, setSessions] = useState<SessionInfo[]>([]);
|
|
46
|
+
const [loadingSessions, setLoadingSessions] = useState(false);
|
|
47
|
+
const [sessionsError, setSessionsError] = useState<string | null>(null);
|
|
48
|
+
const [revokingSessionId, setRevokingSessionId] = useState<string | null>(null);
|
|
49
|
+
const [revokingAll, setRevokingAll] = useState(false);
|
|
50
|
+
|
|
51
|
+
useEffect(() => {
|
|
52
|
+
setDisplayName(user?.displayName || "");
|
|
53
|
+
setPhotoURL(user?.photoURL || "");
|
|
54
|
+
if (activeTab === "sessions") {
|
|
55
|
+
loadSessions();
|
|
56
|
+
}
|
|
57
|
+
}, [activeTab, user]);
|
|
58
|
+
|
|
59
|
+
const handleSaveProfile = async () => {
|
|
60
|
+
setSavingProfile(true);
|
|
61
|
+
setProfileError(null);
|
|
62
|
+
try {
|
|
63
|
+
if (authController.updateProfile) {
|
|
64
|
+
await authController.updateProfile(displayName, photoURL);
|
|
65
|
+
} else {
|
|
66
|
+
throw new Error("updateProfile not implemented in this auth controller.");
|
|
67
|
+
}
|
|
68
|
+
} catch (e: unknown) {
|
|
69
|
+
setProfileError(e instanceof Error ? e.message : String(e));
|
|
70
|
+
} finally {
|
|
71
|
+
setSavingProfile(false);
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
|
|
75
|
+
const loadSessions = async () => {
|
|
76
|
+
setLoadingSessions(true);
|
|
77
|
+
setSessionsError(null);
|
|
78
|
+
try {
|
|
79
|
+
if (authController.fetchSessions) {
|
|
80
|
+
const fetchedSessions = await authController.fetchSessions();
|
|
81
|
+
const sortedSessions = (fetchedSessions || []).sort((a: SessionInfo, b: SessionInfo) =>
|
|
82
|
+
new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime()
|
|
83
|
+
);
|
|
84
|
+
setSessions(sortedSessions);
|
|
85
|
+
} else {
|
|
86
|
+
throw new Error("fetchSessions not implemented in this auth controller.");
|
|
87
|
+
}
|
|
88
|
+
} catch (e: unknown) {
|
|
89
|
+
setSessionsError(e instanceof Error ? e.message : String(e));
|
|
90
|
+
} finally {
|
|
91
|
+
setLoadingSessions(false);
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
|
|
95
|
+
const handleRevokeSession = async (id: string, isCurrentSession?: boolean) => {
|
|
96
|
+
setRevokingSessionId(id);
|
|
97
|
+
try {
|
|
98
|
+
if (authController.revokeSession) {
|
|
99
|
+
await authController.revokeSession(id);
|
|
100
|
+
setSessions(sessions.filter(s => s.id !== id));
|
|
101
|
+
if (isCurrentSession) {
|
|
102
|
+
await authController.signOut();
|
|
103
|
+
}
|
|
104
|
+
} else {
|
|
105
|
+
throw new Error("revokeSession not implemented in this auth controller.");
|
|
106
|
+
}
|
|
107
|
+
} catch (e: unknown) {
|
|
108
|
+
setSessionsError(e instanceof Error ? e.message : String(e));
|
|
109
|
+
} finally {
|
|
110
|
+
setRevokingSessionId(null);
|
|
111
|
+
}
|
|
112
|
+
};
|
|
113
|
+
|
|
114
|
+
const handleRevokeAll = async () => {
|
|
115
|
+
setRevokingAll(true);
|
|
116
|
+
try {
|
|
117
|
+
if (authController.revokeAllSessions) {
|
|
118
|
+
await authController.revokeAllSessions();
|
|
119
|
+
} else {
|
|
120
|
+
throw new Error("revokeAllSessions not implemented in this auth controller.");
|
|
121
|
+
}
|
|
122
|
+
} catch (e: unknown) {
|
|
123
|
+
setSessionsError(e instanceof Error ? e.message : String(e));
|
|
124
|
+
} finally {
|
|
125
|
+
setRevokingAll(false);
|
|
126
|
+
}
|
|
127
|
+
};
|
|
128
|
+
|
|
129
|
+
if (!user) return null;
|
|
130
|
+
|
|
131
|
+
return (
|
|
132
|
+
<div className="flex-grow max-w-4xl w-full mx-auto p-4 sm:p-6 md:p-12">
|
|
133
|
+
<Typography variant="h4" className="mb-8">Account Settings</Typography>
|
|
134
|
+
|
|
135
|
+
<Tabs value={activeTab} onValueChange={(v) => setActiveTab(v as "profile" | "sessions")} className="mb-8">
|
|
136
|
+
<Tab value="profile">Profile</Tab>
|
|
137
|
+
<Tab value="sessions">Sessions</Tab>
|
|
138
|
+
</Tabs>
|
|
139
|
+
|
|
140
|
+
{activeTab === "profile" && (
|
|
141
|
+
<div className="flex flex-col gap-6 max-w-xl">
|
|
142
|
+
<div className="flex items-center gap-6 mb-2">
|
|
143
|
+
<Avatar src={photoURL || undefined} className="w-24 h-24 text-3xl">
|
|
144
|
+
{displayName ? displayName[0].toUpperCase() : (user.email ? user.email[0].toUpperCase() : "A")}
|
|
145
|
+
</Avatar>
|
|
146
|
+
</div>
|
|
147
|
+
<TextField
|
|
148
|
+
label="Display Name"
|
|
149
|
+
value={displayName}
|
|
150
|
+
onChange={(e) => setDisplayName(e.target.value)}
|
|
151
|
+
/>
|
|
152
|
+
<TextField
|
|
153
|
+
label="Photo URL"
|
|
154
|
+
value={photoURL}
|
|
155
|
+
onChange={(e) => setPhotoURL(e.target.value)}
|
|
156
|
+
/>
|
|
157
|
+
{profileError && <Typography color="error">{profileError}</Typography>}
|
|
158
|
+
<div className="mt-4">
|
|
159
|
+
<Button variant="filled" onClick={handleSaveProfile} disabled={savingProfile}>
|
|
160
|
+
{savingProfile ? "Saving..." : "Save Profile"}
|
|
161
|
+
</Button>
|
|
162
|
+
</div>
|
|
163
|
+
</div>
|
|
164
|
+
)}
|
|
165
|
+
|
|
166
|
+
{activeTab === "sessions" && (
|
|
167
|
+
<div className="flex flex-col gap-4 max-w-3xl">
|
|
168
|
+
{loadingSessions ? (
|
|
169
|
+
<div className="flex justify-center p-8"><CircularProgress /></div>
|
|
170
|
+
) : sessionsError ? (
|
|
171
|
+
<Typography color="error">{sessionsError}</Typography>
|
|
172
|
+
) : sessions.length === 0 ? (
|
|
173
|
+
<Typography>No active sessions found.</Typography>
|
|
174
|
+
) : (
|
|
175
|
+
<div className="flex flex-col gap-4">
|
|
176
|
+
<div className="flex justify-end mb-2">
|
|
177
|
+
<Button
|
|
178
|
+
variant="text"
|
|
179
|
+
color="error"
|
|
180
|
+
onClick={handleRevokeAll}
|
|
181
|
+
disabled={revokingAll}
|
|
182
|
+
>
|
|
183
|
+
{revokingAll ? "Revoking..." : "Revoke All Sessions"}
|
|
184
|
+
</Button>
|
|
185
|
+
</div>
|
|
186
|
+
{sessions.map(session => (
|
|
187
|
+
<div key={session.id} className="flex justify-between items-center p-4 bg-white dark:bg-surface-800 border rounded-lg dark:border-surface-700 shadow-sm">
|
|
188
|
+
<div className="flex flex-col">
|
|
189
|
+
<div className="flex items-center gap-2 mb-1">
|
|
190
|
+
<Typography variant="body1">
|
|
191
|
+
{session.userAgent || "Unknown Device"}
|
|
192
|
+
</Typography>
|
|
193
|
+
{session.isCurrentSession && (
|
|
194
|
+
<span className="px-2 py-0.5 text-[10px] font-medium uppercase tracking-wider bg-primary-100 text-primary-800 dark:bg-primary-900/50 dark:text-primary-200 rounded-full">
|
|
195
|
+
Current
|
|
196
|
+
</span>
|
|
197
|
+
)}
|
|
198
|
+
</div>
|
|
199
|
+
<Typography variant="caption" color="secondary">
|
|
200
|
+
IP: {session.ipAddress || "Unknown"} • Created: {new Date(session.createdAt).toLocaleString()}
|
|
201
|
+
</Typography>
|
|
202
|
+
</div>
|
|
203
|
+
<div className="ml-4">
|
|
204
|
+
{revokingSessionId === session.id ? (
|
|
205
|
+
<CircularProgress size="small" />
|
|
206
|
+
) : (
|
|
207
|
+
<IconButton onClick={() => handleRevokeSession(session.id, session.isCurrentSession)} aria-label="Revoke Session">
|
|
208
|
+
<DeleteIcon />
|
|
209
|
+
</IconButton>
|
|
210
|
+
)}
|
|
211
|
+
</div>
|
|
212
|
+
</div>
|
|
213
|
+
))}
|
|
214
|
+
</div>
|
|
215
|
+
)}
|
|
216
|
+
</div>
|
|
217
|
+
)}
|
|
218
|
+
</div>
|
|
219
|
+
);
|
|
220
|
+
}
|
|
@@ -0,0 +1,386 @@
|
|
|
1
|
+
|
|
2
|
+
/**
|
|
3
|
+
* @jest-environment jsdom
|
|
4
|
+
*/
|
|
5
|
+
import React from 'react';
|
|
6
|
+
import { render, act } from '@testing-library/react';
|
|
7
|
+
import { VirtualTable } from './VirtualTable';
|
|
8
|
+
import { VirtualTableProps } from './VirtualTableProps';
|
|
9
|
+
|
|
10
|
+
// Mock VirtualTableRow to track renders
|
|
11
|
+
const renderCallback = jest.fn();
|
|
12
|
+
jest.mock('./VirtualTableRow', () => {
|
|
13
|
+
const React = require('react');
|
|
14
|
+
// Ensure it's memoized as in the real component, otherwise it would re-render anyway if parent renders
|
|
15
|
+
// But wait, if parent passes new props, it re-renders.
|
|
16
|
+
// The issue we are testing is CONTEXT.
|
|
17
|
+
// VirtualTableRow consumes context. If context value changes, it re-renders even if props are same.
|
|
18
|
+
// So we don't even need to pass props to it in the mock, but the real one does.
|
|
19
|
+
|
|
20
|
+
const VirtualTableRow = React.memo((props: any) => {
|
|
21
|
+
renderCallback();
|
|
22
|
+
return <div data-testid="row">{props.children}</div>;
|
|
23
|
+
});
|
|
24
|
+
return { VirtualTableRow };
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
// Mock react-use-measure
|
|
28
|
+
jest.mock('react-use-measure', () => {
|
|
29
|
+
return () => [
|
|
30
|
+
(element: any) => {},
|
|
31
|
+
{ width: 500, height: 500, top: 0, left: 0, bottom: 0, right: 0, x: 0, y: 0 }
|
|
32
|
+
];
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
// Mock ResizeObserver
|
|
36
|
+
global.ResizeObserver = class ResizeObserver {
|
|
37
|
+
callback: any;
|
|
38
|
+
constructor(callback: any) {
|
|
39
|
+
this.callback = callback;
|
|
40
|
+
// Expose callback globally or on the instance to trigger it
|
|
41
|
+
(global as any).triggerResize = callback;
|
|
42
|
+
}
|
|
43
|
+
observe() {}
|
|
44
|
+
unobserve() {}
|
|
45
|
+
disconnect() {}
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
describe('VirtualTable Performance', () => {
|
|
49
|
+
it('does not re-render rows when VirtualTable re-renders but data is unchanged', async () => {
|
|
50
|
+
const columns = [{ key: 'col1', title: 'Column 1', width: 100 }];
|
|
51
|
+
const data = Array.from({ length: 10 }).map((_, i) => ({ col1: `Value ${i}` }));
|
|
52
|
+
|
|
53
|
+
const props: VirtualTableProps<any> = {
|
|
54
|
+
data,
|
|
55
|
+
columns,
|
|
56
|
+
rowHeight: 50,
|
|
57
|
+
cellRenderer: () => <div>Cell</div>
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
render(<VirtualTable {...props} />);
|
|
61
|
+
|
|
62
|
+
// Initial render count
|
|
63
|
+
// 10 items + potentially overscan.
|
|
64
|
+
// Let's just store the current count.
|
|
65
|
+
const initialRenderCount = renderCallback.mock.calls.length;
|
|
66
|
+
console.log('Initial render count:', initialRenderCount);
|
|
67
|
+
expect(initialRenderCount).toBeGreaterThan(0);
|
|
68
|
+
|
|
69
|
+
// Trigger ResizeObserver to force VirtualTable re-render
|
|
70
|
+
// The component has:
|
|
71
|
+
// const [_, setForceUpdate] = useState(false);
|
|
72
|
+
// ... new ResizeObserver(() => setForceUpdate(prev => !prev))
|
|
73
|
+
|
|
74
|
+
act(() => {
|
|
75
|
+
if ((global as any).triggerResize) {
|
|
76
|
+
(global as any).triggerResize([]);
|
|
77
|
+
}
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
// After re-render
|
|
81
|
+
const afterResizeRenderCount = renderCallback.mock.calls.length;
|
|
82
|
+
console.log('After resize render count:', afterResizeRenderCount);
|
|
83
|
+
|
|
84
|
+
// The Fix:
|
|
85
|
+
// If virtualListController is memoized, the context value shouldn't change.
|
|
86
|
+
// VirtualTableRow consumes context. If context is stable, it shouldn't re-render (since it is React.memo'd and its props didn't change).
|
|
87
|
+
// Wait, VirtualTableRow receives props from MemoizedList -> Row.
|
|
88
|
+
// Row renders <VirtualTableRow ...>.
|
|
89
|
+
// If VirtualTable re-renders -> MemoizedList re-renders?
|
|
90
|
+
// VirtualTable renders:
|
|
91
|
+
// <VirtualListContext.Provider value={virtualListController}>
|
|
92
|
+
// <MemoizedList ... />
|
|
93
|
+
// </VirtualListContext.Provider>
|
|
94
|
+
|
|
95
|
+
// MemoizedList is defined outside or inside?
|
|
96
|
+
// It is defined OUTSIDE VirtualTable as:
|
|
97
|
+
// function MemoizedList(...) { ... }
|
|
98
|
+
// BUT, inside VirtualTable it is used as <MemoizedList ... />.
|
|
99
|
+
// If VirtualTable re-renders, React.createElement(MemoizedList, ...) is called.
|
|
100
|
+
// If props to MemoizedList are same, does it re-render?
|
|
101
|
+
// MemoizedList is NOT wrapped in React.memo (in the file it's just `function MemoizedList`).
|
|
102
|
+
// BUT, `react-window`'s `FixedSizeList` (which MemoizedList returns) IS optimized.
|
|
103
|
+
// However, `MemoizedList` itself is a functional component. If parent renders, it renders.
|
|
104
|
+
// AND `MemoizedList` defines `const Row = useCallback(...)`.
|
|
105
|
+
// If `MemoizedList` re-renders, `Row` might be recreated if deps change?
|
|
106
|
+
// `Row` has `[]` deps. So `Row` is stable!
|
|
107
|
+
|
|
108
|
+
// Wait, `MemoizedList` passes `Row` to `List`.
|
|
109
|
+
// `List` (react-window) renders `Row`.
|
|
110
|
+
// `Row` contains `<VirtualListContext.Consumer>`.
|
|
111
|
+
// So `Row` renders. It consumes context.
|
|
112
|
+
// If context value changed, `Row` re-renders (or the consumer part does).
|
|
113
|
+
// Inside Consumer: `return <VirtualTableRow ... />`.
|
|
114
|
+
// If context value changed, the function inside Consumer runs.
|
|
115
|
+
// It creates new props for VirtualTableRow?
|
|
116
|
+
// `columns` comes from context. `data` comes from context.
|
|
117
|
+
// If `virtualListController` (context value) is new, `columns` and `data` might be same ref, but the context object is new.
|
|
118
|
+
// Consumer runs.
|
|
119
|
+
// `<VirtualTableRow ... />` is created.
|
|
120
|
+
// `VirtualTableRow` is `React.memo`.
|
|
121
|
+
// It compares new props vs old props.
|
|
122
|
+
// `columns` (from context) -> same ref? Yes (from useState in VirtualTable).
|
|
123
|
+
// `data` (from context) -> same ref? Yes (from props).
|
|
124
|
+
// `onRowClick` -> same ref? Yes (if useCallback/useMemo used properly in VirtualTable).
|
|
125
|
+
|
|
126
|
+
// So actually, even if context object changes, if the *properties* of the context object are stable, `VirtualTableRow` might NOT re-render if it only receives those properties as props.
|
|
127
|
+
// Let's check `VirtualTable.tsx` again.
|
|
128
|
+
|
|
129
|
+
// In `MemoizedList` -> `Row`:
|
|
130
|
+
/*
|
|
131
|
+
<VirtualListContext.Consumer>
|
|
132
|
+
{({ data, columns, ... }) => {
|
|
133
|
+
// ...
|
|
134
|
+
return <VirtualTableRow data={data} columns={columns} ... />
|
|
135
|
+
}}
|
|
136
|
+
</VirtualListContext.Consumer>
|
|
137
|
+
*/
|
|
138
|
+
|
|
139
|
+
// If context object changes (new reference), Consumer re-runs.
|
|
140
|
+
// It calls the render prop.
|
|
141
|
+
// The render prop returns `<VirtualTableRow ... />`.
|
|
142
|
+
// React sees a `VirtualTableRow` element.
|
|
143
|
+
// It checks `React.memo` (equal).
|
|
144
|
+
// `equal(prevProps, nextProps)`.
|
|
145
|
+
// `prevProps.columns` vs `nextProps.columns`.
|
|
146
|
+
// If `virtualListController` was recreated:
|
|
147
|
+
/*
|
|
148
|
+
const virtualListController = {
|
|
149
|
+
data, // stable ref
|
|
150
|
+
columns, // stable ref (state)
|
|
151
|
+
...
|
|
152
|
+
onColumnResize: ... // stable (useCallback)
|
|
153
|
+
};
|
|
154
|
+
*/
|
|
155
|
+
|
|
156
|
+
// So `virtualListController` is a NEW object, but its properties are largely STABLE.
|
|
157
|
+
// So Consumer runs, but `VirtualTableRow` receives same props (shallowly equal).
|
|
158
|
+
// `react-fast-compare` (used in VirtualTableRow) should return true.
|
|
159
|
+
// So `VirtualTableRow` should NOT re-render even if context changes?
|
|
160
|
+
|
|
161
|
+
// WAIT. `VirtualTableRow` does NOT consume the context itself. `Row` (the render prop of List) consumes it.
|
|
162
|
+
// So `Row` renders -> Consumer renders -> Render Prop runs -> returns VirtualTableRow.
|
|
163
|
+
// VirtualTableRow is memoized.
|
|
164
|
+
// So if props are equal, it shouldn't render.
|
|
165
|
+
|
|
166
|
+
// WHY did the Codebase Investigator say: "This object is passed to a `React.Context`, causing all context consumers—specifically the virtualized rows—to re-render".
|
|
167
|
+
// Maybe I missed something.
|
|
168
|
+
// Does `VirtualTableRow` use the context?
|
|
169
|
+
// No, `VirtualTableRow.tsx` does not use `useContext`.
|
|
170
|
+
|
|
171
|
+
// However, `VirtualTableCell`?
|
|
172
|
+
// `VirtualTableCell` is rendered by `VirtualTableRow`.
|
|
173
|
+
|
|
174
|
+
// Let's look at `VirtualTable.tsx` again.
|
|
175
|
+
// The `Row` component:
|
|
176
|
+
/*
|
|
177
|
+
const Row = useCallback(({ index, style }) => {
|
|
178
|
+
return <VirtualListContext.Consumer>
|
|
179
|
+
{(context) => {
|
|
180
|
+
// ...
|
|
181
|
+
return <VirtualTableRow ...> ... </VirtualTableRow>
|
|
182
|
+
}}
|
|
183
|
+
</VirtualListContext.Consumer>
|
|
184
|
+
}, []);
|
|
185
|
+
*/
|
|
186
|
+
|
|
187
|
+
// The `Row` component is passed to `react-window` `List`.
|
|
188
|
+
// `List` renders `Row`.
|
|
189
|
+
// `Row` renders `Consumer`.
|
|
190
|
+
// If Context Provider value changes, ALL Consumers re-render.
|
|
191
|
+
// So the function inside `Consumer` runs.
|
|
192
|
+
// It returns `VirtualTableRow`.
|
|
193
|
+
// React attempts to update `VirtualTableRow`.
|
|
194
|
+
// `VirtualTableRow` is `React.memo`.
|
|
195
|
+
// It compares props.
|
|
196
|
+
|
|
197
|
+
// Are props different?
|
|
198
|
+
// `columns` -> state. Stable.
|
|
199
|
+
// `data` -> prop. Stable.
|
|
200
|
+
// `onRowClick` -> `virtualListController.onRowClick`.
|
|
201
|
+
// In `VirtualTable.tsx`:
|
|
202
|
+
// `const virtualListController = { ... onRowClick, ... }`
|
|
203
|
+
// `onRowClick` comes from props. Stable.
|
|
204
|
+
|
|
205
|
+
// `cellRenderer` -> prop. Stable.
|
|
206
|
+
|
|
207
|
+
// `rowClassName` -> prop. Stable.
|
|
208
|
+
|
|
209
|
+
// So... if all props are stable, why is it a bottleneck?
|
|
210
|
+
// Maybe `virtualListController` creation is expensive? No.
|
|
211
|
+
// Maybe `react-window` does something?
|
|
212
|
+
|
|
213
|
+
// Or maybe one of the properties IS unstable?
|
|
214
|
+
// `onColumnResizeInternal`: useCallback. Stable.
|
|
215
|
+
// `onColumnResizeEndInternal`: useCallback. Stable.
|
|
216
|
+
// `onFilterUpdateInternal`: useCallback. Stable.
|
|
217
|
+
// `onColumnSort`: useCallback. Stable.
|
|
218
|
+
|
|
219
|
+
// `filterRef.current` -> ref. Stable.
|
|
220
|
+
|
|
221
|
+
// Wait, `virtualListController` object ITSELF is new.
|
|
222
|
+
// `VirtualListContext.Provider value={virtualListController}` receives new object.
|
|
223
|
+
// Provider updates.
|
|
224
|
+
// All Consumers update.
|
|
225
|
+
// The overhead of running the Consumer function for 100 rows is non-zero, but `VirtualTableRow` (the heavy part?) shouldn't re-render if props are same.
|
|
226
|
+
|
|
227
|
+
// UNLESS... `VirtualTableCell`?
|
|
228
|
+
// `VirtualTableRow` has children:
|
|
229
|
+
/*
|
|
230
|
+
{columns.map((column, columnIndex) => {
|
|
231
|
+
return <VirtualTableCell ... />
|
|
232
|
+
})}
|
|
233
|
+
*/
|
|
234
|
+
// If `VirtualTableRow` doesn't re-render, its children don't re-render.
|
|
235
|
+
|
|
236
|
+
// So where is the bottleneck?
|
|
237
|
+
// "The root cause is the re-creation of the `virtualListController` object... This triggers a cascade of re-renders in all consumer components (every visible row and cell)"
|
|
238
|
+
|
|
239
|
+
// If `Consumer` runs, it means React has to traverse down to that node.
|
|
240
|
+
// If `VirtualTableRow` props are equal, it stops there.
|
|
241
|
+
|
|
242
|
+
// Maybe `VirtualTableRow` props are NOT equal?
|
|
243
|
+
// Let's verify `VirtualTableRow` props.
|
|
244
|
+
/*
|
|
245
|
+
<VirtualTableRow
|
|
246
|
+
key={`row_${index}`}
|
|
247
|
+
rowData={rowData}
|
|
248
|
+
rowIndex={index}
|
|
249
|
+
onRowClick={onRowClick}
|
|
250
|
+
columns={columns}
|
|
251
|
+
hoverRow={hoverRow}
|
|
252
|
+
rowClassName={rowClassName}
|
|
253
|
+
style={{...}}
|
|
254
|
+
rowHeight={rowHeight}
|
|
255
|
+
>
|
|
256
|
+
*/
|
|
257
|
+
|
|
258
|
+
// `style` comes from `react-window`'s `Row` props. `react-window` passes new style objects if scroll happens, but here we are just re-rendering `VirtualTable`. `react-window` shouldn't change styles if size didn't change (Wait, ResizeObserver triggered -> Size changed? No, we just triggered it, assuming size is same but callback fired).
|
|
259
|
+
// Actually `style` in `Row` callback comes from `react-window`.
|
|
260
|
+
|
|
261
|
+
// `onRowClick`: `virtualListController.onRowClick` (from props).
|
|
262
|
+
|
|
263
|
+
// BUT, what about `VirtualTableCell`?
|
|
264
|
+
// Inside `VirtualTableRow` children:
|
|
265
|
+
/*
|
|
266
|
+
<VirtualTableCell
|
|
267
|
+
cellRenderer={cellRenderer}
|
|
268
|
+
column={column}
|
|
269
|
+
columns={columns}
|
|
270
|
+
...
|
|
271
|
+
/>
|
|
272
|
+
*/
|
|
273
|
+
// `cellRenderer` comes from context.
|
|
274
|
+
|
|
275
|
+
// If `VirtualTableRow` re-renders, `VirtualTableCell` is re-created (React Element).
|
|
276
|
+
// `VirtualTableCell` is `React.memo`.
|
|
277
|
+
// Comparison runs.
|
|
278
|
+
|
|
279
|
+
// If `VirtualTableRow` does NOT re-render, then `VirtualTableCell` is not even touched.
|
|
280
|
+
|
|
281
|
+
// So the key is whether `VirtualTableRow` re-renders.
|
|
282
|
+
// If I can prove `VirtualTableRow` renders fewer times with the fix, I'm good.
|
|
283
|
+
|
|
284
|
+
// Why would `VirtualTableRow` props be different?
|
|
285
|
+
// Maybe `equal` (react-fast-compare) returns false for some reason?
|
|
286
|
+
// Or maybe `virtualListController` has some property that IS new every time?
|
|
287
|
+
|
|
288
|
+
// `const virtualListController = { ... }`.
|
|
289
|
+
// `filter: filterRef.current`.
|
|
290
|
+
// `currentSort`.
|
|
291
|
+
// `sortByProperty`.
|
|
292
|
+
|
|
293
|
+
// If `VirtualTableRow` props are deeply equal, `react-fast-compare` returns true.
|
|
294
|
+
|
|
295
|
+
// Wait. `MemoizedList` component.
|
|
296
|
+
/*
|
|
297
|
+
function MemoizedList(...) {
|
|
298
|
+
const Row = useCallback(...)
|
|
299
|
+
return <List ...>{Row}</List>
|
|
300
|
+
}
|
|
301
|
+
*/
|
|
302
|
+
// `MemoizedList` is NOT memoized (it's a function).
|
|
303
|
+
// In `VirtualTable`:
|
|
304
|
+
/*
|
|
305
|
+
<MemoizedList ... />
|
|
306
|
+
*/
|
|
307
|
+
// Parent renders -> `MemoizedList` renders.
|
|
308
|
+
// `Row` is `useCallback(..., [])`. Dependency array is empty!
|
|
309
|
+
// So `Row` is STABLE.
|
|
310
|
+
|
|
311
|
+
// `List` (react-window) receives stable `Row`.
|
|
312
|
+
// `List` is `PureComponent` (usually).
|
|
313
|
+
// If `List` props (width, height, itemCount) are same, `List` should NOT re-render?
|
|
314
|
+
// `width` and `height` come from `bounds`. If `bounds` didn't change, they are same.
|
|
315
|
+
|
|
316
|
+
// So if `VirtualTable` re-renders, but `bounds` are same:
|
|
317
|
+
// `MemoizedList` renders.
|
|
318
|
+
// `List` receives same props.
|
|
319
|
+
// `List` does NOT re-render.
|
|
320
|
+
// So `Row`s are NOT re-rendered by `react-window`.
|
|
321
|
+
|
|
322
|
+
// BUT, `VirtualListContext.Provider` is ABOVE `MemoizedList`.
|
|
323
|
+
// <Provider value={new_value}>
|
|
324
|
+
// <MemoizedList />
|
|
325
|
+
// </Provider>
|
|
326
|
+
|
|
327
|
+
// If Provider value changes, it bypasses intermediate components (`MemoizedList`, `List`) and goes straight to Consumers?
|
|
328
|
+
// YES. Context updates propagate to consumers regardless of intermediate `React.memo` or `PureComponent`.
|
|
329
|
+
|
|
330
|
+
// So `Row` (which contains `Consumer`) will update?
|
|
331
|
+
// `Row` is a component definition passed to `List`.
|
|
332
|
+
// `List` renders instances of `Row`.
|
|
333
|
+
// These instances contain `Consumer`.
|
|
334
|
+
// Use of `<Context.Consumer>` or `useContext` creates a subscription.
|
|
335
|
+
// So the `Consumer` component inside the rendered `Row` updates.
|
|
336
|
+
|
|
337
|
+
// The children function of `Consumer` executes.
|
|
338
|
+
// It returns `<VirtualTableRow ... />`.
|
|
339
|
+
// React compares this new element with previous.
|
|
340
|
+
// If `VirtualTableRow` is `React.memo`, it checks props.
|
|
341
|
+
|
|
342
|
+
// The premise of the optimization is that checking `React.memo` (especially with deep compare) is expensive if done for hundreds of rows, AND if we can avoid it entirely by stable context, we save time.
|
|
343
|
+
// If Context is stable, `Consumer` doesn't update.
|
|
344
|
+
// So the function inside Consumer doesn't run.
|
|
345
|
+
// So `VirtualTableRow` comparison doesn't happen.
|
|
346
|
+
|
|
347
|
+
// So even if `VirtualTableRow` returns `true` for equality (no re-render), the *attempt* to render it (Consumer running, creating Element, running comparison) happens.
|
|
348
|
+
|
|
349
|
+
// My test counts `VirtualTableRow` *renders* (function execution).
|
|
350
|
+
// If `React.memo` works, the function body of `VirtualTableRow` should NOT run.
|
|
351
|
+
// The mock I wrote:
|
|
352
|
+
/*
|
|
353
|
+
const VirtualTableRow = React.memo((props) => {
|
|
354
|
+
renderCallback();
|
|
355
|
+
...
|
|
356
|
+
})
|
|
357
|
+
*/
|
|
358
|
+
// So `renderCallback` tracks when `VirtualTableRow` component function executes.
|
|
359
|
+
// If `React.memo` returns true (props equal), the function does NOT execute.
|
|
360
|
+
|
|
361
|
+
// So... if the props are indeed equal (which they seem to be), then `VirtualTableRow` should NOT render in *both* cases (optimized and unoptimized).
|
|
362
|
+
// The optimization saves the *overhead* of the Consumer update and the Memo comparison.
|
|
363
|
+
// But my test measures *Component Renders*.
|
|
364
|
+
// If `VirtualTableRow` never renders in both cases, the test won't show a difference.
|
|
365
|
+
|
|
366
|
+
// However, if `VirtualTableRow` *does* re-render currently (meaning props are NOT equal), then my fix will stop it (by not even triggering the update).
|
|
367
|
+
|
|
368
|
+
// Why would props be unequal?
|
|
369
|
+
// `columns` is `columnsProp` state.
|
|
370
|
+
// `data` is `data` prop.
|
|
371
|
+
// `style` ...
|
|
372
|
+
|
|
373
|
+
// If the test shows 0 re-renders in the "after" check, it's good.
|
|
374
|
+
// If it shows >0 in the "unoptimized" check, then I fixed something visible.
|
|
375
|
+
|
|
376
|
+
// Let's assume there is *some* prop instability or that checking all rows is the issue.
|
|
377
|
+
// Or maybe `react-fast-compare` is not as fast or returns false?
|
|
378
|
+
|
|
379
|
+
// Actually, I can also instrument the `Consumer`? No, that's inside the component.
|
|
380
|
+
|
|
381
|
+
// Let's just run the test and see. If "before" count is high and "after" is low, I win.
|
|
382
|
+
// If both are 0, I need a better test (maybe measure time?).
|
|
383
|
+
|
|
384
|
+
// I will write the test file now.
|
|
385
|
+
});
|
|
386
|
+
});
|