@pilotiq/pilotiq 0.24.1 → 0.24.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +57 -0
- package/boost/guidelines.md +571 -0
- package/boost/skills/pilotiq-actions/SKILL.md +49 -0
- package/boost/skills/pilotiq-actions/rules/dispatch-modes.md +177 -0
- package/boost/skills/pilotiq-actions/rules/factories.md +130 -0
- package/boost/skills/pilotiq-actions/rules/visibility-and-authorization.md +125 -0
- package/boost/skills/pilotiq-fields/SKILL.md +47 -0
- package/boost/skills/pilotiq-fields/rules/field-catalog.md +288 -0
- package/boost/skills/pilotiq-fields/rules/reactive-fields.md +199 -0
- package/boost/skills/pilotiq-fields/rules/validation.md +198 -0
- package/boost/skills/pilotiq-relations/SKILL.md +47 -0
- package/boost/skills/pilotiq-relations/rules/relation-managers.md +256 -0
- package/boost/skills/pilotiq-relations/rules/repeater-relationship.md +177 -0
- package/boost/skills/pilotiq-resource/SKILL.md +61 -0
- package/boost/skills/pilotiq-resource/rules/authorization.md +242 -0
- package/boost/skills/pilotiq-resource/rules/defining-resources.md +228 -0
- package/boost/skills/pilotiq-resource/rules/page-overrides.md +296 -0
- package/dist/Pilotiq.d.ts +31 -0
- package/dist/Pilotiq.d.ts.map +1 -1
- package/dist/Pilotiq.js +3 -1
- package/dist/Pilotiq.js.map +1 -1
- package/dist/PilotiqRegistry.d.ts +13 -0
- package/dist/PilotiqRegistry.d.ts.map +1 -1
- package/dist/PilotiqRegistry.js +15 -0
- package/dist/PilotiqRegistry.js.map +1 -1
- package/dist/pageData/misc.d.ts.map +1 -1
- package/dist/pageData/misc.js +6 -0
- package/dist/pageData/misc.js.map +1 -1
- package/dist/pageData/navigation.d.ts +1 -0
- package/dist/pageData/navigation.d.ts.map +1 -1
- package/dist/pageData/navigation.js +3 -0
- package/dist/pageData/navigation.js.map +1 -1
- package/dist/pageData/relationPages.d.ts.map +1 -1
- package/dist/pageData/relationPages.js +3 -0
- package/dist/pageData/relationPages.js.map +1 -1
- package/dist/pageData/resourcePages.d.ts.map +1 -1
- package/dist/pageData/resourcePages.js +8 -0
- package/dist/pageData/resourcePages.js.map +1 -1
- package/dist/react/AppShell.d.ts +8 -0
- package/dist/react/AppShell.d.ts.map +1 -1
- package/dist/react/AppShell.js.map +1 -1
- package/dist/react/layouts/SidebarLayout.d.ts.map +1 -1
- package/dist/react/layouts/SidebarLayout.js +10 -2
- package/dist/react/layouts/SidebarLayout.js.map +1 -1
- package/dist/react/widgets/StatsOverviewRenderer.d.ts.map +1 -1
- package/dist/react/widgets/StatsOverviewRenderer.js +32 -18
- package/dist/react/widgets/StatsOverviewRenderer.js.map +1 -1
- package/dist/routes/relations.d.ts.map +1 -1
- package/dist/routes/relations.js +25 -18
- package/dist/routes/relations.js.map +1 -1
- package/dist/routes/resources.js.map +1 -1
- package/package.json +10 -5
- package/.turbo/turbo-build.log +0 -8
- package/CLAUDE.md +0 -265
- package/src/Cluster.test.ts +0 -283
- package/src/Cluster.ts +0 -83
- package/src/Column.test.ts +0 -199
- package/src/Column.ts +0 -710
- package/src/Global.test.ts +0 -367
- package/src/Global.ts +0 -169
- package/src/Page.test.ts +0 -114
- package/src/Page.ts +0 -208
- package/src/Pilotiq.perf.test.ts +0 -252
- package/src/Pilotiq.test.ts +0 -129
- package/src/Pilotiq.ts +0 -1158
- package/src/PilotiqRegistry.ts +0 -36
- package/src/PilotiqServiceProvider.ts +0 -121
- package/src/RelationManager.test.ts +0 -400
- package/src/RelationManager.ts +0 -527
- package/src/RenderHook.test.ts +0 -252
- package/src/RenderHook.ts +0 -242
- package/src/Resource.test.ts +0 -284
- package/src/Resource.ts +0 -526
- package/src/RightPanel.test.ts +0 -202
- package/src/RightPanel.ts +0 -132
- package/src/Tab.test.ts +0 -91
- package/src/Tab.ts +0 -156
- package/src/UserMenuItem.ts +0 -145
- package/src/actions/Action.test.ts +0 -2526
- package/src/actions/Action.ts +0 -1515
- package/src/actions/ActionGroup.test.ts +0 -112
- package/src/actions/ActionGroup.ts +0 -173
- package/src/actions/attachFactory.ts +0 -172
- package/src/actions/bulkFactories.ts +0 -168
- package/src/actions/crudFactories.ts +0 -220
- package/src/actions/exportFactory.ts +0 -225
- package/src/actions/factoryHelpers.ts +0 -177
- package/src/actions/importFactory.ts +0 -243
- package/src/actions/index.ts +0 -17
- package/src/actions/m2mFactories.ts +0 -193
- package/src/actions/relationFactories.ts +0 -372
- package/src/applyPageHooks.test.ts +0 -463
- package/src/applyPageHooks.ts +0 -330
- package/src/authorization.test.ts +0 -483
- package/src/breadcrumbs.test.ts +0 -238
- package/src/cells/coerce.test.ts +0 -85
- package/src/cells/coerce.ts +0 -84
- package/src/clusterPaths.ts +0 -35
- package/src/columns/BadgeColumn.test.ts +0 -54
- package/src/columns/BadgeColumn.ts +0 -32
- package/src/columns/BooleanColumn.test.ts +0 -41
- package/src/columns/BooleanColumn.ts +0 -18
- package/src/columns/ColorColumn.test.ts +0 -37
- package/src/columns/ColorColumn.ts +0 -38
- package/src/columns/IconColumn.test.ts +0 -54
- package/src/columns/IconColumn.ts +0 -37
- package/src/columns/ImageColumn.test.ts +0 -41
- package/src/columns/ImageColumn.ts +0 -28
- package/src/columns/SelectColumn.ts +0 -98
- package/src/columns/TextColumn.test.ts +0 -190
- package/src/columns/TextColumn.ts +0 -20
- package/src/columns/TextInputColumn.ts +0 -68
- package/src/columns/ToggleColumn.ts +0 -46
- package/src/columns/editableColumns.test.ts +0 -238
- package/src/columns/index.ts +0 -9
- package/src/defaultGlobalPages.ts +0 -95
- package/src/defaultPages.test.ts +0 -634
- package/src/defaultPages.ts +0 -617
- package/src/defaultViewPage.test.ts +0 -147
- package/src/elements/Form.test.ts +0 -223
- package/src/elements/Form.ts +0 -416
- package/src/elements/ListTabs.ts +0 -28
- package/src/elements/Table.test.ts +0 -422
- package/src/elements/Table.ts +0 -850
- package/src/elements/TableGroup.test.ts +0 -260
- package/src/elements/TableGroup.ts +0 -334
- package/src/elements/dispatchAction.test.ts +0 -463
- package/src/elements/dispatchAction.ts +0 -355
- package/src/elements/dispatchForm.test.ts +0 -477
- package/src/elements/dispatchForm.ts +0 -1993
- package/src/elements/dispatchTable.test.ts +0 -1514
- package/src/elements/dispatchTable.ts +0 -745
- package/src/elements/index.ts +0 -21
- package/src/entries/BadgeEntry.ts +0 -39
- package/src/entries/CodeEntry.test.ts +0 -40
- package/src/entries/CodeEntry.ts +0 -52
- package/src/entries/ColorEntry.ts +0 -63
- package/src/entries/ComponentEntry.test.ts +0 -173
- package/src/entries/ComponentEntry.ts +0 -95
- package/src/entries/Entry.ts +0 -304
- package/src/entries/IconEntry.ts +0 -49
- package/src/entries/ImageEntry.ts +0 -61
- package/src/entries/KeyValueEntry.ts +0 -47
- package/src/entries/RepeatableEntry.test.ts +0 -239
- package/src/entries/RepeatableEntry.ts +0 -173
- package/src/entries/TextEntry.test.ts +0 -394
- package/src/entries/TextEntry.ts +0 -60
- package/src/entries/index.ts +0 -12
- package/src/entries/leaves.test.ts +0 -306
- package/src/entries/registry.ts +0 -54
- package/src/fields/BuilderField.test.ts +0 -1188
- package/src/fields/BuilderField.ts +0 -605
- package/src/fields/BuilderRelationship.test.ts +0 -811
- package/src/fields/CheckboxField.test.ts +0 -44
- package/src/fields/CheckboxField.ts +0 -27
- package/src/fields/CheckboxListField.test.ts +0 -99
- package/src/fields/CheckboxListField.ts +0 -66
- package/src/fields/ColorPickerField.test.ts +0 -33
- package/src/fields/ColorPickerField.ts +0 -25
- package/src/fields/DateField.ts +0 -54
- package/src/fields/DateTimeField.test.ts +0 -55
- package/src/fields/EmailField.ts +0 -16
- package/src/fields/Field.test.ts +0 -654
- package/src/fields/Field.ts +0 -817
- package/src/fields/FileUploadField.test.ts +0 -143
- package/src/fields/FileUploadField.ts +0 -159
- package/src/fields/HiddenField.test.ts +0 -27
- package/src/fields/HiddenField.ts +0 -28
- package/src/fields/KeyValueField.test.ts +0 -105
- package/src/fields/KeyValueField.ts +0 -55
- package/src/fields/MarkdownField.test.ts +0 -167
- package/src/fields/MarkdownField.ts +0 -162
- package/src/fields/NumberField.ts +0 -33
- package/src/fields/RadioField.test.ts +0 -94
- package/src/fields/RadioField.ts +0 -67
- package/src/fields/RepeaterField.test.ts +0 -1806
- package/src/fields/RepeaterField.ts +0 -939
- package/src/fields/RepeaterRelationship.test.ts +0 -1923
- package/src/fields/RepeaterSimple.test.ts +0 -248
- package/src/fields/RowButton.test.ts +0 -219
- package/src/fields/RowButton.ts +0 -135
- package/src/fields/SelectField.test.ts +0 -192
- package/src/fields/SelectField.ts +0 -235
- package/src/fields/SliderField.test.ts +0 -50
- package/src/fields/SliderField.ts +0 -53
- package/src/fields/SlugField.ts +0 -24
- package/src/fields/TagsInputField.test.ts +0 -154
- package/src/fields/TagsInputField.ts +0 -133
- package/src/fields/TextField.test.ts +0 -213
- package/src/fields/TextField.ts +0 -177
- package/src/fields/TextareaField.test.ts +0 -58
- package/src/fields/TextareaField.ts +0 -59
- package/src/fields/ToggleButtonsField.test.ts +0 -106
- package/src/fields/ToggleButtonsField.ts +0 -59
- package/src/fields/ToggleField.ts +0 -16
- package/src/fields/disableOptionsWhenSelectedInSiblingRepeaterItems.test.ts +0 -319
- package/src/fields/optionsResolver.ts +0 -95
- package/src/fields/resolveField.ts +0 -28
- package/src/filters/BooleanFilter.ts +0 -35
- package/src/filters/DateRangeFilter.test.ts +0 -194
- package/src/filters/DateRangeFilter.ts +0 -148
- package/src/filters/Filter.test.ts +0 -268
- package/src/filters/Filter.ts +0 -184
- package/src/filters/FormFilter.test.ts +0 -238
- package/src/filters/FormFilter.ts +0 -215
- package/src/filters/MultiSelectFilter.test.ts +0 -119
- package/src/filters/MultiSelectFilter.ts +0 -78
- package/src/filters/QueryBuilderFilter.test.ts +0 -662
- package/src/filters/QueryBuilderFilter.ts +0 -398
- package/src/filters/SelectFilter.ts +0 -46
- package/src/filters/TernaryFilter.test.ts +0 -160
- package/src/filters/TernaryFilter.ts +0 -72
- package/src/filters/TrashedFilter.test.ts +0 -149
- package/src/filters/TrashedFilter.ts +0 -55
- package/src/filters/queryBuilder/BooleanConstraint.ts +0 -31
- package/src/filters/queryBuilder/Constraint.ts +0 -115
- package/src/filters/queryBuilder/DateConstraint.ts +0 -69
- package/src/filters/queryBuilder/NumberConstraint.ts +0 -66
- package/src/filters/queryBuilder/SelectConstraint.ts +0 -72
- package/src/filters/queryBuilder/TextConstraint.ts +0 -64
- package/src/filters/queryBuilder/index.ts +0 -12
- package/src/icons/index.ts +0 -2
- package/src/icons/lucide.ts +0 -204
- package/src/icons/registry.test.ts +0 -56
- package/src/icons/registry.ts +0 -41
- package/src/icons/types.ts +0 -47
- package/src/index.ts +0 -525
- package/src/io/csv.test.ts +0 -142
- package/src/io/csv.ts +0 -170
- package/src/nestedRelationManagerData.test.ts +0 -547
- package/src/notifications/Notification.test.ts +0 -210
- package/src/notifications/Notification.ts +0 -354
- package/src/notifications/broadcast.test.ts +0 -110
- package/src/notifications/broadcast.ts +0 -95
- package/src/notifications/database.test.ts +0 -383
- package/src/notifications/database.ts +0 -398
- package/src/notifications/databaseNotifications.test.ts +0 -187
- package/src/notifications/dispatchNotificationAction.test.ts +0 -341
- package/src/notifications/dispatchNotificationAction.ts +0 -142
- package/src/notifications/flash.test.ts +0 -89
- package/src/notifications/flash.ts +0 -71
- package/src/notifications/index.ts +0 -45
- package/src/notifications/registerBroadcastAuth.test.ts +0 -134
- package/src/notifications/registerBroadcastAuth.ts +0 -100
- package/src/notifications/resolveSavedNotification.test.ts +0 -82
- package/src/notifications/resolveSavedNotification.ts +0 -59
- package/src/notifications/types.ts +0 -93
- package/src/orm/m2mAccessor.ts +0 -66
- package/src/orm/modelDefaults.test.ts +0 -633
- package/src/orm/modelDefaults.ts +0 -666
- package/src/pageData/breadcrumbs.ts +0 -288
- package/src/pageData/forms.ts +0 -578
- package/src/pageData/helpers.ts +0 -857
- package/src/pageData/misc.ts +0 -347
- package/src/pageData/navigation.ts +0 -842
- package/src/pageData/relationPages.ts +0 -1248
- package/src/pageData/relationTabs.ts +0 -286
- package/src/pageData/resourcePages.ts +0 -609
- package/src/pageData.test.ts +0 -1545
- package/src/pageData.ts +0 -341
- package/src/plugins/index.ts +0 -8
- package/src/plugins/themeEditor.test.ts +0 -36
- package/src/plugins/themeEditor.ts +0 -45
- package/src/react/AppShell.tsx +0 -251
- package/src/react/CollabExtensionFactoryRegistry.ts +0 -55
- package/src/react/CollabRoomContext.ts +0 -98
- package/src/react/CollabTextRendererRegistry.ts +0 -102
- package/src/react/CommandPalette.tsx +0 -375
- package/src/react/CurrentUserContext.tsx +0 -50
- package/src/react/CustomPageWrapperGate.tsx +0 -69
- package/src/react/CustomPageWrapperRegistry.ts +0 -45
- package/src/react/FieldFocusReporterRegistry.ts +0 -37
- package/src/react/FieldLabelSlotRegistry.ts +0 -30
- package/src/react/FieldPresenceRegistry.ts +0 -46
- package/src/react/FormCollabBindingRegistry.ts +0 -242
- package/src/react/FormStateContext.tsx +0 -591
- package/src/react/HeadHooks.tsx +0 -126
- package/src/react/MarkdownEditorRegistry.test.ts +0 -38
- package/src/react/MarkdownEditorRegistry.ts +0 -107
- package/src/react/NotificationActionStrip.tsx +0 -263
- package/src/react/NotificationBell.tsx +0 -426
- package/src/react/PendingSuggestionApplierRegistry.test.ts +0 -97
- package/src/react/PendingSuggestionApplierRegistry.ts +0 -98
- package/src/react/PendingSuggestionOverlayRegistry.ts +0 -54
- package/src/react/PendingSuggestionsContext.tsx +0 -172
- package/src/react/RecordWrapperGate.tsx +0 -58
- package/src/react/RecordWrapperRegistry.ts +0 -39
- package/src/react/RenderHookSlot.tsx +0 -32
- package/src/react/RightSidebar.tsx +0 -257
- package/src/react/RightSidebarContext.tsx +0 -234
- package/src/react/RightSidebarTrigger.tsx +0 -53
- package/src/react/RowCoordsContext.tsx +0 -23
- package/src/react/SchemaRenderer.tsx +0 -549
- package/src/react/SearchTrigger.tsx +0 -46
- package/src/react/ThemeProvider.tsx +0 -93
- package/src/react/ThemeSettingsPage.tsx +0 -579
- package/src/react/ThemeToggle.tsx +0 -20
- package/src/react/Toaster.tsx +0 -158
- package/src/react/UserMenu.tsx +0 -196
- package/src/react/WidgetDataContext.tsx +0 -157
- package/src/react/cells/EditableCell.tsx +0 -389
- package/src/react/component-slots.test.ts +0 -103
- package/src/react/component-slots.ts +0 -116
- package/src/react/fieldJsHandler.test.ts +0 -166
- package/src/react/fieldJsHandler.ts +0 -79
- package/src/react/fields/BuilderInput.tsx +0 -1078
- package/src/react/fields/CheckboxInput.tsx +0 -39
- package/src/react/fields/CheckboxListInput.tsx +0 -102
- package/src/react/fields/ColorInput.tsx +0 -71
- package/src/react/fields/DateFieldInput.tsx +0 -70
- package/src/react/fields/DateTimeInput.tsx +0 -62
- package/src/react/fields/FieldShell.tsx +0 -348
- package/src/react/fields/FileUploadInput.tsx +0 -639
- package/src/react/fields/HiddenInput.tsx +0 -17
- package/src/react/fields/KeyValueInput.tsx +0 -230
- package/src/react/fields/MarkdownInput.tsx +0 -560
- package/src/react/fields/RadioInput.tsx +0 -81
- package/src/react/fields/RepeaterInput.test.ts +0 -116
- package/src/react/fields/RepeaterInput.tsx +0 -1420
- package/src/react/fields/SelectFieldInput.tsx +0 -280
- package/src/react/fields/SliderInput.tsx +0 -81
- package/src/react/fields/TagsInput.tsx +0 -283
- package/src/react/fields/TextLikeInput.tsx +0 -256
- package/src/react/fields/ToggleButtonsInput.tsx +0 -60
- package/src/react/fields/ToggleFieldInput.tsx +0 -56
- package/src/react/fields/relationshipRenameDispatch.test.ts +0 -106
- package/src/react/fields/relationshipRenameDispatch.ts +0 -97
- package/src/react/fields/repeaterReconcile.test.ts +0 -114
- package/src/react/fields/repeaterReconcile.ts +0 -104
- package/src/react/fields/rowChromeButton.tsx +0 -336
- package/src/react/fields/rowState.ts +0 -106
- package/src/react/fields/syncRowGates.test.ts +0 -202
- package/src/react/fields/syncRowGates.ts +0 -66
- package/src/react/fields/textInputControls.tsx +0 -238
- package/src/react/fields/useRowReorderDnd.ts +0 -78
- package/src/react/formStateHelpers.test.ts +0 -508
- package/src/react/formStateHelpers.ts +0 -381
- package/src/react/hooks/use-mobile.ts +0 -19
- package/src/react/icon-context.tsx +0 -60
- package/src/react/index.ts +0 -194
- package/src/react/layouts/SidebarLayout.tsx +0 -250
- package/src/react/layouts/TopbarLayout.tsx +0 -258
- package/src/react/navigate.tsx +0 -37
- package/src/react/onProviderSynced.test.ts +0 -90
- package/src/react/parseRecordEditUrl.test.ts +0 -122
- package/src/react/parseRecordEditUrl.ts +0 -94
- package/src/react/persistedState.ts +0 -40
- package/src/react/registry.ts +0 -48
- package/src/react/right-panel-registry.tsx +0 -47
- package/src/react/schemaRenderer/AlertRenderer.tsx +0 -112
- package/src/react/schemaRenderer/EntryRenderer.tsx +0 -501
- package/src/react/schemaRenderer/SectionRenderer.tsx +0 -120
- package/src/react/schemaRenderer/SimpleElements.tsx +0 -306
- package/src/react/schemaRenderer/TabsRenderer.tsx +0 -62
- package/src/react/schemaRenderer/WizardRenderer.tsx +0 -338
- package/src/react/schemaRenderer/action/ActionGroupTrigger.tsx +0 -177
- package/src/react/schemaRenderer/action/ActionModalDialog.tsx +0 -273
- package/src/react/schemaRenderer/action/ConfirmActionDialog.tsx +0 -61
- package/src/react/schemaRenderer/action/HandlerActionButton.tsx +0 -43
- package/src/react/schemaRenderer/action/MethodActionButton.tsx +0 -64
- package/src/react/schemaRenderer/action/buttons.tsx +0 -99
- package/src/react/schemaRenderer/action/helpers.ts +0 -140
- package/src/react/schemaRenderer/action/renderAction.tsx +0 -245
- package/src/react/schemaRenderer/columnFormat.ts +0 -65
- package/src/react/schemaRenderer/constants.ts +0 -50
- package/src/react/schemaRenderer/form/FormRenderer.tsx +0 -274
- package/src/react/schemaRenderer/form/renderField.tsx +0 -511
- package/src/react/schemaRenderer/helpers.tsx +0 -81
- package/src/react/schemaRenderer/table/CardsLayoutBody.tsx +0 -308
- package/src/react/schemaRenderer/table/TableRenderer.tsx +0 -123
- package/src/react/schemaRenderer/table/TableRendererBody.tsx +0 -974
- package/src/react/schemaRenderer/table/filters.tsx +0 -1233
- package/src/react/schemaRenderer/table/formatCell.tsx +0 -264
- package/src/react/schemaRenderer/table/links.tsx +0 -112
- package/src/react/schemaRenderer/table/renderRowActions.tsx +0 -52
- package/src/react/schemaRenderer/table/url.tsx +0 -143
- package/src/react/theme-preview/apply.ts +0 -99
- package/src/react/theme-preview/build-html.ts +0 -436
- package/src/react/ui/button.tsx +0 -51
- package/src/react/ui/calendar.tsx +0 -67
- package/src/react/ui/checkbox.tsx +0 -29
- package/src/react/ui/dialog.tsx +0 -108
- package/src/react/ui/dropdown-menu.tsx +0 -97
- package/src/react/ui/input.tsx +0 -20
- package/src/react/ui/label.tsx +0 -21
- package/src/react/ui/popover.tsx +0 -50
- package/src/react/ui/select.tsx +0 -169
- package/src/react/ui/separator.tsx +0 -25
- package/src/react/ui/sheet.tsx +0 -136
- package/src/react/ui/sidebar.tsx +0 -723
- package/src/react/ui/skeleton.tsx +0 -13
- package/src/react/ui/slider.tsx +0 -34
- package/src/react/ui/switch.tsx +0 -28
- package/src/react/ui/table.tsx +0 -105
- package/src/react/ui/tabs.tsx +0 -63
- package/src/react/ui/textarea.tsx +0 -18
- package/src/react/ui/tooltip.tsx +0 -64
- package/src/react/useResizableWidth.ts +0 -139
- package/src/react/utils.ts +0 -6
- package/src/react/widgetRegistry.test.ts +0 -43
- package/src/react/widgetRegistry.ts +0 -50
- package/src/react/widgets/StatsOverviewRenderer.tsx +0 -232
- package/src/react/widgets/TableWidgetRenderer.tsx +0 -231
- package/src/react/widgets/ViewRenderer.tsx +0 -71
- package/src/relationManagerData.test.ts +0 -1595
- package/src/richtext/index.ts +0 -8
- package/src/richtext/registry.ts +0 -89
- package/src/routes/globals.ts +0 -148
- package/src/routes/guard.test.ts +0 -325
- package/src/routes/helpers.ts +0 -704
- package/src/routes/pages.ts +0 -175
- package/src/routes/panel.ts +0 -204
- package/src/routes/relations.ts +0 -1243
- package/src/routes/resources.ts +0 -781
- package/src/routes/theme.ts +0 -91
- package/src/routes-nested-relations.test.ts +0 -676
- package/src/routes-relations.test.ts +0 -972
- package/src/routes.test.ts +0 -2027
- package/src/routes.ts +0 -303
- package/src/schema/Alert.test.ts +0 -109
- package/src/schema/Alert.ts +0 -131
- package/src/schema/Block.ts +0 -169
- package/src/schema/Breadcrumbs.ts +0 -40
- package/src/schema/Card.ts +0 -35
- package/src/schema/Divider.ts +0 -20
- package/src/schema/Element.ts +0 -219
- package/src/schema/EmptyState.test.ts +0 -37
- package/src/schema/EmptyState.ts +0 -63
- package/src/schema/Fieldset.ts +0 -43
- package/src/schema/Grid.ts +0 -43
- package/src/schema/Group.ts +0 -30
- package/src/schema/Heading.ts +0 -39
- package/src/schema/Html.ts +0 -67
- package/src/schema/Icon.ts +0 -54
- package/src/schema/Image.ts +0 -57
- package/src/schema/LinkTag.ts +0 -41
- package/src/schema/Markdown.ts +0 -85
- package/src/schema/MetaTag.ts +0 -41
- package/src/schema/RelationTabs.ts +0 -71
- package/src/schema/ScriptTag.ts +0 -55
- package/src/schema/Section.ts +0 -160
- package/src/schema/ServerDataElement.test.ts +0 -140
- package/src/schema/ServerDataElement.ts +0 -156
- package/src/schema/SlotComponent.test.ts +0 -77
- package/src/schema/SlotComponent.ts +0 -71
- package/src/schema/Split.ts +0 -50
- package/src/schema/Stat.test.ts +0 -118
- package/src/schema/Stat.ts +0 -154
- package/src/schema/StatsOverview.test.ts +0 -141
- package/src/schema/StatsOverview.ts +0 -119
- package/src/schema/StyleTag.ts +0 -35
- package/src/schema/TableWidget.test.ts +0 -297
- package/src/schema/TableWidget.ts +0 -289
- package/src/schema/Tabs.ts +0 -79
- package/src/schema/Text.ts +0 -58
- package/src/schema/UnorderedList.ts +0 -49
- package/src/schema/View.test.ts +0 -111
- package/src/schema/View.ts +0 -127
- package/src/schema/Wizard.ts +0 -220
- package/src/schema/containers.test.ts +0 -564
- package/src/schema/headTags.test.ts +0 -134
- package/src/schema/index.ts +0 -40
- package/src/schema/primes.test.ts +0 -269
- package/src/schema/resolveSchema.test.ts +0 -379
- package/src/schema/resolveSchema.ts +0 -917
- package/src/schema/sanitize.ts +0 -58
- package/src/search.test.ts +0 -446
- package/src/search.ts +0 -178
- package/src/sessionFilters.test.ts +0 -375
- package/src/sessionFilters.ts +0 -143
- package/src/slot-components/index.ts +0 -10
- package/src/slot-components/registry.ts +0 -56
- package/src/styles/file-upload.css +0 -13
- package/src/summarizers/Summarizer.test.ts +0 -84
- package/src/summarizers/Summarizer.ts +0 -123
- package/src/summarizers/index.ts +0 -11
- package/src/theme/base-colors.ts +0 -68
- package/src/theme/chart-colors.ts +0 -50
- package/src/theme/colors.ts +0 -447
- package/src/theme/generate-css.test.ts +0 -139
- package/src/theme/generate-css.ts +0 -44
- package/src/theme/generate-scale.test.ts +0 -106
- package/src/theme/generate-scale.ts +0 -97
- package/src/theme/icon-map.ts +0 -42
- package/src/theme/index.ts +0 -34
- package/src/theme/migrate.test.ts +0 -178
- package/src/theme/migrate.ts +0 -81
- package/src/theme/presets.ts +0 -135
- package/src/theme/radius.ts +0 -18
- package/src/theme/resolve.test.ts +0 -238
- package/src/theme/resolve.ts +0 -96
- package/src/theme/spacing.ts +0 -18
- package/src/theme/storage.test.ts +0 -126
- package/src/theme/storage.ts +0 -106
- package/src/theme/theme-colors.ts +0 -88
- package/src/theme/types.ts +0 -125
- package/src/uploads/UploadAdapter.ts +0 -35
- package/src/uploads/index.ts +0 -2
- package/src/uploads/localUpload.test.ts +0 -70
- package/src/uploads/localUpload.ts +0 -84
- package/src/validation/Validator.ts +0 -49
- package/src/validation/index.ts +0 -28
- package/src/validation/rules.ts +0 -78
- package/src/validation/runValidators.ts +0 -435
- package/src/validation/uniqueValidator.test.ts +0 -196
- package/src/validation/uniqueValidator.ts +0 -133
- package/src/validation/validators.test.ts +0 -268
- package/src/vite.test.ts +0 -184
- package/src/vite.ts +0 -787
- package/src/widgets/index.ts +0 -10
- package/src/widgets/registry.ts +0 -45
- package/src/widgets.test.ts +0 -592
- package/tsconfig.build.json +0 -11
- package/tsconfig.json +0 -4
- package/tsconfig.test.json +0 -10
- package/views/react/Dashboard.tsx +0 -27
- package/views/react/Resources/Form.tsx +0 -102
- package/views/react/Resources/Index.tsx +0 -49
|
@@ -1,375 +0,0 @@
|
|
|
1
|
-
import { describe, it } from 'node:test'
|
|
2
|
-
import assert from 'node:assert/strict'
|
|
3
|
-
|
|
4
|
-
import {
|
|
5
|
-
listFiltersKey,
|
|
6
|
-
lastTabKey,
|
|
7
|
-
readPersistedListQuery,
|
|
8
|
-
writePersistedListQuery,
|
|
9
|
-
readPersistedLastTab,
|
|
10
|
-
writePersistedLastTab,
|
|
11
|
-
encodePersistedQuery,
|
|
12
|
-
} from './sessionFilters.js'
|
|
13
|
-
|
|
14
|
-
/**
|
|
15
|
-
* Minimal SessionInstance stand-in — the same shape the rudder
|
|
16
|
-
* `SessionInstance.get / put` exposes. Tracks `put` calls so the no-op
|
|
17
|
-
* deep-equal short-circuit can be asserted.
|
|
18
|
-
*/
|
|
19
|
-
function makeSession() {
|
|
20
|
-
const data: Record<string, unknown> = {}
|
|
21
|
-
const puts: Array<[string, unknown]> = []
|
|
22
|
-
return {
|
|
23
|
-
data,
|
|
24
|
-
puts,
|
|
25
|
-
get<T>(key: string, fallback?: T): T | undefined {
|
|
26
|
-
return (key in data ? data[key] : fallback) as T | undefined
|
|
27
|
-
},
|
|
28
|
-
put(key: string, value: unknown): void {
|
|
29
|
-
data[key] = value
|
|
30
|
-
puts.push([key, value])
|
|
31
|
-
},
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
describe('listFiltersKey', () => {
|
|
36
|
-
it('joins prefix + basePath + slug + slot:<tab> with single colons', () => {
|
|
37
|
-
assert.equal(listFiltersKey('/admin', 'posts'), 'pilotiq:filters:/admin:posts:slot:')
|
|
38
|
-
assert.equal(listFiltersKey('/admin', 'posts', ''), 'pilotiq:filters:/admin:posts:slot:')
|
|
39
|
-
assert.equal(listFiltersKey('/admin', 'posts', 'drafts'), 'pilotiq:filters:/admin:posts:slot:drafts')
|
|
40
|
-
})
|
|
41
|
-
|
|
42
|
-
it('different slugs produce distinct keys', () => {
|
|
43
|
-
assert.notEqual(listFiltersKey('/admin', 'posts'), listFiltersKey('/admin', 'users'))
|
|
44
|
-
})
|
|
45
|
-
|
|
46
|
-
it('different tabs on the same slug produce distinct keys', () => {
|
|
47
|
-
assert.notEqual(
|
|
48
|
-
listFiltersKey('/admin', 'posts', 'drafts'),
|
|
49
|
-
listFiltersKey('/admin', 'posts', 'published'),
|
|
50
|
-
)
|
|
51
|
-
})
|
|
52
|
-
})
|
|
53
|
-
|
|
54
|
-
describe('lastTabKey', () => {
|
|
55
|
-
it('joins prefix + basePath + slug + lastTab', () => {
|
|
56
|
-
assert.equal(lastTabKey('/admin', 'posts'), 'pilotiq:filters:/admin:posts:lastTab')
|
|
57
|
-
})
|
|
58
|
-
|
|
59
|
-
it('does not collide with any slot key', () => {
|
|
60
|
-
// The slot namespace is `:slot:<tab>`; lastTab key is `:lastTab`. The
|
|
61
|
-
// `:slot:` infix prevents a tab literally named `lastTab` from clobbering.
|
|
62
|
-
assert.notEqual(lastTabKey('/admin', 'posts'), listFiltersKey('/admin', 'posts', 'lastTab'))
|
|
63
|
-
})
|
|
64
|
-
})
|
|
65
|
-
|
|
66
|
-
describe('writePersistedListQuery', () => {
|
|
67
|
-
it('stores filter values and reserved-but-persistable keys', () => {
|
|
68
|
-
const session = makeSession()
|
|
69
|
-
const req = { session }
|
|
70
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
71
|
-
writePersistedListQuery(req, key, {
|
|
72
|
-
status: 'draft',
|
|
73
|
-
group: 'authorId',
|
|
74
|
-
search: 'foo',
|
|
75
|
-
sort: 'createdAt:desc',
|
|
76
|
-
perPage: '25',
|
|
77
|
-
})
|
|
78
|
-
assert.deepEqual(session.data[key], {
|
|
79
|
-
status: 'draft', group: 'authorId', search: 'foo', sort: 'createdAt:desc', perPage: '25',
|
|
80
|
-
})
|
|
81
|
-
})
|
|
82
|
-
|
|
83
|
-
it('skips page (resets to 1 on restore) and tab (separate state)', () => {
|
|
84
|
-
const session = makeSession()
|
|
85
|
-
const req = { session }
|
|
86
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
87
|
-
writePersistedListQuery(req, key, {
|
|
88
|
-
status: 'draft',
|
|
89
|
-
page: '3',
|
|
90
|
-
tab: 'published',
|
|
91
|
-
})
|
|
92
|
-
assert.deepEqual(session.data[key], { status: 'draft' })
|
|
93
|
-
})
|
|
94
|
-
|
|
95
|
-
it('skips Tier-3 prefixed page keys (Table.queryStringIdentifier)', () => {
|
|
96
|
-
const session = makeSession()
|
|
97
|
-
const req = { session }
|
|
98
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
99
|
-
writePersistedListQuery(req, key, {
|
|
100
|
-
orders_status: 'draft',
|
|
101
|
-
orders_page: '3',
|
|
102
|
-
page: '4', // bare also dropped
|
|
103
|
-
})
|
|
104
|
-
assert.deepEqual(session.data[key], { orders_status: 'draft' })
|
|
105
|
-
})
|
|
106
|
-
|
|
107
|
-
it('skips groupKey (drill-in is page-state, not filter-state)', () => {
|
|
108
|
-
const session = makeSession()
|
|
109
|
-
const req = { session }
|
|
110
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
111
|
-
writePersistedListQuery(req, key, {
|
|
112
|
-
status: 'draft',
|
|
113
|
-
groupKey: 'archive-2025',
|
|
114
|
-
})
|
|
115
|
-
assert.deepEqual(session.data[key], { status: 'draft' })
|
|
116
|
-
})
|
|
117
|
-
|
|
118
|
-
it('skips Tier-3 prefixed groupKey keys', () => {
|
|
119
|
-
const session = makeSession()
|
|
120
|
-
const req = { session }
|
|
121
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
122
|
-
writePersistedListQuery(req, key, {
|
|
123
|
-
orders_status: 'draft',
|
|
124
|
-
orders_groupKey: 'old',
|
|
125
|
-
groupKey: 'older', // bare also dropped
|
|
126
|
-
})
|
|
127
|
-
assert.deepEqual(session.data[key], { orders_status: 'draft' })
|
|
128
|
-
})
|
|
129
|
-
|
|
130
|
-
it('preserves empty-string values (explicit-clear marker)', () => {
|
|
131
|
-
const session = makeSession()
|
|
132
|
-
const req = { session }
|
|
133
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
134
|
-
writePersistedListQuery(req, key, { status: '' })
|
|
135
|
-
assert.deepEqual(session.data[key], { status: '' })
|
|
136
|
-
})
|
|
137
|
-
|
|
138
|
-
it('skips non-string values defensively', () => {
|
|
139
|
-
const session = makeSession()
|
|
140
|
-
const req = { session }
|
|
141
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
142
|
-
writePersistedListQuery(req, key, {
|
|
143
|
-
status: 'draft',
|
|
144
|
-
junk: 42 as unknown as string,
|
|
145
|
-
blob: { x: 1 } as unknown as string,
|
|
146
|
-
})
|
|
147
|
-
assert.deepEqual(session.data[key], { status: 'draft' })
|
|
148
|
-
})
|
|
149
|
-
|
|
150
|
-
it('no-ops when the slice deep-equals the stored value', () => {
|
|
151
|
-
const session = makeSession()
|
|
152
|
-
const req = { session }
|
|
153
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
154
|
-
writePersistedListQuery(req, key, { status: 'draft' })
|
|
155
|
-
writePersistedListQuery(req, key, { status: 'draft' })
|
|
156
|
-
assert.equal(session.puts.length, 1)
|
|
157
|
-
})
|
|
158
|
-
|
|
159
|
-
it('writes when the stored value is a different shape', () => {
|
|
160
|
-
const session = makeSession()
|
|
161
|
-
const req = { session }
|
|
162
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
163
|
-
writePersistedListQuery(req, key, { status: 'draft' })
|
|
164
|
-
writePersistedListQuery(req, key, { status: 'draft', sort: 'id:asc' })
|
|
165
|
-
assert.equal(session.puts.length, 2)
|
|
166
|
-
})
|
|
167
|
-
|
|
168
|
-
it('writes the empty slice (clears prior state)', () => {
|
|
169
|
-
const session = makeSession()
|
|
170
|
-
const req = { session }
|
|
171
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
172
|
-
writePersistedListQuery(req, key, { status: 'draft' })
|
|
173
|
-
writePersistedListQuery(req, key, {})
|
|
174
|
-
assert.deepEqual(session.data[key], {})
|
|
175
|
-
})
|
|
176
|
-
|
|
177
|
-
it('no-ops silently when no session is on req', () => {
|
|
178
|
-
assert.doesNotThrow(() =>
|
|
179
|
-
writePersistedListQuery({}, listFiltersKey('/admin', 'posts'), { status: 'draft' }),
|
|
180
|
-
)
|
|
181
|
-
})
|
|
182
|
-
|
|
183
|
-
it('no-ops silently on undefined req', () => {
|
|
184
|
-
assert.doesNotThrow(() =>
|
|
185
|
-
writePersistedListQuery(undefined, listFiltersKey('/admin', 'posts'), { status: 'draft' }),
|
|
186
|
-
)
|
|
187
|
-
})
|
|
188
|
-
|
|
189
|
-
it('per-tab keying — same slug, different tabs land in distinct slots', () => {
|
|
190
|
-
const session = makeSession()
|
|
191
|
-
const req = { session }
|
|
192
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'drafts'), { status: 'draft' })
|
|
193
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'published'), { sort: 'title:asc' })
|
|
194
|
-
assert.deepEqual(session.data[listFiltersKey('/admin', 'posts', 'drafts')], { status: 'draft' })
|
|
195
|
-
assert.deepEqual(session.data[listFiltersKey('/admin', 'posts', 'published')], { sort: 'title:asc' })
|
|
196
|
-
})
|
|
197
|
-
})
|
|
198
|
-
|
|
199
|
-
describe('readPersistedListQuery', () => {
|
|
200
|
-
it('returns the stored slice', () => {
|
|
201
|
-
const session = makeSession()
|
|
202
|
-
const req = { session }
|
|
203
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
204
|
-
writePersistedListQuery(req, key, { status: 'draft' })
|
|
205
|
-
assert.deepEqual(readPersistedListQuery(req, key), { status: 'draft' })
|
|
206
|
-
})
|
|
207
|
-
|
|
208
|
-
it('returns undefined when nothing was stored', () => {
|
|
209
|
-
const session = makeSession()
|
|
210
|
-
const req = { session }
|
|
211
|
-
assert.equal(readPersistedListQuery(req, listFiltersKey('/admin', 'posts')), undefined)
|
|
212
|
-
})
|
|
213
|
-
|
|
214
|
-
it('returns undefined when stored value is the wrong shape', () => {
|
|
215
|
-
const session = makeSession()
|
|
216
|
-
const req = { session }
|
|
217
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
218
|
-
session.data[key] = ['not', 'a', 'map']
|
|
219
|
-
assert.equal(readPersistedListQuery(req, key), undefined)
|
|
220
|
-
session.data[key] = 'string'
|
|
221
|
-
assert.equal(readPersistedListQuery(req, key), undefined)
|
|
222
|
-
})
|
|
223
|
-
|
|
224
|
-
it('drops non-string entries defensively', () => {
|
|
225
|
-
const session = makeSession()
|
|
226
|
-
const req = { session }
|
|
227
|
-
const key = listFiltersKey('/admin', 'posts')
|
|
228
|
-
session.data[key] = { status: 'draft', n: 42, m: { x: 1 } }
|
|
229
|
-
assert.deepEqual(readPersistedListQuery(req, key), { status: 'draft' })
|
|
230
|
-
})
|
|
231
|
-
|
|
232
|
-
it('returns undefined when no session is mounted', () => {
|
|
233
|
-
assert.equal(readPersistedListQuery({}, listFiltersKey('/admin', 'posts')), undefined)
|
|
234
|
-
})
|
|
235
|
-
|
|
236
|
-
it('per-tab keying — reading one tab\'s slot does not pull in another\'s', () => {
|
|
237
|
-
const session = makeSession()
|
|
238
|
-
const req = { session }
|
|
239
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'drafts'), { status: 'draft' })
|
|
240
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'published'), { sort: 'title:asc' })
|
|
241
|
-
assert.deepEqual(readPersistedListQuery(req, listFiltersKey('/admin', 'posts', 'drafts')), { status: 'draft' })
|
|
242
|
-
assert.deepEqual(readPersistedListQuery(req, listFiltersKey('/admin', 'posts', 'published')), { sort: 'title:asc' })
|
|
243
|
-
})
|
|
244
|
-
})
|
|
245
|
-
|
|
246
|
-
describe('lastTab pointer', () => {
|
|
247
|
-
it('writePersistedLastTab + readPersistedLastTab round-trip', () => {
|
|
248
|
-
const session = makeSession()
|
|
249
|
-
const req = { session }
|
|
250
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
251
|
-
assert.equal(readPersistedLastTab(req, '/admin', 'posts'), 'drafts')
|
|
252
|
-
})
|
|
253
|
-
|
|
254
|
-
it('returns undefined when no pointer was written', () => {
|
|
255
|
-
const session = makeSession()
|
|
256
|
-
const req = { session }
|
|
257
|
-
assert.equal(readPersistedLastTab(req, '/admin', 'posts'), undefined)
|
|
258
|
-
})
|
|
259
|
-
|
|
260
|
-
it('overwrites on subsequent writes', () => {
|
|
261
|
-
const session = makeSession()
|
|
262
|
-
const req = { session }
|
|
263
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
264
|
-
writePersistedLastTab(req, '/admin', 'posts', 'published')
|
|
265
|
-
assert.equal(readPersistedLastTab(req, '/admin', 'posts'), 'published')
|
|
266
|
-
})
|
|
267
|
-
|
|
268
|
-
it('no-ops when the pointer is unchanged', () => {
|
|
269
|
-
const session = makeSession()
|
|
270
|
-
const req = { session }
|
|
271
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
272
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
273
|
-
assert.equal(session.puts.length, 1)
|
|
274
|
-
})
|
|
275
|
-
|
|
276
|
-
it('persists empty string explicitly (user moved off a named tab back to default)', () => {
|
|
277
|
-
const session = makeSession()
|
|
278
|
-
const req = { session }
|
|
279
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
280
|
-
writePersistedLastTab(req, '/admin', 'posts', '')
|
|
281
|
-
assert.equal(readPersistedLastTab(req, '/admin', 'posts'), '')
|
|
282
|
-
})
|
|
283
|
-
|
|
284
|
-
it('returns undefined when stored value is non-string', () => {
|
|
285
|
-
const session = makeSession()
|
|
286
|
-
const req = { session }
|
|
287
|
-
session.data[lastTabKey('/admin', 'posts')] = 42 as unknown
|
|
288
|
-
assert.equal(readPersistedLastTab(req, '/admin', 'posts'), undefined)
|
|
289
|
-
})
|
|
290
|
-
|
|
291
|
-
it('no-ops silently when no session is mounted', () => {
|
|
292
|
-
assert.doesNotThrow(() => writePersistedLastTab({}, '/admin', 'posts', 'drafts'))
|
|
293
|
-
assert.equal(readPersistedLastTab({}, '/admin', 'posts'), undefined)
|
|
294
|
-
})
|
|
295
|
-
})
|
|
296
|
-
|
|
297
|
-
describe('encodePersistedQuery', () => {
|
|
298
|
-
it('builds a stable URLSearchParams string', () => {
|
|
299
|
-
const qs = encodePersistedQuery({ status: 'draft', sort: 'id:desc' })
|
|
300
|
-
const params = new URLSearchParams(qs)
|
|
301
|
-
assert.equal(params.get('status'), 'draft')
|
|
302
|
-
assert.equal(params.get('sort'), 'id:desc')
|
|
303
|
-
})
|
|
304
|
-
|
|
305
|
-
it('drops empty-value entries (explicit-clear markers do not redirect)', () => {
|
|
306
|
-
assert.equal(encodePersistedQuery({ status: '' }), '')
|
|
307
|
-
})
|
|
308
|
-
|
|
309
|
-
it('returns an empty string for an empty slice and no tab', () => {
|
|
310
|
-
assert.equal(encodePersistedQuery({}), '')
|
|
311
|
-
assert.equal(encodePersistedQuery({}, ''), '')
|
|
312
|
-
})
|
|
313
|
-
|
|
314
|
-
it('prepends ?tab=<name> when tab arg is non-empty', () => {
|
|
315
|
-
const qs = encodePersistedQuery({ status: 'draft' }, 'published')
|
|
316
|
-
const params = new URLSearchParams(qs)
|
|
317
|
-
assert.equal(params.get('tab'), 'published')
|
|
318
|
-
assert.equal(params.get('status'), 'draft')
|
|
319
|
-
})
|
|
320
|
-
|
|
321
|
-
it('emits ?tab=<name> alone when slice is empty', () => {
|
|
322
|
-
const qs = encodePersistedQuery({}, 'published')
|
|
323
|
-
const params = new URLSearchParams(qs)
|
|
324
|
-
assert.equal(params.get('tab'), 'published')
|
|
325
|
-
assert.equal(params.size, 1)
|
|
326
|
-
})
|
|
327
|
-
|
|
328
|
-
it('omits tab when empty (default-tab restore needs no tab param)', () => {
|
|
329
|
-
const qs = encodePersistedQuery({ status: 'draft' }, '')
|
|
330
|
-
const params = new URLSearchParams(qs)
|
|
331
|
-
assert.equal(params.get('tab'), null)
|
|
332
|
-
assert.equal(params.get('status'), 'draft')
|
|
333
|
-
})
|
|
334
|
-
})
|
|
335
|
-
|
|
336
|
-
describe('per-tab persistence — full round-trip', () => {
|
|
337
|
-
it('two tabs each retain their own filter slice across switches', () => {
|
|
338
|
-
const session = makeSession()
|
|
339
|
-
const req = { session }
|
|
340
|
-
// User visits ?tab=drafts&status=draft → slot:drafts + lastTab=drafts.
|
|
341
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'drafts'), { tab: 'drafts', status: 'draft' })
|
|
342
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
343
|
-
// Then ?tab=published&sort=title:asc → slot:published + lastTab=published.
|
|
344
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'published'), { tab: 'published', sort: 'title:asc' })
|
|
345
|
-
writePersistedLastTab(req, '/admin', 'posts', 'published')
|
|
346
|
-
// Reads remain isolated per tab.
|
|
347
|
-
assert.deepEqual(readPersistedListQuery(req, listFiltersKey('/admin', 'posts', 'drafts')), { status: 'draft' })
|
|
348
|
-
assert.deepEqual(readPersistedListQuery(req, listFiltersKey('/admin', 'posts', 'published')), { sort: 'title:asc' })
|
|
349
|
-
// lastTab pointer reflects the most recent visit.
|
|
350
|
-
assert.equal(readPersistedLastTab(req, '/admin', 'posts'), 'published')
|
|
351
|
-
})
|
|
352
|
-
|
|
353
|
-
it('bare-visit restore = readLastTab → readSlot(lastTab) → encode(slot, lastTab)', () => {
|
|
354
|
-
const session = makeSession()
|
|
355
|
-
const req = { session }
|
|
356
|
-
writePersistedListQuery(req, listFiltersKey('/admin', 'posts', 'drafts'), { tab: 'drafts', status: 'draft' })
|
|
357
|
-
writePersistedLastTab(req, '/admin', 'posts', 'drafts')
|
|
358
|
-
|
|
359
|
-
// Simulating the bare-visit restore in routes.ts.
|
|
360
|
-
const restoreTab = readPersistedLastTab(req, '/admin', 'posts') ?? ''
|
|
361
|
-
const slot = readPersistedListQuery(req, listFiltersKey('/admin', 'posts', restoreTab)) ?? {}
|
|
362
|
-
const qs = encodePersistedQuery(slot, restoreTab)
|
|
363
|
-
const params = new URLSearchParams(qs)
|
|
364
|
-
assert.equal(params.get('tab'), 'drafts')
|
|
365
|
-
assert.equal(params.get('status'), 'draft')
|
|
366
|
-
})
|
|
367
|
-
|
|
368
|
-
it('bare-visit restore on a never-visited resource returns no qs (no redirect)', () => {
|
|
369
|
-
const session = makeSession()
|
|
370
|
-
const req = { session }
|
|
371
|
-
const restoreTab = readPersistedLastTab(req, '/admin', 'posts') ?? ''
|
|
372
|
-
const slot = readPersistedListQuery(req, listFiltersKey('/admin', 'posts', restoreTab))
|
|
373
|
-
assert.equal(slot, undefined)
|
|
374
|
-
})
|
|
375
|
-
})
|
package/src/sessionFilters.ts
DELETED
|
@@ -1,143 +0,0 @@
|
|
|
1
|
-
// Duck-typed adapter over `@rudderjs/session` — pilotiq doesn't peer
|
|
2
|
-
// depend, so helpers no-op silently when no session is mounted on the
|
|
3
|
-
// request.
|
|
4
|
-
|
|
5
|
-
const PREFIX = 'pilotiq:filters:'
|
|
6
|
-
|
|
7
|
-
const EXCLUDED_KEYS = new Set(['page', 'tab', 'groupKey'])
|
|
8
|
-
|
|
9
|
-
// Heuristic also catches `<prefix>_page` from `Table.queryStringIdentifier`.
|
|
10
|
-
// A filter literally named `something_page` would be dropped too, but
|
|
11
|
-
// that's an unusual filter name.
|
|
12
|
-
function isPageKey(key: string): boolean {
|
|
13
|
-
if (key === 'page') return true
|
|
14
|
-
return key.endsWith('_page')
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
// Sibling heuristic for `<prefix>_groupKey`. Group drill-in is page-
|
|
18
|
-
// state (click-to-drill, × to clear), not filter-state — restoring it
|
|
19
|
-
// on a bare visit would land the user on the bucket they last drilled
|
|
20
|
-
// into instead of the banded list they probably expect.
|
|
21
|
-
function isGroupKeyKey(key: string): boolean {
|
|
22
|
-
if (key === 'groupKey') return true
|
|
23
|
-
return key.endsWith('_groupKey')
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
interface StorableSession {
|
|
27
|
-
get<T>(key: string, fallback?: T): T | undefined
|
|
28
|
-
put(key: string, value: unknown): void
|
|
29
|
-
}
|
|
30
|
-
|
|
31
|
-
function getSession(req: unknown): StorableSession | undefined {
|
|
32
|
-
if (!req || typeof req !== 'object') return undefined
|
|
33
|
-
const session = (req as { session?: unknown }).session
|
|
34
|
-
if (!session || typeof session !== 'object') return undefined
|
|
35
|
-
const s = session as Record<string, unknown>
|
|
36
|
-
if (typeof s['get'] !== 'function' || typeof s['put'] !== 'function') return undefined
|
|
37
|
-
return session as StorableSession
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
// Per-tab slot key. `tab === ''` is the "no tab" / "default tab" slot —
|
|
41
|
-
// resources without ListTabs always store under it. Tab name comes from
|
|
42
|
-
// `?tab=<name>` verbatim — keying by the URL value (not by the tab's
|
|
43
|
-
// "default" status) keeps writers + readers symmetric without the
|
|
44
|
-
// route handler having to know the schema's default-tab name.
|
|
45
|
-
export function listFiltersKey(basePath: string, slug: string, tab: string = ''): string {
|
|
46
|
-
return `${PREFIX}${basePath}:${slug}:slot:${tab}`
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
// Pointer to whichever tab the user was last on. Read on bare-URL visits
|
|
50
|
-
// to decide which slot to restore — without it, bare visits would always
|
|
51
|
-
// restore the no-tab slot (a regression vs v1, where one slot held
|
|
52
|
-
// whatever tab+filters combo the user last applied).
|
|
53
|
-
export function lastTabKey(basePath: string, slug: string): string {
|
|
54
|
-
return `${PREFIX}${basePath}:${slug}:lastTab`
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
export function readPersistedListQuery(
|
|
58
|
-
req: unknown,
|
|
59
|
-
key: string,
|
|
60
|
-
): Record<string, string> | undefined {
|
|
61
|
-
const session = getSession(req)
|
|
62
|
-
if (!session) return undefined
|
|
63
|
-
const v = session.get<unknown>(key)
|
|
64
|
-
if (!v || typeof v !== 'object' || Array.isArray(v)) return undefined
|
|
65
|
-
// Filter out non-string values defensively — a malformed cookie that
|
|
66
|
-
// round-tripped through some other writer shouldn't crash the page.
|
|
67
|
-
const out: Record<string, string> = {}
|
|
68
|
-
for (const [k, val] of Object.entries(v as Record<string, unknown>)) {
|
|
69
|
-
if (typeof val === 'string') out[k] = val
|
|
70
|
-
}
|
|
71
|
-
return out
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
export function writePersistedListQuery(
|
|
75
|
-
req: unknown,
|
|
76
|
-
key: string,
|
|
77
|
-
query: Record<string, unknown>,
|
|
78
|
-
): void {
|
|
79
|
-
const session = getSession(req)
|
|
80
|
-
if (!session) return
|
|
81
|
-
const slice: Record<string, string> = {}
|
|
82
|
-
for (const [k, v] of Object.entries(query)) {
|
|
83
|
-
if (EXCLUDED_KEYS.has(k)) continue
|
|
84
|
-
if (isPageKey(k)) continue
|
|
85
|
-
if (isGroupKeyKey(k)) continue
|
|
86
|
-
if (typeof v !== 'string') continue
|
|
87
|
-
slice[k] = v
|
|
88
|
-
}
|
|
89
|
-
const prev = session.get<Record<string, unknown> | undefined>(key)
|
|
90
|
-
if (prev && shallowEqualStringMap(prev, slice)) return
|
|
91
|
-
session.put(key, slice)
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
export function readPersistedLastTab(
|
|
95
|
-
req: unknown,
|
|
96
|
-
basePath: string,
|
|
97
|
-
slug: string,
|
|
98
|
-
): string | undefined {
|
|
99
|
-
const session = getSession(req)
|
|
100
|
-
if (!session) return undefined
|
|
101
|
-
const v = session.get<unknown>(lastTabKey(basePath, slug))
|
|
102
|
-
if (typeof v !== 'string') return undefined
|
|
103
|
-
return v
|
|
104
|
-
}
|
|
105
|
-
|
|
106
|
-
export function writePersistedLastTab(
|
|
107
|
-
req: unknown,
|
|
108
|
-
basePath: string,
|
|
109
|
-
slug: string,
|
|
110
|
-
tab: string,
|
|
111
|
-
): void {
|
|
112
|
-
const session = getSession(req)
|
|
113
|
-
if (!session) return
|
|
114
|
-
const key = lastTabKey(basePath, slug)
|
|
115
|
-
if (session.get<string>(key) === tab) return
|
|
116
|
-
session.put(key, tab)
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
export function encodePersistedQuery(
|
|
120
|
-
slice: Record<string, string>,
|
|
121
|
-
tab: string = '',
|
|
122
|
-
): string {
|
|
123
|
-
const params = new URLSearchParams()
|
|
124
|
-
if (tab !== '') params.set('tab', tab)
|
|
125
|
-
for (const [k, v] of Object.entries(slice)) {
|
|
126
|
-
if (v === '') continue
|
|
127
|
-
params.set(k, v)
|
|
128
|
-
}
|
|
129
|
-
return params.toString()
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
function shallowEqualStringMap(
|
|
133
|
-
a: Record<string, unknown>,
|
|
134
|
-
b: Record<string, string>,
|
|
135
|
-
): boolean {
|
|
136
|
-
const ak = Object.keys(a)
|
|
137
|
-
const bk = Object.keys(b)
|
|
138
|
-
if (ak.length !== bk.length) return false
|
|
139
|
-
for (const k of bk) {
|
|
140
|
-
if (a[k] !== b[k]) return false
|
|
141
|
-
}
|
|
142
|
-
return true
|
|
143
|
-
}
|
|
@@ -1,10 +0,0 @@
|
|
|
1
|
-
// Slot-component runtime registry — opt-in component registration for
|
|
2
|
-
// `SlotComponent` schema elements. Imported by panel `bootstrap/providers.ts`
|
|
3
|
-
// or by a plugin's `register(panel)` step (e.g. `@pilotiq-pro/ai`'s
|
|
4
|
-
// resource-header agents dropdown).
|
|
5
|
-
export {
|
|
6
|
-
registerSlotComponents,
|
|
7
|
-
getSlotComponent,
|
|
8
|
-
type SlotComponent,
|
|
9
|
-
type SlotComponentProps,
|
|
10
|
-
} from './registry.js'
|
|
@@ -1,56 +0,0 @@
|
|
|
1
|
-
import type { ComponentType } from 'react'
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Render-time component registry for `SlotComponent` schema elements.
|
|
5
|
-
* Mirrors the widget / entry registry pattern: users register components
|
|
6
|
-
* by name in their app's `bootstrap/providers.ts` (or via a plugin's
|
|
7
|
-
* `register(panel)` hook), the `SlotComponent` ships the registered name
|
|
8
|
-
* on the wire (`{ component: 'BookmarkButton' }`), and the renderer
|
|
9
|
-
* looks up the actual component at render through `getSlotComponent()`.
|
|
10
|
-
*
|
|
11
|
-
* Why a runtime registry rather than the build-time `_components.ts`
|
|
12
|
-
* manifest the icon system uses: slot components typically live in plugin
|
|
13
|
-
* packages or app-level bootstrap files, not on Resource / Global / Page
|
|
14
|
-
* top-level statics, so the existing manifest walker doesn't see them. A
|
|
15
|
-
* runtime register-by-name keeps the wire compact and mirrors the
|
|
16
|
-
* `registerWidgetComponents / registerEntryComponents` precedent.
|
|
17
|
-
*
|
|
18
|
-
* @example
|
|
19
|
-
* // bootstrap/providers.ts
|
|
20
|
-
* import { registerSlotComponents } from '@pilotiq/pilotiq/slot-components'
|
|
21
|
-
* import { BookmarkButton } from '#components/BookmarkButton.js'
|
|
22
|
-
* registerSlotComponents({ BookmarkButton })
|
|
23
|
-
*
|
|
24
|
-
* // Then a render-hook contribution (or any schema slot) can reference it:
|
|
25
|
-
* panel.renderHook(
|
|
26
|
-
* 'panels::resource.pages.edit-record.header.actions.before',
|
|
27
|
-
* () => [SlotComponent.make('BookmarkButton').props({ … })],
|
|
28
|
-
* )
|
|
29
|
-
*/
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* Props handed to every registered slot component. The schema-side
|
|
33
|
-
* `SlotComponent.props({...})` setter ships through verbatim — pack
|
|
34
|
-
* everything the component needs into that bag (basePath, recordId,
|
|
35
|
-
* pre-resolved agent metas, etc).
|
|
36
|
-
*/
|
|
37
|
-
export type SlotComponentProps = Record<string, unknown>
|
|
38
|
-
export type SlotComponent = ComponentType<SlotComponentProps>
|
|
39
|
-
|
|
40
|
-
const registry: Record<string, SlotComponent> = {}
|
|
41
|
-
|
|
42
|
-
export function registerSlotComponents(map: Record<string, SlotComponent>): void {
|
|
43
|
-
for (const key of Object.keys(map)) {
|
|
44
|
-
const c = map[key]
|
|
45
|
-
if (c) registry[key] = c
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
export function getSlotComponent(name: string): SlotComponent | undefined {
|
|
50
|
-
return registry[name]
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
/** Test-only: clear the registry. Not exported from the public entry. */
|
|
54
|
-
export function _resetSlotComponentRegistryForTests(): void {
|
|
55
|
-
for (const key of Object.keys(registry)) delete registry[key]
|
|
56
|
-
}
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
/* Stylesheet for `FileUploadField`'s image-cropping affordance.
|
|
2
|
-
*
|
|
3
|
-
* `react-image-crop` is a declared dep of `@pilotiq/pilotiq` and its
|
|
4
|
-
* CSS is required for the crop UI to render correctly. Consumers
|
|
5
|
-
* pull in the styles by importing this subpath from their app's CSS
|
|
6
|
-
* (Tailwind / global stylesheet):
|
|
7
|
-
*
|
|
8
|
-
* @import "@pilotiq/pilotiq/styles/file-upload.css";
|
|
9
|
-
*
|
|
10
|
-
* Importing through this subpath means consumers don't need to
|
|
11
|
-
* declare `react-image-crop` as a direct dep — the CSS @import below
|
|
12
|
-
* resolves through pilotiq's own node_modules. */
|
|
13
|
-
@import "react-image-crop/dist/ReactCrop.css";
|
|
@@ -1,84 +0,0 @@
|
|
|
1
|
-
import { describe, it } from 'node:test'
|
|
2
|
-
import assert from 'node:assert/strict'
|
|
3
|
-
|
|
4
|
-
import { Sum, Average, Count, Range } from './Summarizer.js'
|
|
5
|
-
|
|
6
|
-
describe('Summarizers', () => {
|
|
7
|
-
describe('Sum', () => {
|
|
8
|
-
it('sums numeric values', () => {
|
|
9
|
-
assert.equal(Sum.make().compute([1, 2, 3, 4]), '10')
|
|
10
|
-
})
|
|
11
|
-
|
|
12
|
-
it('coerces strings and ignores non-numerics', () => {
|
|
13
|
-
assert.equal(Sum.make().compute(['10', '5', null, undefined, 'abc']), '15')
|
|
14
|
-
})
|
|
15
|
-
|
|
16
|
-
it('returns 0 over an empty list', () => {
|
|
17
|
-
assert.equal(Sum.make().compute([]), '0')
|
|
18
|
-
})
|
|
19
|
-
|
|
20
|
-
it('applies a custom format function', () => {
|
|
21
|
-
const formatted = Sum.make()
|
|
22
|
-
.format((n) => new Intl.NumberFormat('en-US', { style: 'currency', currency: 'USD' }).format(n))
|
|
23
|
-
.compute([1.5, 2.25, 3])
|
|
24
|
-
assert.equal(formatted, '$6.75')
|
|
25
|
-
})
|
|
26
|
-
})
|
|
27
|
-
|
|
28
|
-
describe('Average', () => {
|
|
29
|
-
it('averages numeric values', () => {
|
|
30
|
-
assert.equal(Average.make().compute([2, 4, 6]), '4')
|
|
31
|
-
})
|
|
32
|
-
|
|
33
|
-
it('returns 0 for an empty list', () => {
|
|
34
|
-
assert.equal(Average.make().compute([]), '0')
|
|
35
|
-
})
|
|
36
|
-
|
|
37
|
-
it('skips non-numerics in the denominator', () => {
|
|
38
|
-
assert.equal(Average.make().compute([10, 'bad', 20]), '15')
|
|
39
|
-
})
|
|
40
|
-
})
|
|
41
|
-
|
|
42
|
-
describe('Count', () => {
|
|
43
|
-
it('counts every entry, including non-numerics', () => {
|
|
44
|
-
assert.equal(Count.make().compute([1, 'a', null, undefined]), '4')
|
|
45
|
-
})
|
|
46
|
-
|
|
47
|
-
it('returns 0 for empty input', () => {
|
|
48
|
-
assert.equal(Count.make().compute([]), '0')
|
|
49
|
-
})
|
|
50
|
-
|
|
51
|
-
it('ignores label and format options for the value (Count is integer-only)', () => {
|
|
52
|
-
const c = Count.make().label('Total entries').format(() => 'should be ignored')
|
|
53
|
-
assert.equal(c.compute([1, 2, 3]), '3')
|
|
54
|
-
assert.equal(c.toMeta().label, 'Total entries')
|
|
55
|
-
})
|
|
56
|
-
})
|
|
57
|
-
|
|
58
|
-
describe('Range', () => {
|
|
59
|
-
it('returns "min..max" over numeric values', () => {
|
|
60
|
-
assert.equal(Range.make().compute([3, 1, 4, 1, 5, 9, 2]), '1..9')
|
|
61
|
-
})
|
|
62
|
-
|
|
63
|
-
it('returns "—" when no numeric values are present', () => {
|
|
64
|
-
assert.equal(Range.make().compute([null, undefined, 'abc']), '—')
|
|
65
|
-
})
|
|
66
|
-
|
|
67
|
-
it('uses format function on each end of the range', () => {
|
|
68
|
-
const r = Range.make().format((n) => `$${n}`).compute([5, 25, 10])
|
|
69
|
-
assert.equal(r, '$5..$25')
|
|
70
|
-
})
|
|
71
|
-
})
|
|
72
|
-
|
|
73
|
-
describe('toResult / toMeta', () => {
|
|
74
|
-
it('toMeta carries kind + label only (no value)', () => {
|
|
75
|
-
const m = Sum.make().label('Total').toMeta()
|
|
76
|
-
assert.deepEqual(m, { kind: 'sum', label: 'Total' })
|
|
77
|
-
})
|
|
78
|
-
|
|
79
|
-
it('toResult bundles label + computed value', () => {
|
|
80
|
-
const r = Average.make().label('Avg').toResult([10, 20])
|
|
81
|
-
assert.deepEqual(r, { kind: 'average', label: 'Avg', value: '15' })
|
|
82
|
-
})
|
|
83
|
-
})
|
|
84
|
-
})
|