@fayz-ai/saas 0.1.6
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/dist/app/AdminShell.d.ts +31 -0
- package/dist/app/AdminShell.d.ts.map +1 -0
- package/dist/app/LoginPage.d.ts +17 -0
- package/dist/app/LoginPage.d.ts.map +1 -0
- package/dist/app/config.d.ts +83 -0
- package/dist/app/config.d.ts.map +1 -0
- package/dist/app/createFayzApp.d.ts +11 -0
- package/dist/app/createFayzApp.d.ts.map +1 -0
- package/dist/app/routing.d.ts +10 -0
- package/dist/app/routing.d.ts.map +1 -0
- package/dist/app/scaffold.d.ts +17 -0
- package/dist/app/scaffold.d.ts.map +1 -0
- package/dist/archetype-lookup.d.ts +30 -0
- package/dist/archetype-lookup.d.ts.map +1 -0
- package/dist/billing/index.cjs +12 -0
- package/dist/billing/index.cjs.map +1 -0
- package/dist/billing/index.d.ts +3 -0
- package/dist/billing/index.d.ts.map +1 -0
- package/dist/billing/index.js +3 -0
- package/dist/billing/index.js.map +1 -0
- package/dist/billing/store.d.ts +35 -0
- package/dist/billing/store.d.ts.map +1 -0
- package/dist/chunk-4XTQ4V5L.cjs +124 -0
- package/dist/chunk-4XTQ4V5L.cjs.map +1 -0
- package/dist/chunk-5GYQK5IW.js +21 -0
- package/dist/chunk-5GYQK5IW.js.map +1 -0
- package/dist/chunk-DJW67TCY.js +781 -0
- package/dist/chunk-DJW67TCY.js.map +1 -0
- package/dist/chunk-KYID2MR6.cjs +23 -0
- package/dist/chunk-KYID2MR6.cjs.map +1 -0
- package/dist/chunk-QGMDVLLW.cjs +816 -0
- package/dist/chunk-QGMDVLLW.cjs.map +1 -0
- package/dist/chunk-XNWCGJYH.js +97 -0
- package/dist/chunk-XNWCGJYH.js.map +1 -0
- package/dist/components/shared/PersonLink.d.ts +18 -0
- package/dist/components/shared/PersonLink.d.ts.map +1 -0
- package/dist/crud/CrudCardGrid.d.ts +13 -0
- package/dist/crud/CrudCardGrid.d.ts.map +1 -0
- package/dist/crud/CrudDetailPage.d.ts +18 -0
- package/dist/crud/CrudDetailPage.d.ts.map +1 -0
- package/dist/crud/CrudFormPage.d.ts +18 -0
- package/dist/crud/CrudFormPage.d.ts.map +1 -0
- package/dist/crud/CrudListView.d.ts +51 -0
- package/dist/crud/CrudListView.d.ts.map +1 -0
- package/dist/crud/CrudPage.d.ts +20 -0
- package/dist/crud/CrudPage.d.ts.map +1 -0
- package/dist/crud/DeleteConfirmDialog.d.ts +11 -0
- package/dist/crud/DeleteConfirmDialog.d.ts.map +1 -0
- package/dist/crud/EntityOverview.d.ts +14 -0
- package/dist/crud/EntityOverview.d.ts.map +1 -0
- package/dist/crud/ImportWizard.d.ts +18 -0
- package/dist/crud/ImportWizard.d.ts.map +1 -0
- package/dist/crud/archetypes/ActiveToggle.d.ts +8 -0
- package/dist/crud/archetypes/ActiveToggle.d.ts.map +1 -0
- package/dist/crud/archetypes/ArchetypeStatusBar.d.ts +11 -0
- package/dist/crud/archetypes/ArchetypeStatusBar.d.ts.map +1 -0
- package/dist/crud/archetypes/BlockSettingsPopover.d.ts +15 -0
- package/dist/crud/archetypes/BlockSettingsPopover.d.ts.map +1 -0
- package/dist/crud/archetypes/EntityLink.d.ts +32 -0
- package/dist/crud/archetypes/EntityLink.d.ts.map +1 -0
- package/dist/crud/archetypes/LocationDetailTabs.d.ts +11 -0
- package/dist/crud/archetypes/LocationDetailTabs.d.ts.map +1 -0
- package/dist/crud/archetypes/LocationFormLayout.d.ts +12 -0
- package/dist/crud/archetypes/LocationFormLayout.d.ts.map +1 -0
- package/dist/crud/archetypes/PersonDetailTabs.d.ts +22 -0
- package/dist/crud/archetypes/PersonDetailTabs.d.ts.map +1 -0
- package/dist/crud/archetypes/PersonFormLayout.d.ts +13 -0
- package/dist/crud/archetypes/PersonFormLayout.d.ts.map +1 -0
- package/dist/crud/archetypes/PersonPicker.d.ts +23 -0
- package/dist/crud/archetypes/PersonPicker.d.ts.map +1 -0
- package/dist/crud/archetypes/ProductDetailTabs.d.ts +11 -0
- package/dist/crud/archetypes/ProductDetailTabs.d.ts.map +1 -0
- package/dist/crud/archetypes/ProductFormLayout.d.ts +12 -0
- package/dist/crud/archetypes/ProductFormLayout.d.ts.map +1 -0
- package/dist/crud/archetypes/ScheduleEditor.d.ts +7 -0
- package/dist/crud/archetypes/ScheduleEditor.d.ts.map +1 -0
- package/dist/crud/archetypes/ServiceDetailTabs.d.ts +10 -0
- package/dist/crud/archetypes/ServiceDetailTabs.d.ts.map +1 -0
- package/dist/crud/archetypes/ServiceFormLayout.d.ts +12 -0
- package/dist/crud/archetypes/ServiceFormLayout.d.ts.map +1 -0
- package/dist/crud/archetypes/SubjectDetailTabs.d.ts +12 -0
- package/dist/crud/archetypes/SubjectDetailTabs.d.ts.map +1 -0
- package/dist/crud/archetypes/SubjectFormLayout.d.ts +14 -0
- package/dist/crud/archetypes/SubjectFormLayout.d.ts.map +1 -0
- package/dist/crud/createCrudPage.d.ts +20 -0
- package/dist/crud/createCrudPage.d.ts.map +1 -0
- package/dist/crud/csv-export.d.ts +3 -0
- package/dist/crud/csv-export.d.ts.map +1 -0
- package/dist/crud/fieldToColumn.d.ts +14 -0
- package/dist/crud/fieldToColumn.d.ts.map +1 -0
- package/dist/crud/index.d.ts +12 -0
- package/dist/crud/index.d.ts.map +1 -0
- package/dist/hooks/useFieldRules.d.ts +5 -0
- package/dist/hooks/useFieldRules.d.ts.map +1 -0
- package/dist/hooks/useModuleNavigation.d.ts +25 -0
- package/dist/hooks/useModuleNavigation.d.ts.map +1 -0
- package/dist/hooks/usePluginPrefs.d.ts +23 -0
- package/dist/hooks/usePluginPrefs.d.ts.map +1 -0
- package/dist/index.cjs +13880 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.ts +46 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +13730 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/currency.d.ts +13 -0
- package/dist/lib/currency.d.ts.map +1 -0
- package/dist/lib/entity-routes.d.ts +24 -0
- package/dist/lib/entity-routes.d.ts.map +1 -0
- package/dist/lib/schedule-config.d.ts +32 -0
- package/dist/lib/schedule-config.d.ts.map +1 -0
- package/dist/lib/schedule-service.d.ts +76 -0
- package/dist/lib/schedule-service.d.ts.map +1 -0
- package/dist/org/adapters/mock.d.ts +4 -0
- package/dist/org/adapters/mock.d.ts.map +1 -0
- package/dist/org/adapters/supabase.d.ts +7 -0
- package/dist/org/adapters/supabase.d.ts.map +1 -0
- package/dist/org/context.d.ts +29 -0
- package/dist/org/context.d.ts.map +1 -0
- package/dist/org/index.cjs +44 -0
- package/dist/org/index.cjs.map +1 -0
- package/dist/org/index.d.ts +8 -0
- package/dist/org/index.d.ts.map +1 -0
- package/dist/org/index.js +3 -0
- package/dist/org/index.js.map +1 -0
- package/dist/org/store.d.ts +16 -0
- package/dist/org/store.d.ts.map +1 -0
- package/dist/permissions/PermissionGate.d.ts +12 -0
- package/dist/permissions/PermissionGate.d.ts.map +1 -0
- package/dist/permissions/context.d.ts +37 -0
- package/dist/permissions/context.d.ts.map +1 -0
- package/dist/permissions/index.cjs +32 -0
- package/dist/permissions/index.cjs.map +1 -0
- package/dist/permissions/index.d.ts +5 -0
- package/dist/permissions/index.d.ts.map +1 -0
- package/dist/permissions/index.js +3 -0
- package/dist/permissions/index.js.map +1 -0
- package/dist/permissions/store.d.ts +21 -0
- package/dist/permissions/store.d.ts.map +1 -0
- package/dist/placeholder.d.ts +7 -0
- package/dist/placeholder.d.ts.map +1 -0
- package/dist/plugins/ModuleActionBar.d.ts +27 -0
- package/dist/plugins/ModuleActionBar.d.ts.map +1 -0
- package/dist/plugins/PluginRegistryManager.d.ts +13 -0
- package/dist/plugins/PluginRegistryManager.d.ts.map +1 -0
- package/dist/plugins/QuickActionsButton.d.ts +12 -0
- package/dist/plugins/QuickActionsButton.d.ts.map +1 -0
- package/dist/plugins/SettingsGroup.d.ts +25 -0
- package/dist/plugins/SettingsGroup.d.ts.map +1 -0
- package/dist/plugins/WidgetSlot.d.ts +8 -0
- package/dist/plugins/WidgetSlot.d.ts.map +1 -0
- package/dist/plugins/createPluginContext.d.ts +28 -0
- package/dist/plugins/createPluginContext.d.ts.map +1 -0
- package/dist/plugins/createViewRouter.d.ts +38 -0
- package/dist/plugins/createViewRouter.d.ts.map +1 -0
- package/dist/shell/components/auth/CenteredLogin.d.ts +13 -0
- package/dist/shell/components/auth/CenteredLogin.d.ts.map +1 -0
- package/dist/shell/components/auth/LoginForm.d.ts +12 -0
- package/dist/shell/components/auth/LoginForm.d.ts.map +1 -0
- package/dist/shell/components/auth/LoginPage.d.ts +15 -0
- package/dist/shell/components/auth/LoginPage.d.ts.map +1 -0
- package/dist/shell/components/auth/OAuthButtons.d.ts +10 -0
- package/dist/shell/components/auth/OAuthButtons.d.ts.map +1 -0
- package/dist/shell/components/auth/ProtectedRoute.d.ts +11 -0
- package/dist/shell/components/auth/ProtectedRoute.d.ts.map +1 -0
- package/dist/shell/components/auth/RecoveryForm.d.ts +9 -0
- package/dist/shell/components/auth/RecoveryForm.d.ts.map +1 -0
- package/dist/shell/components/auth/SignupForm.d.ts +10 -0
- package/dist/shell/components/auth/SignupForm.d.ts.map +1 -0
- package/dist/shell/components/auth/SplitLogin.d.ts +14 -0
- package/dist/shell/components/auth/SplitLogin.d.ts.map +1 -0
- package/dist/shell/components/auth/index.d.ts +7 -0
- package/dist/shell/components/auth/index.d.ts.map +1 -0
- package/dist/shell/components/billing/BillingPage.d.ts +7 -0
- package/dist/shell/components/billing/BillingPage.d.ts.map +1 -0
- package/dist/shell/components/billing/InvoiceList.d.ts +9 -0
- package/dist/shell/components/billing/InvoiceList.d.ts.map +1 -0
- package/dist/shell/components/billing/PaywallGate.d.ts +11 -0
- package/dist/shell/components/billing/PaywallGate.d.ts.map +1 -0
- package/dist/shell/components/billing/PlanSelector.d.ts +13 -0
- package/dist/shell/components/billing/PlanSelector.d.ts.map +1 -0
- package/dist/shell/components/billing/SubscriptionCard.d.ts +12 -0
- package/dist/shell/components/billing/SubscriptionCard.d.ts.map +1 -0
- package/dist/shell/components/billing/index.d.ts +6 -0
- package/dist/shell/components/billing/index.d.ts.map +1 -0
- package/dist/shell/components/chat/ChatFab.d.ts +9 -0
- package/dist/shell/components/chat/ChatFab.d.ts.map +1 -0
- package/dist/shell/components/chat/ChatPanel.d.ts +10 -0
- package/dist/shell/components/chat/ChatPanel.d.ts.map +1 -0
- package/dist/shell/components/chat/ChatSuggestions.d.ts +14 -0
- package/dist/shell/components/chat/ChatSuggestions.d.ts.map +1 -0
- package/dist/shell/components/chat/index.d.ts +3 -0
- package/dist/shell/components/chat/index.d.ts.map +1 -0
- package/dist/shell/components/crud/archetypes/ClientOrdersTab.d.ts +16 -0
- package/dist/shell/components/crud/archetypes/ClientOrdersTab.d.ts.map +1 -0
- package/dist/shell/components/layout/BottomNav.d.ts +24 -0
- package/dist/shell/components/layout/BottomNav.d.ts.map +1 -0
- package/dist/shell/components/layout/CommandPalette.d.ts +29 -0
- package/dist/shell/components/layout/CommandPalette.d.ts.map +1 -0
- package/dist/shell/components/layout/LocationSwitchOverlay.d.ts +3 -0
- package/dist/shell/components/layout/LocationSwitchOverlay.d.ts.map +1 -0
- package/dist/shell/components/layout/MockModeBanner.d.ts +3 -0
- package/dist/shell/components/layout/MockModeBanner.d.ts.map +1 -0
- package/dist/shell/components/layout/PrintChrome.d.ts +10 -0
- package/dist/shell/components/layout/PrintChrome.d.ts.map +1 -0
- package/dist/shell/components/layout/index.d.ts +6 -0
- package/dist/shell/components/layout/index.d.ts.map +1 -0
- package/dist/shell/components/notifications/ChangelogFeed.d.ts +8 -0
- package/dist/shell/components/notifications/ChangelogFeed.d.ts.map +1 -0
- package/dist/shell/components/notifications/NotificationBell.d.ts +9 -0
- package/dist/shell/components/notifications/NotificationBell.d.ts.map +1 -0
- package/dist/shell/components/notifications/NotificationInbox.d.ts +13 -0
- package/dist/shell/components/notifications/NotificationInbox.d.ts.map +1 -0
- package/dist/shell/components/notifications/ToastProvider.d.ts +2 -0
- package/dist/shell/components/notifications/ToastProvider.d.ts.map +1 -0
- package/dist/shell/components/notifications/index.d.ts +5 -0
- package/dist/shell/components/notifications/index.d.ts.map +1 -0
- package/dist/shell/components/organization/ImpersonationBanner.d.ts +3 -0
- package/dist/shell/components/organization/ImpersonationBanner.d.ts.map +1 -0
- package/dist/shell/components/organization/InviteMemberDialog.d.ts +10 -0
- package/dist/shell/components/organization/InviteMemberDialog.d.ts.map +1 -0
- package/dist/shell/components/organization/OrgInitializer.d.ts +7 -0
- package/dist/shell/components/organization/OrgInitializer.d.ts.map +1 -0
- package/dist/shell/components/organization/OrgSwitcher.d.ts +7 -0
- package/dist/shell/components/organization/OrgSwitcher.d.ts.map +1 -0
- package/dist/shell/components/organization/PermissionGate.d.ts +11 -0
- package/dist/shell/components/organization/PermissionGate.d.ts.map +1 -0
- package/dist/shell/components/organization/PermissionMatrixEditor.d.ts +17 -0
- package/dist/shell/components/organization/PermissionMatrixEditor.d.ts.map +1 -0
- package/dist/shell/components/organization/PermissionProfilesTab.d.ts +3 -0
- package/dist/shell/components/organization/PermissionProfilesTab.d.ts.map +1 -0
- package/dist/shell/components/organization/TeamTab.d.ts +3 -0
- package/dist/shell/components/organization/TeamTab.d.ts.map +1 -0
- package/dist/shell/components/organization/TenantOnboarding.d.ts +9 -0
- package/dist/shell/components/organization/TenantOnboarding.d.ts.map +1 -0
- package/dist/shell/components/organization/index.d.ts +8 -0
- package/dist/shell/components/organization/index.d.ts.map +1 -0
- package/dist/shell/components/plugins/PluginSettingsPanel.d.ts +23 -0
- package/dist/shell/components/plugins/PluginSettingsPanel.d.ts.map +1 -0
- package/dist/shell/components/settings/BrandingSettings.d.ts +12 -0
- package/dist/shell/components/settings/BrandingSettings.d.ts.map +1 -0
- package/dist/shell/components/settings/CompanySettings.d.ts +13 -0
- package/dist/shell/components/settings/CompanySettings.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedBrandingSettings.d.ts +2 -0
- package/dist/shell/components/settings/ConnectedBrandingSettings.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedCompanySettings.d.ts +2 -0
- package/dist/shell/components/settings/ConnectedCompanySettings.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedFieldRulesSettings.d.ts +2 -0
- package/dist/shell/components/settings/ConnectedFieldRulesSettings.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedHolidaysSettings.d.ts +3 -0
- package/dist/shell/components/settings/ConnectedHolidaysSettings.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedLocationsOverview.d.ts +3 -0
- package/dist/shell/components/settings/ConnectedLocationsOverview.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedSecuritySettings.d.ts +2 -0
- package/dist/shell/components/settings/ConnectedSecuritySettings.d.ts.map +1 -0
- package/dist/shell/components/settings/ConnectedUserProfile.d.ts +2 -0
- package/dist/shell/components/settings/ConnectedUserProfile.d.ts.map +1 -0
- package/dist/shell/components/settings/FieldRulesSettings.d.ts +11 -0
- package/dist/shell/components/settings/FieldRulesSettings.d.ts.map +1 -0
- package/dist/shell/components/settings/HolidaysSettings.d.ts +21 -0
- package/dist/shell/components/settings/HolidaysSettings.d.ts.map +1 -0
- package/dist/shell/components/settings/LocationsCrudPage.d.ts +6 -0
- package/dist/shell/components/settings/LocationsCrudPage.d.ts.map +1 -0
- package/dist/shell/components/settings/LocationsOverview.d.ts +19 -0
- package/dist/shell/components/settings/LocationsOverview.d.ts.map +1 -0
- package/dist/shell/components/settings/SecuritySettings.d.ts +17 -0
- package/dist/shell/components/settings/SecuritySettings.d.ts.map +1 -0
- package/dist/shell/components/settings/SettingsPage.d.ts +18 -0
- package/dist/shell/components/settings/SettingsPage.d.ts.map +1 -0
- package/dist/shell/components/settings/UserProfile.d.ts +12 -0
- package/dist/shell/components/settings/UserProfile.d.ts.map +1 -0
- package/dist/shell/components/settings/index.d.ts +11 -0
- package/dist/shell/components/settings/index.d.ts.map +1 -0
- package/dist/shell/config/index.d.ts +7 -0
- package/dist/shell/config/index.d.ts.map +1 -0
- package/dist/shell/config/permissions.d.ts +7 -0
- package/dist/shell/config/permissions.d.ts.map +1 -0
- package/dist/shell/config/tailwind-preset.d.ts +4 -0
- package/dist/shell/config/tailwind-preset.d.ts.map +1 -0
- package/dist/shell/config/theme/dark.d.ts +3 -0
- package/dist/shell/config/theme/dark.d.ts.map +1 -0
- package/dist/shell/config/theme/index.d.ts +6 -0
- package/dist/shell/config/theme/index.d.ts.map +1 -0
- package/dist/shell/config/theme/light.d.ts +3 -0
- package/dist/shell/config/theme/light.d.ts.map +1 -0
- package/dist/shell/config/theme/tokens.d.ts +97 -0
- package/dist/shell/config/theme/tokens.d.ts.map +1 -0
- package/dist/shell/config/theme/utils.d.ts +23 -0
- package/dist/shell/config/theme/utils.d.ts.map +1 -0
- package/dist/shell/createSaasApp.d.ts +145 -0
- package/dist/shell/createSaasApp.d.ts.map +1 -0
- package/dist/shell/hooks/index.d.ts +19 -0
- package/dist/shell/hooks/index.d.ts.map +1 -0
- package/dist/shell/hooks/useAITools.d.ts +21 -0
- package/dist/shell/hooks/useAITools.d.ts.map +1 -0
- package/dist/shell/hooks/useAuth.d.ts +16 -0
- package/dist/shell/hooks/useAuth.d.ts.map +1 -0
- package/dist/shell/hooks/useBilling.d.ts +14 -0
- package/dist/shell/hooks/useBilling.d.ts.map +1 -0
- package/dist/shell/hooks/useChat.d.ts +16 -0
- package/dist/shell/hooks/useChat.d.ts.map +1 -0
- package/dist/shell/hooks/useFieldRules.d.ts +6 -0
- package/dist/shell/hooks/useFieldRules.d.ts.map +1 -0
- package/dist/shell/hooks/useFormatters.d.ts +7 -0
- package/dist/shell/hooks/useFormatters.d.ts.map +1 -0
- package/dist/shell/hooks/useLayout.d.ts +16 -0
- package/dist/shell/hooks/useLayout.d.ts.map +1 -0
- package/dist/shell/hooks/useModuleNavigation.d.ts +25 -0
- package/dist/shell/hooks/useModuleNavigation.d.ts.map +1 -0
- package/dist/shell/hooks/useNotifications.d.ts +11 -0
- package/dist/shell/hooks/useNotifications.d.ts.map +1 -0
- package/dist/shell/hooks/usePermission.d.ts +16 -0
- package/dist/shell/hooks/usePermission.d.ts.map +1 -0
- package/dist/shell/hooks/usePluginPrefs.d.ts +23 -0
- package/dist/shell/hooks/usePluginPrefs.d.ts.map +1 -0
- package/dist/shell/hooks/usePlugins.d.ts +5 -0
- package/dist/shell/hooks/usePlugins.d.ts.map +1 -0
- package/dist/shell/hooks/useTenant.d.ts +23 -0
- package/dist/shell/hooks/useTenant.d.ts.map +1 -0
- package/dist/shell/hooks/useTenantPlugins.d.ts +10 -0
- package/dist/shell/hooks/useTenantPlugins.d.ts.map +1 -0
- package/dist/shell/hooks/useTranslation.d.ts +5 -0
- package/dist/shell/hooks/useTranslation.d.ts.map +1 -0
- package/dist/shell/lib/api.d.ts +28 -0
- package/dist/shell/lib/api.d.ts.map +1 -0
- package/dist/shell/lib/apply-field-rules.d.ts +4 -0
- package/dist/shell/lib/apply-field-rules.d.ts.map +1 -0
- package/dist/shell/lib/auth-adapters/mock.d.ts +3 -0
- package/dist/shell/lib/auth-adapters/mock.d.ts.map +1 -0
- package/dist/shell/lib/auth-adapters/supabase.d.ts +3 -0
- package/dist/shell/lib/auth-adapters/supabase.d.ts.map +1 -0
- package/dist/shell/lib/auth-context.d.ts +5 -0
- package/dist/shell/lib/auth-context.d.ts.map +1 -0
- package/dist/shell/lib/cache.d.ts +18 -0
- package/dist/shell/lib/cache.d.ts.map +1 -0
- package/dist/shell/lib/cn.d.ts +2 -0
- package/dist/shell/lib/cn.d.ts.map +1 -0
- package/dist/shell/lib/core-ai-tools.d.ts +15 -0
- package/dist/shell/lib/core-ai-tools.d.ts.map +1 -0
- package/dist/shell/lib/create-client-orders-provider.d.ts +3 -0
- package/dist/shell/lib/create-client-orders-provider.d.ts.map +1 -0
- package/dist/shell/lib/csv.d.ts +14 -0
- package/dist/shell/lib/csv.d.ts.map +1 -0
- package/dist/shell/lib/dedup.d.ts +2 -0
- package/dist/shell/lib/dedup.d.ts.map +1 -0
- package/dist/shell/lib/entity-registry.d.ts +19 -0
- package/dist/shell/lib/entity-registry.d.ts.map +1 -0
- package/dist/shell/lib/entity-routes.d.ts +24 -0
- package/dist/shell/lib/entity-routes.d.ts.map +1 -0
- package/dist/shell/lib/format.d.ts +7 -0
- package/dist/shell/lib/format.d.ts.map +1 -0
- package/dist/shell/lib/i18n.d.ts +4 -0
- package/dist/shell/lib/i18n.d.ts.map +1 -0
- package/dist/shell/lib/locale-config.d.ts +8 -0
- package/dist/shell/lib/locale-config.d.ts.map +1 -0
- package/dist/shell/lib/org-adapters/mock.d.ts +4 -0
- package/dist/shell/lib/org-adapters/mock.d.ts.map +1 -0
- package/dist/shell/lib/org-adapters/supabase.d.ts +3 -0
- package/dist/shell/lib/org-adapters/supabase.d.ts.map +1 -0
- package/dist/shell/lib/org-context.d.ts +5 -0
- package/dist/shell/lib/org-context.d.ts.map +1 -0
- package/dist/shell/lib/plugins.d.ts +2 -0
- package/dist/shell/lib/plugins.d.ts.map +1 -0
- package/dist/shell/lib/router.d.ts +15 -0
- package/dist/shell/lib/router.d.ts.map +1 -0
- package/dist/shell/lib/schedule-config.d.ts +32 -0
- package/dist/shell/lib/schedule-config.d.ts.map +1 -0
- package/dist/shell/lib/schedule-service.d.ts +76 -0
- package/dist/shell/lib/schedule-service.d.ts.map +1 -0
- package/dist/shell/lib/supabase.d.ts +9 -0
- package/dist/shell/lib/supabase.d.ts.map +1 -0
- package/dist/shell/stores/auth.store.d.ts +12 -0
- package/dist/shell/stores/auth.store.d.ts.map +1 -0
- package/dist/shell/stores/billing.store.d.ts +2 -0
- package/dist/shell/stores/billing.store.d.ts.map +1 -0
- package/dist/shell/stores/chat.store.d.ts +20 -0
- package/dist/shell/stores/chat.store.d.ts.map +1 -0
- package/dist/shell/stores/index.d.ts +13 -0
- package/dist/shell/stores/index.d.ts.map +1 -0
- package/dist/shell/stores/invite.store.d.ts +11 -0
- package/dist/shell/stores/invite.store.d.ts.map +1 -0
- package/dist/shell/stores/layout.store.d.ts +2 -0
- package/dist/shell/stores/layout.store.d.ts.map +1 -0
- package/dist/shell/stores/locale.store.d.ts +7 -0
- package/dist/shell/stores/locale.store.d.ts.map +1 -0
- package/dist/shell/stores/location.store.d.ts +20 -0
- package/dist/shell/stores/location.store.d.ts.map +1 -0
- package/dist/shell/stores/notifications.store.d.ts +12 -0
- package/dist/shell/stores/notifications.store.d.ts.map +1 -0
- package/dist/shell/stores/organization.store.d.ts +2 -0
- package/dist/shell/stores/organization.store.d.ts.map +1 -0
- package/dist/shell/stores/permissions.store.d.ts +2 -0
- package/dist/shell/stores/permissions.store.d.ts.map +1 -0
- package/dist/shell/stores/plugin.store.d.ts +24 -0
- package/dist/shell/stores/plugin.store.d.ts.map +1 -0
- package/dist/shell/stores/tenant.store.d.ts +11 -0
- package/dist/shell/stores/tenant.store.d.ts.map +1 -0
- package/dist/shell/stores/theme.store.d.ts +17 -0
- package/dist/shell/stores/theme.store.d.ts.map +1 -0
- package/dist/shell/test-setup.d.ts +2 -0
- package/dist/shell/test-setup.d.ts.map +1 -0
- package/dist/shell/types/auth-adapter.d.ts +14 -0
- package/dist/shell/types/auth-adapter.d.ts.map +1 -0
- package/dist/shell/types/auth.d.ts +34 -0
- package/dist/shell/types/auth.d.ts.map +1 -0
- package/dist/shell/types/billing.d.ts +74 -0
- package/dist/shell/types/billing.d.ts.map +1 -0
- package/dist/shell/types/client-orders.d.ts +43 -0
- package/dist/shell/types/client-orders.d.ts.map +1 -0
- package/dist/shell/types/crud.d.ts +122 -0
- package/dist/shell/types/crud.d.ts.map +1 -0
- package/dist/shell/types/entities.d.ts +150 -0
- package/dist/shell/types/entities.d.ts.map +1 -0
- package/dist/shell/types/entity-lookup.d.ts +18 -0
- package/dist/shell/types/entity-lookup.d.ts.map +1 -0
- package/dist/shell/types/index.d.ts +17 -0
- package/dist/shell/types/index.d.ts.map +1 -0
- package/dist/shell/types/integrations.d.ts +48 -0
- package/dist/shell/types/integrations.d.ts.map +1 -0
- package/dist/shell/types/invite.d.ts +23 -0
- package/dist/shell/types/invite.d.ts.map +1 -0
- package/dist/shell/types/layout.d.ts +26 -0
- package/dist/shell/types/layout.d.ts.map +1 -0
- package/dist/shell/types/notifications.d.ts +31 -0
- package/dist/shell/types/notifications.d.ts.map +1 -0
- package/dist/shell/types/org-adapter.d.ts +72 -0
- package/dist/shell/types/org-adapter.d.ts.map +1 -0
- package/dist/shell/types/permissions.d.ts +39 -0
- package/dist/shell/types/permissions.d.ts.map +1 -0
- package/dist/shell/types/platform.d.ts +25 -0
- package/dist/shell/types/platform.d.ts.map +1 -0
- package/dist/shell/types/plugins.d.ts +308 -0
- package/dist/shell/types/plugins.d.ts.map +1 -0
- package/dist/shell/types/tenant.d.ts +74 -0
- package/dist/shell/types/tenant.d.ts.map +1 -0
- package/dist/stores/createCrudStore.d.ts +32 -0
- package/dist/stores/createCrudStore.d.ts.map +1 -0
- package/dist/supabase/client.d.ts +18 -0
- package/dist/supabase/client.d.ts.map +1 -0
- package/package.json +72 -0
- package/src/app/AdminShell.tsx +449 -0
- package/src/app/LoginPage.tsx +264 -0
- package/src/app/config.ts +146 -0
- package/src/app/createFayzApp.tsx +254 -0
- package/src/app/routing.tsx +65 -0
- package/src/app/scaffold.tsx +175 -0
- package/src/archetype-lookup.ts +160 -0
- package/src/billing/index.ts +7 -0
- package/src/billing/store.ts +62 -0
- package/src/components/shared/PersonLink.tsx +199 -0
- package/src/crud/CrudCardGrid.tsx +92 -0
- package/src/crud/CrudDetailPage.tsx +366 -0
- package/src/crud/CrudFormPage.tsx +467 -0
- package/src/crud/CrudListView.tsx +223 -0
- package/src/crud/CrudPage.tsx +491 -0
- package/src/crud/DeleteConfirmDialog.tsx +41 -0
- package/src/crud/EntityOverview.tsx +230 -0
- package/src/crud/ImportWizard.tsx +690 -0
- package/src/crud/archetypes/ActiveToggle.tsx +23 -0
- package/src/crud/archetypes/ArchetypeStatusBar.tsx +58 -0
- package/src/crud/archetypes/BlockSettingsPopover.tsx +322 -0
- package/src/crud/archetypes/EntityLink.tsx +74 -0
- package/src/crud/archetypes/LocationDetailTabs.tsx +72 -0
- package/src/crud/archetypes/LocationFormLayout.tsx +173 -0
- package/src/crud/archetypes/PersonDetailTabs.tsx +87 -0
- package/src/crud/archetypes/PersonFormLayout.tsx +184 -0
- package/src/crud/archetypes/PersonPicker.tsx +299 -0
- package/src/crud/archetypes/ProductDetailTabs.tsx +72 -0
- package/src/crud/archetypes/ProductFormLayout.tsx +197 -0
- package/src/crud/archetypes/ScheduleEditor.tsx +796 -0
- package/src/crud/archetypes/ServiceDetailTabs.tsx +60 -0
- package/src/crud/archetypes/ServiceFormLayout.tsx +155 -0
- package/src/crud/archetypes/SubjectDetailTabs.tsx +142 -0
- package/src/crud/archetypes/SubjectFormLayout.tsx +164 -0
- package/src/crud/createCrudPage.tsx +65 -0
- package/src/crud/csv-export.ts +17 -0
- package/src/crud/fieldToColumn.tsx +109 -0
- package/src/crud/index.ts +11 -0
- package/src/hooks/useFieldRules.ts +10 -0
- package/src/hooks/useModuleNavigation.ts +124 -0
- package/src/hooks/usePluginPrefs.ts +113 -0
- package/src/index.ts +168 -0
- package/src/lib/currency.ts +19 -0
- package/src/lib/entity-routes.ts +53 -0
- package/src/lib/schedule-config.ts +42 -0
- package/src/lib/schedule-service.ts +307 -0
- package/src/org/adapters/mock.ts +353 -0
- package/src/org/adapters/supabase.ts +509 -0
- package/src/org/context.tsx +174 -0
- package/src/org/index.ts +19 -0
- package/src/org/store.ts +70 -0
- package/src/permissions/PermissionGate.tsx +17 -0
- package/src/permissions/context.tsx +126 -0
- package/src/permissions/index.ts +13 -0
- package/src/permissions/store.ts +68 -0
- package/src/placeholder.ts +32 -0
- package/src/plugins/ModuleActionBar.tsx +57 -0
- package/src/plugins/PluginRegistryManager.tsx +110 -0
- package/src/plugins/QuickActionsButton.tsx +73 -0
- package/src/plugins/SettingsGroup.tsx +100 -0
- package/src/plugins/WidgetSlot.tsx +38 -0
- package/src/plugins/createPluginContext.tsx +68 -0
- package/src/plugins/createViewRouter.tsx +45 -0
- package/src/shell/components/auth/CenteredLogin.tsx +86 -0
- package/src/shell/components/auth/LoginForm.tsx +129 -0
- package/src/shell/components/auth/LoginPage.tsx +22 -0
- package/src/shell/components/auth/OAuthButtons.tsx +66 -0
- package/src/shell/components/auth/ProtectedRoute.tsx +52 -0
- package/src/shell/components/auth/RecoveryForm.tsx +110 -0
- package/src/shell/components/auth/SignupForm.tsx +170 -0
- package/src/shell/components/auth/SplitLogin.tsx +124 -0
- package/src/shell/components/auth/index.ts +6 -0
- package/src/shell/components/billing/BillingPage.tsx +56 -0
- package/src/shell/components/billing/InvoiceList.tsx +123 -0
- package/src/shell/components/billing/PaywallGate.tsx +80 -0
- package/src/shell/components/billing/PlanSelector.tsx +125 -0
- package/src/shell/components/billing/SubscriptionCard.tsx +108 -0
- package/src/shell/components/billing/index.ts +5 -0
- package/src/shell/components/chat/ChatFab.tsx +195 -0
- package/src/shell/components/chat/ChatPanel.tsx +170 -0
- package/src/shell/components/chat/ChatSuggestions.tsx +102 -0
- package/src/shell/components/chat/index.ts +2 -0
- package/src/shell/components/crud/archetypes/ClientOrdersTab.tsx +212 -0
- package/src/shell/components/layout/BottomNav.tsx +131 -0
- package/src/shell/components/layout/CommandPalette.tsx +257 -0
- package/src/shell/components/layout/LocationSwitchOverlay.tsx +64 -0
- package/src/shell/components/layout/MockModeBanner.tsx +17 -0
- package/src/shell/components/layout/PrintChrome.tsx +200 -0
- package/src/shell/components/layout/index.ts +5 -0
- package/src/shell/components/notifications/ChangelogFeed.tsx +85 -0
- package/src/shell/components/notifications/NotificationBell.tsx +44 -0
- package/src/shell/components/notifications/NotificationInbox.tsx +175 -0
- package/src/shell/components/notifications/ToastProvider.tsx +2 -0
- package/src/shell/components/notifications/index.ts +4 -0
- package/src/shell/components/organization/ImpersonationBanner.tsx +29 -0
- package/src/shell/components/organization/InviteMemberDialog.tsx +161 -0
- package/src/shell/components/organization/OrgInitializer.tsx +179 -0
- package/src/shell/components/organization/OrgSwitcher.tsx +177 -0
- package/src/shell/components/organization/PermissionGate.tsx +16 -0
- package/src/shell/components/organization/PermissionMatrixEditor.tsx +206 -0
- package/src/shell/components/organization/PermissionProfilesTab.tsx +148 -0
- package/src/shell/components/organization/TeamTab.tsx +243 -0
- package/src/shell/components/organization/TenantOnboarding.tsx +179 -0
- package/src/shell/components/organization/index.ts +7 -0
- package/src/shell/components/plugins/PluginSettingsPanel.tsx +189 -0
- package/src/shell/components/settings/BrandingSettings.tsx +255 -0
- package/src/shell/components/settings/CompanySettings.tsx +188 -0
- package/src/shell/components/settings/ConnectedBrandingSettings.tsx +82 -0
- package/src/shell/components/settings/ConnectedCompanySettings.tsx +63 -0
- package/src/shell/components/settings/ConnectedFieldRulesSettings.tsx +47 -0
- package/src/shell/components/settings/ConnectedHolidaysSettings.tsx +99 -0
- package/src/shell/components/settings/ConnectedLocationsOverview.tsx +57 -0
- package/src/shell/components/settings/ConnectedSecuritySettings.tsx +59 -0
- package/src/shell/components/settings/ConnectedUserProfile.tsx +59 -0
- package/src/shell/components/settings/FieldRulesSettings.tsx +511 -0
- package/src/shell/components/settings/HolidaysSettings.tsx +141 -0
- package/src/shell/components/settings/LocationsCrudPage.ts +36 -0
- package/src/shell/components/settings/LocationsOverview.tsx +171 -0
- package/src/shell/components/settings/SecuritySettings.tsx +208 -0
- package/src/shell/components/settings/SettingsPage.tsx +162 -0
- package/src/shell/components/settings/UserProfile.tsx +151 -0
- package/src/shell/components/settings/index.ts +10 -0
- package/src/shell/config/index.ts +6 -0
- package/src/shell/config/permissions.ts +31 -0
- package/src/shell/config/tailwind-preset.ts +164 -0
- package/src/shell/config/theme/dark.ts +63 -0
- package/src/shell/config/theme/index.ts +5 -0
- package/src/shell/config/theme/light.ts +63 -0
- package/src/shell/config/theme/tokens.ts +105 -0
- package/src/shell/config/theme/utils.ts +249 -0
- package/src/shell/createSaasApp.tsx +922 -0
- package/src/shell/hooks/index.ts +18 -0
- package/src/shell/hooks/useAITools.ts +139 -0
- package/src/shell/hooks/useAuth.ts +109 -0
- package/src/shell/hooks/useBilling.ts +151 -0
- package/src/shell/hooks/useChat.ts +84 -0
- package/src/shell/hooks/useFieldRules.ts +19 -0
- package/src/shell/hooks/useFormatters.ts +56 -0
- package/src/shell/hooks/useLayout.ts +76 -0
- package/src/shell/hooks/useModuleNavigation.ts +124 -0
- package/src/shell/hooks/useNotifications.ts +114 -0
- package/src/shell/hooks/usePermission.ts +101 -0
- package/src/shell/hooks/usePluginPrefs.ts +113 -0
- package/src/shell/hooks/usePlugins.ts +18 -0
- package/src/shell/hooks/useTenant.ts +106 -0
- package/src/shell/hooks/useTenantPlugins.ts +103 -0
- package/src/shell/hooks/useTranslation.ts +10 -0
- package/src/shell/lib/api.ts +67 -0
- package/src/shell/lib/apply-field-rules.ts +14 -0
- package/src/shell/lib/auth-adapters/mock.ts +97 -0
- package/src/shell/lib/auth-adapters/supabase.ts +96 -0
- package/src/shell/lib/auth-context.ts +18 -0
- package/src/shell/lib/cache.ts +124 -0
- package/src/shell/lib/cn.ts +1 -0
- package/src/shell/lib/core-ai-tools.ts +90 -0
- package/src/shell/lib/create-client-orders-provider.ts +105 -0
- package/src/shell/lib/csv.ts +54 -0
- package/src/shell/lib/dedup.ts +17 -0
- package/src/shell/lib/entity-registry.ts +39 -0
- package/src/shell/lib/entity-routes.ts +53 -0
- package/src/shell/lib/format.ts +58 -0
- package/src/shell/lib/i18n.ts +3723 -0
- package/src/shell/lib/locale-config.ts +14 -0
- package/src/shell/lib/org-adapters/mock.ts +318 -0
- package/src/shell/lib/org-adapters/supabase.ts +533 -0
- package/src/shell/lib/org-context.ts +18 -0
- package/src/shell/lib/plugins.ts +11 -0
- package/src/shell/lib/router.ts +86 -0
- package/src/shell/lib/schedule-config.ts +42 -0
- package/src/shell/lib/schedule-service.ts +307 -0
- package/src/shell/lib/supabase.ts +19 -0
- package/src/shell/stores/auth.store.ts +29 -0
- package/src/shell/stores/billing.store.ts +1 -0
- package/src/shell/stores/chat.store.ts +45 -0
- package/src/shell/stores/index.ts +12 -0
- package/src/shell/stores/invite.store.ts +18 -0
- package/src/shell/stores/layout.store.ts +1 -0
- package/src/shell/stores/locale.store.ts +24 -0
- package/src/shell/stores/location.store.ts +89 -0
- package/src/shell/stores/notifications.store.ts +48 -0
- package/src/shell/stores/organization.store.ts +3 -0
- package/src/shell/stores/permissions.store.ts +1 -0
- package/src/shell/stores/plugin.store.ts +50 -0
- package/src/shell/stores/tenant.store.ts +26 -0
- package/src/shell/stores/theme.store.ts +162 -0
- package/src/shell/styles.css +433 -0
- package/src/shell/test-setup.ts +1 -0
- package/src/shell/types/auth-adapter.ts +11 -0
- package/src/shell/types/auth.ts +38 -0
- package/src/shell/types/billing.ts +90 -0
- package/src/shell/types/client-orders.ts +58 -0
- package/src/shell/types/crud.ts +134 -0
- package/src/shell/types/entities.ts +169 -0
- package/src/shell/types/entity-lookup.ts +25 -0
- package/src/shell/types/index.ts +16 -0
- package/src/shell/types/integrations.ts +51 -0
- package/src/shell/types/invite.ts +23 -0
- package/src/shell/types/layout.ts +29 -0
- package/src/shell/types/notifications.ts +33 -0
- package/src/shell/types/org-adapter.ts +72 -0
- package/src/shell/types/permissions.ts +64 -0
- package/src/shell/types/platform.ts +25 -0
- package/src/shell/types/plugins.ts +359 -0
- package/src/shell/types/tenant.ts +79 -0
- package/src/stores/createCrudStore.ts +138 -0
- package/src/supabase/client.ts +54 -0
|
@@ -0,0 +1,467 @@
|
|
|
1
|
+
import React, { useState, useEffect } from 'react'
|
|
2
|
+
import { ArrowLeft, ImagePlus, Loader2 } from 'lucide-react'
|
|
3
|
+
import { Card, CardContent } from '@fayz-ai/ui'
|
|
4
|
+
import { Button } from '@fayz-ai/ui'
|
|
5
|
+
import { Input } from '@fayz-ai/ui'
|
|
6
|
+
import { Checkbox } from '@fayz-ai/ui'
|
|
7
|
+
import { CurrencyInput } from '@fayz-ai/ui'
|
|
8
|
+
import { toast } from '@fayz-ai/ui'
|
|
9
|
+
import { useSaveBar, useBackHandler } from '@fayz-ai/ui'
|
|
10
|
+
import { PersonFormLayout } from './archetypes/PersonFormLayout'
|
|
11
|
+
import { ProductFormLayout } from './archetypes/ProductFormLayout'
|
|
12
|
+
import { ServiceFormLayout } from './archetypes/ServiceFormLayout'
|
|
13
|
+
import { LocationFormLayout } from './archetypes/LocationFormLayout'
|
|
14
|
+
import { SubjectFormLayout } from './archetypes/SubjectFormLayout'
|
|
15
|
+
import { useTranslation } from '@fayz-ai/core'
|
|
16
|
+
import type { FieldDef, FieldGroup, EntityDef } from '@fayz-ai/core'
|
|
17
|
+
import type { FormLayout } from '@fayz-ai/core'
|
|
18
|
+
|
|
19
|
+
interface CrudFormPageProps {
|
|
20
|
+
entityDef: EntityDef
|
|
21
|
+
mode: 'create' | 'edit'
|
|
22
|
+
initialData?: Record<string, any>
|
|
23
|
+
onSubmit: (data: Record<string, any>) => void | Promise<void>
|
|
24
|
+
onCancel: () => void
|
|
25
|
+
namePlural: string
|
|
26
|
+
/** Embedded mode — no breadcrumb, compact layout. For use inside modals/panels. */
|
|
27
|
+
embedded?: boolean
|
|
28
|
+
/** Hide the internal breadcrumb while keeping the full-page SaveBar/Escape flow.
|
|
29
|
+
* Use when a parent already renders its own header (e.g. SubpageHeader). */
|
|
30
|
+
hideBreadcrumb?: boolean
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/** Derive a currency symbol from an ISO code (e.g. USD → "$") when not given. */
|
|
34
|
+
function currencySymbolFor(code: string | undefined, locale: string): string {
|
|
35
|
+
if (!code) return 'R$'
|
|
36
|
+
try {
|
|
37
|
+
const parts = new Intl.NumberFormat(locale, { style: 'currency', currency: code }).formatToParts(0)
|
|
38
|
+
return parts.find((p) => p.type === 'currency')?.value ?? code
|
|
39
|
+
} catch {
|
|
40
|
+
return code
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
function getDefaultValues(fields: FieldDef[]): Record<string, any> {
|
|
45
|
+
const values: Record<string, any> = {}
|
|
46
|
+
for (const field of fields) {
|
|
47
|
+
if (field.showInForm === false) continue
|
|
48
|
+
// Computed fields are read-only and never part of the payload.
|
|
49
|
+
if (field.type === 'computed') continue
|
|
50
|
+
if (field.defaultValue != null) {
|
|
51
|
+
values[field.key] = field.defaultValue
|
|
52
|
+
} else {
|
|
53
|
+
switch (field.type) {
|
|
54
|
+
case 'boolean':
|
|
55
|
+
values[field.key] = false
|
|
56
|
+
break
|
|
57
|
+
case 'number':
|
|
58
|
+
case 'currency':
|
|
59
|
+
values[field.key] = ''
|
|
60
|
+
break
|
|
61
|
+
default:
|
|
62
|
+
values[field.key] = ''
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
return values
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
function renderField(field: FieldDef, value: any, onChange: (val: any) => void, allValues: Record<string, any> = {}) {
|
|
70
|
+
const baseClass = 'flex h-9 w-full rounded-input border border-input bg-background px-3 py-1.5 text-sm shadow-sm ring-offset-background placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50'
|
|
71
|
+
|
|
72
|
+
switch (field.type) {
|
|
73
|
+
case 'currency': {
|
|
74
|
+
const locale = field.currencyLocale ?? 'pt-BR'
|
|
75
|
+
return (
|
|
76
|
+
<CurrencyInput
|
|
77
|
+
value={Number(value) || 0}
|
|
78
|
+
onChange={(n) => onChange(n)}
|
|
79
|
+
currencyCode={field.currency}
|
|
80
|
+
locale={locale}
|
|
81
|
+
symbol={field.currencySymbol ?? currencySymbolFor(field.currency, locale)}
|
|
82
|
+
/>
|
|
83
|
+
)
|
|
84
|
+
}
|
|
85
|
+
case 'segmented': {
|
|
86
|
+
const options = (field.options ?? []).map((o) =>
|
|
87
|
+
typeof o === 'string' ? { label: o, value: o, description: undefined } : o,
|
|
88
|
+
)
|
|
89
|
+
return (
|
|
90
|
+
<div className="grid gap-2 sm:grid-cols-2 lg:grid-cols-4">
|
|
91
|
+
{options.map((o) => {
|
|
92
|
+
const active = value === o.value
|
|
93
|
+
return (
|
|
94
|
+
<button
|
|
95
|
+
key={o.value}
|
|
96
|
+
type="button"
|
|
97
|
+
onClick={() => onChange(o.value)}
|
|
98
|
+
className={`rounded-lg border-2 p-3 text-left transition-all ${
|
|
99
|
+
active
|
|
100
|
+
? 'border-primary bg-primary/5'
|
|
101
|
+
: 'border-transparent bg-muted/20 hover:bg-muted/40'
|
|
102
|
+
}`}
|
|
103
|
+
>
|
|
104
|
+
<p className={`text-sm font-medium ${active ? 'text-primary' : ''}`}>{o.label}</p>
|
|
105
|
+
{'description' in o && o.description && (
|
|
106
|
+
<p className="text-[10px] text-muted-foreground mt-0.5 leading-tight">{o.description}</p>
|
|
107
|
+
)}
|
|
108
|
+
</button>
|
|
109
|
+
)
|
|
110
|
+
})}
|
|
111
|
+
</div>
|
|
112
|
+
)
|
|
113
|
+
}
|
|
114
|
+
case 'computed': {
|
|
115
|
+
const result = field.compute ? field.compute(allValues) : null
|
|
116
|
+
const tone = result?.tone === 'positive' ? 'text-success'
|
|
117
|
+
: result?.tone === 'negative' ? 'text-destructive'
|
|
118
|
+
: 'text-foreground'
|
|
119
|
+
return (
|
|
120
|
+
<div className="mt-1 rounded-lg border bg-muted/20 px-3 py-2 text-sm tabular-nums text-right">
|
|
121
|
+
{result ? <span className={tone}>{result.display}</span> : <span className="text-muted-foreground">—</span>}
|
|
122
|
+
</div>
|
|
123
|
+
)
|
|
124
|
+
}
|
|
125
|
+
case 'textarea':
|
|
126
|
+
return (
|
|
127
|
+
<textarea
|
|
128
|
+
value={value ?? ''}
|
|
129
|
+
onChange={(e) => onChange(e.target.value)}
|
|
130
|
+
placeholder={field.placeholder}
|
|
131
|
+
required={field.required}
|
|
132
|
+
rows={2}
|
|
133
|
+
className={`${baseClass} min-h-[60px] py-1.5`}
|
|
134
|
+
/>
|
|
135
|
+
)
|
|
136
|
+
case 'select': {
|
|
137
|
+
const options = (field.options ?? []).map((o) =>
|
|
138
|
+
typeof o === 'string' ? { label: o, value: o } : o,
|
|
139
|
+
)
|
|
140
|
+
return (
|
|
141
|
+
<select
|
|
142
|
+
value={value ?? ''}
|
|
143
|
+
onChange={(e) => onChange(e.target.value)}
|
|
144
|
+
required={field.required}
|
|
145
|
+
className={baseClass}
|
|
146
|
+
>
|
|
147
|
+
<option value="">{field.placeholder ?? `Select ${field.label.toLowerCase()}...`}</option>
|
|
148
|
+
{options.map((o) => (
|
|
149
|
+
<option key={o.value} value={o.value}>{o.label}</option>
|
|
150
|
+
))}
|
|
151
|
+
</select>
|
|
152
|
+
)
|
|
153
|
+
}
|
|
154
|
+
case 'boolean':
|
|
155
|
+
return (
|
|
156
|
+
<label className="flex items-center gap-2.5 cursor-pointer">
|
|
157
|
+
<Checkbox
|
|
158
|
+
checked={!!value}
|
|
159
|
+
onChange={(checked) => onChange(checked)}
|
|
160
|
+
/>
|
|
161
|
+
<span className="text-sm">{field.label}</span>
|
|
162
|
+
</label>
|
|
163
|
+
)
|
|
164
|
+
case 'number':
|
|
165
|
+
return (
|
|
166
|
+
<Input
|
|
167
|
+
type="number"
|
|
168
|
+
value={value ?? ''}
|
|
169
|
+
onChange={(e) => onChange(e.target.value === '' ? '' : Number(e.target.value))}
|
|
170
|
+
placeholder={field.placeholder}
|
|
171
|
+
required={field.required}
|
|
172
|
+
min={field.min}
|
|
173
|
+
max={field.max}
|
|
174
|
+
/>
|
|
175
|
+
)
|
|
176
|
+
case 'date':
|
|
177
|
+
return <Input type="date" value={value ?? ''} onChange={(e) => onChange(e.target.value)} required={field.required} />
|
|
178
|
+
case 'datetime':
|
|
179
|
+
return <Input type="datetime-local" value={value ?? ''} onChange={(e) => onChange(e.target.value)} required={field.required} />
|
|
180
|
+
case 'time':
|
|
181
|
+
return <Input type="time" value={value ?? ''} onChange={(e) => onChange(e.target.value)} required={field.required} />
|
|
182
|
+
case 'color':
|
|
183
|
+
return (
|
|
184
|
+
<div className="flex items-center gap-2">
|
|
185
|
+
<input
|
|
186
|
+
type="color"
|
|
187
|
+
value={value || '#6b7280'}
|
|
188
|
+
onChange={(e) => onChange(e.target.value)}
|
|
189
|
+
className="h-9 w-12 rounded-md border border-input cursor-pointer p-0.5"
|
|
190
|
+
/>
|
|
191
|
+
<Input
|
|
192
|
+
type="text"
|
|
193
|
+
value={value ?? ''}
|
|
194
|
+
onChange={(e) => onChange(e.target.value)}
|
|
195
|
+
placeholder="#000000"
|
|
196
|
+
className="flex-1 font-mono text-xs"
|
|
197
|
+
/>
|
|
198
|
+
</div>
|
|
199
|
+
)
|
|
200
|
+
default:
|
|
201
|
+
return (
|
|
202
|
+
<Input
|
|
203
|
+
type={field.type === 'email' ? 'email' : field.type === 'phone' ? 'tel' : field.type === 'url' ? 'url' : 'text'}
|
|
204
|
+
value={value ?? ''}
|
|
205
|
+
onChange={(e) => onChange(e.target.value)}
|
|
206
|
+
placeholder={field.placeholder}
|
|
207
|
+
required={field.required}
|
|
208
|
+
/>
|
|
209
|
+
)
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
function FormFieldItem({ field, value, onChange, allValues }: { field: FieldDef; value: any; onChange: (val: any) => void; allValues?: Record<string, any> }) {
|
|
214
|
+
return (
|
|
215
|
+
<div className={`grid gap-1 ${field.span === 2 ? 'md:col-span-2' : ''}`}>
|
|
216
|
+
{field.type !== 'boolean' && (
|
|
217
|
+
<label className="text-sm font-medium text-foreground">
|
|
218
|
+
{field.label}
|
|
219
|
+
{field.required && <span className="text-destructive ml-0.5">*</span>}
|
|
220
|
+
</label>
|
|
221
|
+
)}
|
|
222
|
+
{renderField(field, value, onChange, allValues)}
|
|
223
|
+
{field.hint && <p className="text-[10px] text-muted-foreground mt-0.5">{field.hint}</p>}
|
|
224
|
+
</div>
|
|
225
|
+
)
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
function FormGroup({
|
|
229
|
+
group,
|
|
230
|
+
fields,
|
|
231
|
+
values,
|
|
232
|
+
onChange,
|
|
233
|
+
}: {
|
|
234
|
+
group: FieldGroup
|
|
235
|
+
fields: FieldDef[]
|
|
236
|
+
values: Record<string, any>
|
|
237
|
+
onChange: (key: string, val: any) => void
|
|
238
|
+
}) {
|
|
239
|
+
const cols = group.columns ?? 2
|
|
240
|
+
|
|
241
|
+
const grid = (
|
|
242
|
+
<div className={`grid gap-3 ${cols >= 2 ? 'md:grid-cols-2' : ''} ${cols >= 3 ? 'lg:grid-cols-3' : ''}`}>
|
|
243
|
+
{fields.map((field) => (
|
|
244
|
+
<FormFieldItem
|
|
245
|
+
key={field.key}
|
|
246
|
+
field={field}
|
|
247
|
+
value={values[field.key]}
|
|
248
|
+
onChange={(val) => onChange(field.key, val)}
|
|
249
|
+
allValues={values}
|
|
250
|
+
/>
|
|
251
|
+
))}
|
|
252
|
+
</div>
|
|
253
|
+
)
|
|
254
|
+
|
|
255
|
+
return (
|
|
256
|
+
<div>
|
|
257
|
+
<h3 className="text-sm font-semibold text-foreground">{group.label}</h3>
|
|
258
|
+
{group.description && (
|
|
259
|
+
<p className="text-xs text-muted-foreground mt-0.5 mb-3">{group.description}</p>
|
|
260
|
+
)}
|
|
261
|
+
<Card>
|
|
262
|
+
<CardContent className="pt-4">
|
|
263
|
+
{group.imageSlot ? (
|
|
264
|
+
<div className="flex gap-4">
|
|
265
|
+
{/* Decorative image slot — non-functional placeholder for now. */}
|
|
266
|
+
<div className="shrink-0">
|
|
267
|
+
<div className="flex h-20 w-20 cursor-pointer items-center justify-center rounded-lg border-2 border-dashed border-muted text-muted-foreground transition-colors hover:border-primary/30 hover:text-primary/50">
|
|
268
|
+
<ImagePlus className="h-5 w-5" />
|
|
269
|
+
</div>
|
|
270
|
+
</div>
|
|
271
|
+
<div className="flex-1">{grid}</div>
|
|
272
|
+
</div>
|
|
273
|
+
) : (
|
|
274
|
+
grid
|
|
275
|
+
)}
|
|
276
|
+
</CardContent>
|
|
277
|
+
</Card>
|
|
278
|
+
</div>
|
|
279
|
+
)
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
export function CrudFormPage({ entityDef, mode, initialData, onSubmit, onCancel, namePlural, embedded, hideBreadcrumb }: CrudFormPageProps) {
|
|
283
|
+
const t = useTranslation()
|
|
284
|
+
const formFields = entityDef.fields.filter((f) => f.showInForm !== false)
|
|
285
|
+
const displayField = entityDef.displayField ?? entityDef.fields[0]?.key ?? 'id'
|
|
286
|
+
const computeInit = () =>
|
|
287
|
+
mode === 'edit' && initialData ? { ...getDefaultValues(formFields), ...initialData } : getDefaultValues(formFields)
|
|
288
|
+
const [values, setValues] = useState<Record<string, any>>(computeInit)
|
|
289
|
+
const [saving, setSaving] = useState(false)
|
|
290
|
+
const initialRef = React.useRef<string>(JSON.stringify(values))
|
|
291
|
+
|
|
292
|
+
useEffect(() => {
|
|
293
|
+
const init = computeInit()
|
|
294
|
+
setValues(init)
|
|
295
|
+
initialRef.current = JSON.stringify(init)
|
|
296
|
+
}, [mode, initialData])
|
|
297
|
+
|
|
298
|
+
const computedKeys = new Set(formFields.filter((f) => f.type === 'computed').map((f) => f.key))
|
|
299
|
+
|
|
300
|
+
const submit = async () => {
|
|
301
|
+
setSaving(true)
|
|
302
|
+
try {
|
|
303
|
+
// Sanitize: convert empty strings to null so the DB doesn't choke on e.g. empty date fields.
|
|
304
|
+
// Computed fields are read-only and never persisted.
|
|
305
|
+
const sanitized: Record<string, any> = {}
|
|
306
|
+
for (const [key, val] of Object.entries(values)) {
|
|
307
|
+
if (computedKeys.has(key)) continue
|
|
308
|
+
sanitized[key] = val === '' ? null : val
|
|
309
|
+
}
|
|
310
|
+
await onSubmit(sanitized)
|
|
311
|
+
} catch (err: any) {
|
|
312
|
+
const message = err?.message || 'Something went wrong'
|
|
313
|
+
toast.error(t('crud.form.failedToSave', { entity: entityDef.name.toLowerCase() }), { description: message })
|
|
314
|
+
} finally {
|
|
315
|
+
setSaving(false)
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
const handleSubmit = (e: React.FormEvent) => {
|
|
320
|
+
e.preventDefault()
|
|
321
|
+
void submit()
|
|
322
|
+
}
|
|
323
|
+
|
|
324
|
+
const dirty = JSON.stringify(values) !== initialRef.current
|
|
325
|
+
|
|
326
|
+
// Discard reverts the form to its pristine state and stays on the page (the
|
|
327
|
+
// SaveBar then hides). Leaving the page is a separate action — the breadcrumb
|
|
328
|
+
// link, or pressing Escape again once there's nothing left to discard.
|
|
329
|
+
const discard = () => { setValues(computeInit()) }
|
|
330
|
+
|
|
331
|
+
// Full-page CRUD forms surface Save/Discard via the app-wide floating SaveBar.
|
|
332
|
+
// Embedded forms (rendered inside other widgets) keep their inline buttons.
|
|
333
|
+
useSaveBar({
|
|
334
|
+
dirty: !embedded && dirty,
|
|
335
|
+
saving,
|
|
336
|
+
onSave: () => { void submit() },
|
|
337
|
+
onDiscard: discard,
|
|
338
|
+
saveLabel: mode === 'create' ? t('crud.form.addTitle', { entity: entityDef.name }) : t('crud.form.saveChanges'),
|
|
339
|
+
})
|
|
340
|
+
|
|
341
|
+
// Register "leave to parent" with the app-wide Escape key. While the form is
|
|
342
|
+
// dirty the SaveBar owns Escape (→ discard); once pristine, Escape navigates
|
|
343
|
+
// back — giving the two-step "discard, then leave" flow for free.
|
|
344
|
+
useBackHandler(embedded ? undefined : onCancel)
|
|
345
|
+
|
|
346
|
+
const handleChange = (key: string, val: any) => {
|
|
347
|
+
setValues((prev) => ({ ...prev, [key]: val }))
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
const title = mode === 'create' ? t('crud.form.addTitle', { entity: entityDef.name }) : t('crud.form.editTitle', { entity: entityDef.name })
|
|
351
|
+
const breadcrumbLabel = mode === 'create'
|
|
352
|
+
? t('crud.form.newBreadcrumb', { entity: entityDef.name })
|
|
353
|
+
: (values[displayField] || initialData?.[displayField] || t('common.edit'))
|
|
354
|
+
|
|
355
|
+
// Organize fields by groups
|
|
356
|
+
const groups = entityDef.fieldGroups ?? []
|
|
357
|
+
const groupFieldMap = new Map<string, FieldDef[]>()
|
|
358
|
+
const ungroupedFields: FieldDef[] = []
|
|
359
|
+
|
|
360
|
+
for (const field of formFields) {
|
|
361
|
+
if (field.group) {
|
|
362
|
+
const existing = groupFieldMap.get(field.group) ?? []
|
|
363
|
+
existing.push(field)
|
|
364
|
+
groupFieldMap.set(field.group, existing)
|
|
365
|
+
} else {
|
|
366
|
+
ungroupedFields.push(field)
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
|
|
370
|
+
const hasGroups = groups.length > 0 || groupFieldMap.size > 0
|
|
371
|
+
|
|
372
|
+
const archetypeLayoutProps = {
|
|
373
|
+
fields: formFields,
|
|
374
|
+
allFields: entityDef.fields,
|
|
375
|
+
fieldGroups: groups,
|
|
376
|
+
values,
|
|
377
|
+
onChange: handleChange,
|
|
378
|
+
renderField,
|
|
379
|
+
entityIcon: entityDef.icon,
|
|
380
|
+
compact: embedded,
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
function renderFormBody(layout?: FormLayout) {
|
|
384
|
+
switch (layout) {
|
|
385
|
+
case 'person':
|
|
386
|
+
return <PersonFormLayout {...archetypeLayoutProps} />
|
|
387
|
+
case 'product':
|
|
388
|
+
return <ProductFormLayout {...archetypeLayoutProps} />
|
|
389
|
+
case 'service':
|
|
390
|
+
return <ServiceFormLayout {...archetypeLayoutProps} />
|
|
391
|
+
case 'location':
|
|
392
|
+
return <LocationFormLayout {...archetypeLayoutProps} />
|
|
393
|
+
case 'subject':
|
|
394
|
+
return <SubjectFormLayout {...archetypeLayoutProps} />
|
|
395
|
+
default:
|
|
396
|
+
// Generic layout — ungrouped + grouped fields
|
|
397
|
+
return (
|
|
398
|
+
<>
|
|
399
|
+
{ungroupedFields.length > 0 && (
|
|
400
|
+
<Card>
|
|
401
|
+
<CardContent className="pt-5">
|
|
402
|
+
<div className={`grid gap-4 ${!hasGroups ? '' : 'md:grid-cols-2'}`}>
|
|
403
|
+
{ungroupedFields.map((field) => (
|
|
404
|
+
<FormFieldItem
|
|
405
|
+
key={field.key}
|
|
406
|
+
field={field}
|
|
407
|
+
value={values[field.key]}
|
|
408
|
+
onChange={(val) => handleChange(field.key, val)}
|
|
409
|
+
allValues={values}
|
|
410
|
+
/>
|
|
411
|
+
))}
|
|
412
|
+
</div>
|
|
413
|
+
</CardContent>
|
|
414
|
+
</Card>
|
|
415
|
+
)}
|
|
416
|
+
{groups.map((group) => {
|
|
417
|
+
const fields = groupFieldMap.get(group.id)
|
|
418
|
+
if (!fields || fields.length === 0) return null
|
|
419
|
+
return (
|
|
420
|
+
<FormGroup
|
|
421
|
+
key={group.id}
|
|
422
|
+
group={group}
|
|
423
|
+
fields={fields}
|
|
424
|
+
values={values}
|
|
425
|
+
onChange={handleChange}
|
|
426
|
+
/>
|
|
427
|
+
)
|
|
428
|
+
})}
|
|
429
|
+
</>
|
|
430
|
+
)
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
|
|
434
|
+
return (
|
|
435
|
+
<div className={`w-full flex flex-col items-center ${embedded ? 'text-sm' : ''}`}>
|
|
436
|
+
<div className={`w-full ${embedded ? 'space-y-3' : 'max-w-3xl space-y-5'}`}>
|
|
437
|
+
{/* Breadcrumb + subtitle — hidden in embedded mode or when a parent owns the header */}
|
|
438
|
+
{!embedded && !hideBreadcrumb && (
|
|
439
|
+
<nav className="flex items-center gap-1.5 text-sm text-muted-foreground">
|
|
440
|
+
<button type="button" onClick={onCancel} className="inline-flex items-center gap-1 hover:text-foreground transition-colors">
|
|
441
|
+
<ArrowLeft className="h-3.5 w-3.5" />
|
|
442
|
+
{namePlural}
|
|
443
|
+
</button>
|
|
444
|
+
<span>/</span>
|
|
445
|
+
<span className="text-foreground font-medium truncate max-w-[200px]">{breadcrumbLabel}</span>
|
|
446
|
+
</nav>
|
|
447
|
+
)}
|
|
448
|
+
|
|
449
|
+
{/* Form */}
|
|
450
|
+
<form onSubmit={handleSubmit} className={embedded ? 'space-y-3' : 'space-y-5'}>
|
|
451
|
+
{renderFormBody(entityDef.layout)}
|
|
452
|
+
|
|
453
|
+
{/* Submit — embedded forms keep inline buttons; full-page forms use the floating SaveBar */}
|
|
454
|
+
{embedded && (
|
|
455
|
+
<div className="flex items-center justify-end gap-2 pt-1">
|
|
456
|
+
<Button type="button" variant="outline" size="sm" onClick={onCancel} disabled={saving}>{t('common.cancel')}</Button>
|
|
457
|
+
<Button type="submit" size="sm" disabled={saving}>
|
|
458
|
+
{saving && <Loader2 className="animate-spin mr-1.5 h-3 w-3" />}
|
|
459
|
+
{saving ? t('common.saving') : mode === 'create' ? t('crud.form.addTitle', { entity: entityDef.name }) : t('crud.form.saveChanges')}
|
|
460
|
+
</Button>
|
|
461
|
+
</div>
|
|
462
|
+
)}
|
|
463
|
+
</form>
|
|
464
|
+
</div>
|
|
465
|
+
</div>
|
|
466
|
+
)
|
|
467
|
+
}
|
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
import React, { useMemo } from 'react'
|
|
2
|
+
import { MoreVertical, Upload, Download } from 'lucide-react'
|
|
3
|
+
import type { ColumnDef as TanStackColumnDef } from '@tanstack/react-table'
|
|
4
|
+
import { useTranslation } from '@fayz-ai/core'
|
|
5
|
+
import type { EntityDef } from '@fayz-ai/core'
|
|
6
|
+
import { Card } from '@fayz-ai/ui'
|
|
7
|
+
import { Button } from '@fayz-ai/ui'
|
|
8
|
+
import { DataTable } from '@fayz-ai/ui'
|
|
9
|
+
import { Dropdown, DropdownTrigger, DropdownContent, DropdownItem } from '@fayz-ai/ui'
|
|
10
|
+
import { fieldToColumns } from './fieldToColumn'
|
|
11
|
+
import { CrudCardGrid } from './CrudCardGrid'
|
|
12
|
+
import { PermissionGate } from '../permissions/PermissionGate'
|
|
13
|
+
|
|
14
|
+
/** A resolved facet: a field plus the option pills to render. */
|
|
15
|
+
export interface CrudFacet {
|
|
16
|
+
field: string
|
|
17
|
+
options: { value: string; label: string }[]
|
|
18
|
+
allLabel?: string
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface CrudListViewProps<T extends { id: string }> {
|
|
22
|
+
entityDef: EntityDef<T>
|
|
23
|
+
/** TanStack columns. When omitted, derived from entityDef.fields (no actions column). */
|
|
24
|
+
columns?: TanStackColumnDef<T, any>[]
|
|
25
|
+
/** null = initial load (skeleton); [] = empty state. */
|
|
26
|
+
items: T[] | null
|
|
27
|
+
total: number
|
|
28
|
+
display?: 'table' | 'cards'
|
|
29
|
+
/** Search box — rendered when onSearchChange is provided. */
|
|
30
|
+
search?: string
|
|
31
|
+
onSearchChange?: (value: string) => void
|
|
32
|
+
searchPlaceholder?: string
|
|
33
|
+
/** Facet pills below the search box. */
|
|
34
|
+
facets?: CrudFacet[]
|
|
35
|
+
activeFilters?: Record<string, string | undefined>
|
|
36
|
+
onFacetChange?: (field: string, value: string | undefined) => void
|
|
37
|
+
onNew?: () => void
|
|
38
|
+
addLabel?: string
|
|
39
|
+
onRowClick?: (row: T) => void
|
|
40
|
+
/** Delete action for the card grid (table actions live in the columns). */
|
|
41
|
+
onDelete?: (row: T) => void
|
|
42
|
+
onImport?: () => void
|
|
43
|
+
onExport?: () => void
|
|
44
|
+
feature?: string
|
|
45
|
+
readOnly?: boolean
|
|
46
|
+
emptyMessage?: string
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
function pillClass(active: boolean): string {
|
|
50
|
+
return [
|
|
51
|
+
'rounded-full px-3 py-1 text-sm font-medium transition-colors',
|
|
52
|
+
active
|
|
53
|
+
? 'bg-foreground text-background'
|
|
54
|
+
: 'bg-muted text-muted-foreground hover:bg-muted/70',
|
|
55
|
+
].join(' ')
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* The canonical generic list — header (count + New + import/export), search,
|
|
60
|
+
* faceted filter pills, and a DataTable/card grid. Decoupled from data and
|
|
61
|
+
* routing: feed it `items` and callbacks. Used by CrudPage (store-backed) and by
|
|
62
|
+
* plugins that own their data/routing (e.g. the inventory product list).
|
|
63
|
+
*/
|
|
64
|
+
export function CrudListView<T extends { id: string }>({
|
|
65
|
+
entityDef,
|
|
66
|
+
columns,
|
|
67
|
+
items,
|
|
68
|
+
total,
|
|
69
|
+
display = 'table',
|
|
70
|
+
search,
|
|
71
|
+
onSearchChange,
|
|
72
|
+
searchPlaceholder,
|
|
73
|
+
facets,
|
|
74
|
+
activeFilters,
|
|
75
|
+
onFacetChange,
|
|
76
|
+
onNew,
|
|
77
|
+
addLabel,
|
|
78
|
+
onRowClick,
|
|
79
|
+
onDelete,
|
|
80
|
+
onImport,
|
|
81
|
+
onExport,
|
|
82
|
+
feature,
|
|
83
|
+
readOnly,
|
|
84
|
+
emptyMessage,
|
|
85
|
+
}: CrudListViewProps<T>) {
|
|
86
|
+
const t = useTranslation()
|
|
87
|
+
const namePlural = entityDef.namePlural ?? entityDef.name + 's'
|
|
88
|
+
const displayField = entityDef.displayField ?? entityDef.fields[0]?.key ?? 'id'
|
|
89
|
+
|
|
90
|
+
const derivedColumns = useMemo<TanStackColumnDef<T, any>[]>(() => {
|
|
91
|
+
return fieldToColumns(entityDef.fields).map((col) => ({
|
|
92
|
+
accessorKey: col.key,
|
|
93
|
+
header: col.label,
|
|
94
|
+
enableSorting: col.sortable,
|
|
95
|
+
cell: ({ row }: any) => {
|
|
96
|
+
const value = row.original[col.key]
|
|
97
|
+
const rendered = col.render(value, row.original)
|
|
98
|
+
return col.key === displayField
|
|
99
|
+
? <span className="font-medium text-foreground">{rendered}</span>
|
|
100
|
+
: rendered
|
|
101
|
+
},
|
|
102
|
+
}))
|
|
103
|
+
}, [entityDef.fields, displayField])
|
|
104
|
+
|
|
105
|
+
const tanColumns = columns ?? derivedColumns
|
|
106
|
+
|
|
107
|
+
const isInitialLoad = items === null
|
|
108
|
+
const isEmpty = items !== null && items.length === 0
|
|
109
|
+
const hasImportExport = Boolean(onImport || onExport)
|
|
110
|
+
|
|
111
|
+
const newButton = onNew && !readOnly
|
|
112
|
+
? <Button onClick={onNew}>{addLabel ?? t('crud.list.addEntity', { entity: entityDef.name })}</Button>
|
|
113
|
+
: null
|
|
114
|
+
|
|
115
|
+
return (
|
|
116
|
+
<div className="space-y-6">
|
|
117
|
+
<div className="flex items-center justify-between">
|
|
118
|
+
<div>
|
|
119
|
+
<h1 className="text-2xl font-bold">{namePlural}</h1>
|
|
120
|
+
{isInitialLoad ? (
|
|
121
|
+
<div className="h-5 w-32 animate-pulse rounded bg-muted mt-1" />
|
|
122
|
+
) : (
|
|
123
|
+
<p className="text-muted-foreground">{t('crud.list.totalCount', { count: String(total), entities: namePlural.toLowerCase() })}</p>
|
|
124
|
+
)}
|
|
125
|
+
</div>
|
|
126
|
+
<div className="flex items-center gap-2">
|
|
127
|
+
{newButton && (feature ? <PermissionGate feature={feature} action="create">{newButton}</PermissionGate> : newButton)}
|
|
128
|
+
{hasImportExport && (
|
|
129
|
+
<Dropdown>
|
|
130
|
+
<DropdownTrigger asChild>
|
|
131
|
+
<Button variant="outline" size="icon" className="h-9 w-9 shrink-0">
|
|
132
|
+
<MoreVertical className="h-4 w-4" />
|
|
133
|
+
</Button>
|
|
134
|
+
</DropdownTrigger>
|
|
135
|
+
<DropdownContent align="end">
|
|
136
|
+
{onImport && (
|
|
137
|
+
<DropdownItem onClick={onImport}>
|
|
138
|
+
<Upload className="h-4 w-4 mr-2" />
|
|
139
|
+
{t('crud.import.action')}
|
|
140
|
+
</DropdownItem>
|
|
141
|
+
)}
|
|
142
|
+
{onExport && (
|
|
143
|
+
<DropdownItem onClick={onExport}>
|
|
144
|
+
<Download className="h-4 w-4 mr-2" />
|
|
145
|
+
{t('crud.export.action')}
|
|
146
|
+
</DropdownItem>
|
|
147
|
+
)}
|
|
148
|
+
</DropdownContent>
|
|
149
|
+
</Dropdown>
|
|
150
|
+
)}
|
|
151
|
+
</div>
|
|
152
|
+
</div>
|
|
153
|
+
|
|
154
|
+
{onSearchChange && (
|
|
155
|
+
<div className="relative max-w-sm">
|
|
156
|
+
<svg className="absolute left-2.5 top-1/2 -translate-y-1/2 h-3.5 w-3.5 text-muted-foreground" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round"><circle cx="11" cy="11" r="8"/><path d="m21 21-4.3-4.3"/></svg>
|
|
157
|
+
<input
|
|
158
|
+
type="text"
|
|
159
|
+
placeholder={searchPlaceholder ?? t('crud.list.search', { entities: namePlural.toLowerCase() })}
|
|
160
|
+
value={search ?? ''}
|
|
161
|
+
onChange={(e) => onSearchChange(e.target.value)}
|
|
162
|
+
className="w-full rounded-input border border-input bg-card shadow-[inset_0_1px_0_rgb(0_0_0_/0.06)] pl-8 pr-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-ring"
|
|
163
|
+
/>
|
|
164
|
+
</div>
|
|
165
|
+
)}
|
|
166
|
+
|
|
167
|
+
{facets?.map((facet) => {
|
|
168
|
+
const active = activeFilters?.[facet.field]
|
|
169
|
+
return (
|
|
170
|
+
<div key={facet.field} className="flex flex-wrap gap-2">
|
|
171
|
+
<button type="button" onClick={() => onFacetChange?.(facet.field, undefined)} className={pillClass(active == null)}>
|
|
172
|
+
{facet.allLabel ?? t('crud.list.allFacet')}
|
|
173
|
+
</button>
|
|
174
|
+
{facet.options.map((o) => (
|
|
175
|
+
<button key={o.value} type="button" onClick={() => onFacetChange?.(facet.field, o.value)} className={pillClass(active === o.value)}>
|
|
176
|
+
{o.label}
|
|
177
|
+
</button>
|
|
178
|
+
))}
|
|
179
|
+
</div>
|
|
180
|
+
)
|
|
181
|
+
})}
|
|
182
|
+
|
|
183
|
+
{isInitialLoad ? (
|
|
184
|
+
display === 'cards' ? (
|
|
185
|
+
<div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-3">
|
|
186
|
+
{Array.from({ length: 2 }).map((_, i) => (
|
|
187
|
+
<Card key={i} className="overflow-hidden">
|
|
188
|
+
<div className="p-5 space-y-3">
|
|
189
|
+
<div className="h-5 w-2/3 animate-pulse rounded bg-muted" />
|
|
190
|
+
<div className="space-y-2">
|
|
191
|
+
<div className="flex justify-between"><div className="h-4 w-20 animate-pulse rounded bg-muted" /><div className="h-4 w-16 animate-pulse rounded bg-muted" /></div>
|
|
192
|
+
<div className="flex justify-between"><div className="h-4 w-24 animate-pulse rounded bg-muted" /><div className="h-4 w-12 animate-pulse rounded bg-muted" /></div>
|
|
193
|
+
</div>
|
|
194
|
+
</div>
|
|
195
|
+
</Card>
|
|
196
|
+
))}
|
|
197
|
+
</div>
|
|
198
|
+
) : (
|
|
199
|
+
<DataTable columns={tanColumns} data={[]} loading skeletonRows={3} />
|
|
200
|
+
)
|
|
201
|
+
) : isEmpty ? (
|
|
202
|
+
<DataTable
|
|
203
|
+
columns={tanColumns}
|
|
204
|
+
data={[]}
|
|
205
|
+
emptyMessage={emptyMessage ?? t('crud.list.empty', { entities: namePlural.toLowerCase() })}
|
|
206
|
+
/>
|
|
207
|
+
) : display === 'cards' ? (
|
|
208
|
+
<CrudCardGrid
|
|
209
|
+
items={items ?? []}
|
|
210
|
+
entityDef={entityDef as EntityDef<any>}
|
|
211
|
+
onEdit={(item) => onRowClick?.(item as T)}
|
|
212
|
+
onDelete={(item) => onDelete?.(item as T)}
|
|
213
|
+
/>
|
|
214
|
+
) : (
|
|
215
|
+
<DataTable
|
|
216
|
+
columns={tanColumns}
|
|
217
|
+
data={items ?? []}
|
|
218
|
+
onRowClick={(row) => onRowClick?.(row as T)}
|
|
219
|
+
/>
|
|
220
|
+
)}
|
|
221
|
+
</div>
|
|
222
|
+
)
|
|
223
|
+
}
|