@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,108 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Calendar, AlertCircle } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { Button } from '@fayz-ai/ui'
|
|
5
|
+
import { Badge } from '@fayz-ai/ui'
|
|
6
|
+
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@fayz-ai/ui'
|
|
7
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
8
|
+
import type { Subscription, Plan } from '../../types'
|
|
9
|
+
|
|
10
|
+
interface SubscriptionCardProps {
|
|
11
|
+
subscription: Subscription
|
|
12
|
+
plan?: Plan
|
|
13
|
+
onCancel?: () => void
|
|
14
|
+
onResume?: () => void
|
|
15
|
+
loading?: boolean
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
const STATUS_KEYS: Record<string, { key: string; variant: 'default' | 'secondary' | 'destructive' | 'outline' }> = {
|
|
19
|
+
active: { key: 'billing.status.active', variant: 'default' },
|
|
20
|
+
trialing: { key: 'billing.status.trialing', variant: 'secondary' },
|
|
21
|
+
past_due: { key: 'billing.status.pastDue', variant: 'destructive' },
|
|
22
|
+
canceled: { key: 'billing.status.canceled', variant: 'outline' },
|
|
23
|
+
paused: { key: 'billing.status.paused', variant: 'outline' },
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
function formatDate(dateString: string): string {
|
|
27
|
+
return new Date(dateString).toLocaleDateString('en-US', {
|
|
28
|
+
month: 'long',
|
|
29
|
+
day: 'numeric',
|
|
30
|
+
year: 'numeric',
|
|
31
|
+
})
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export function SubscriptionCard({
|
|
35
|
+
subscription,
|
|
36
|
+
plan,
|
|
37
|
+
onCancel,
|
|
38
|
+
onResume,
|
|
39
|
+
loading,
|
|
40
|
+
}: SubscriptionCardProps) {
|
|
41
|
+
const { t } = useTranslation()
|
|
42
|
+
const statusInfo = STATUS_KEYS[subscription.status] ?? STATUS_KEYS.active
|
|
43
|
+
|
|
44
|
+
return (
|
|
45
|
+
<Card>
|
|
46
|
+
<CardHeader>
|
|
47
|
+
<div className="flex items-center justify-between">
|
|
48
|
+
<CardTitle className="text-lg">
|
|
49
|
+
{plan?.name ?? t('billing.currentPlan')}
|
|
50
|
+
</CardTitle>
|
|
51
|
+
<Badge variant={statusInfo.variant}>{t(statusInfo.key)}</Badge>
|
|
52
|
+
</div>
|
|
53
|
+
{plan && <CardDescription>{plan.description}</CardDescription>}
|
|
54
|
+
</CardHeader>
|
|
55
|
+
|
|
56
|
+
<CardContent className="space-y-4">
|
|
57
|
+
<div className="flex items-center gap-2 text-sm text-muted-foreground">
|
|
58
|
+
<Calendar className="h-4 w-4" />
|
|
59
|
+
<span>
|
|
60
|
+
{subscription.cancelAtPeriodEnd
|
|
61
|
+
? t('billing.cancelsOn', { date: formatDate(subscription.currentPeriodEnd) })
|
|
62
|
+
: t('billing.nextBilling', { date: formatDate(subscription.currentPeriodEnd) })}
|
|
63
|
+
</span>
|
|
64
|
+
</div>
|
|
65
|
+
|
|
66
|
+
{subscription.trialEnd && new Date(subscription.trialEnd) > new Date() && (
|
|
67
|
+
<div className="flex items-center gap-2 text-sm text-muted-foreground">
|
|
68
|
+
<AlertCircle className="h-4 w-4" />
|
|
69
|
+
<span>{t('billing.trialEnds', { date: formatDate(subscription.trialEnd) })}</span>
|
|
70
|
+
</div>
|
|
71
|
+
)}
|
|
72
|
+
|
|
73
|
+
{plan && (
|
|
74
|
+
<div className="text-sm">
|
|
75
|
+
<span className="text-2xl font-bold">
|
|
76
|
+
${subscription.interval === 'monthly' ? (plan.prices?.monthly ?? plan.priceMonthly) : (plan.prices?.yearly ?? plan.priceYearly)}
|
|
77
|
+
</span>
|
|
78
|
+
<span className="text-muted-foreground">
|
|
79
|
+
/{subscription.interval === 'monthly' ? 'mo' : 'yr'}
|
|
80
|
+
</span>
|
|
81
|
+
</div>
|
|
82
|
+
)}
|
|
83
|
+
|
|
84
|
+
{subscription.status === 'past_due' && (
|
|
85
|
+
<div className={cn(
|
|
86
|
+
'rounded-md border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive'
|
|
87
|
+
)}>
|
|
88
|
+
{t('billing.pastDueWarning')}
|
|
89
|
+
</div>
|
|
90
|
+
)}
|
|
91
|
+
</CardContent>
|
|
92
|
+
|
|
93
|
+
<CardFooter className="gap-2">
|
|
94
|
+
{subscription.cancelAtPeriodEnd && onResume ? (
|
|
95
|
+
<Button variant="default" onClick={onResume} disabled={loading}>
|
|
96
|
+
{loading ? t('billing.resuming') : t('billing.resumeSubscription')}
|
|
97
|
+
</Button>
|
|
98
|
+
) : (
|
|
99
|
+
onCancel && subscription.status === 'active' && (
|
|
100
|
+
<Button variant="outline" onClick={onCancel} disabled={loading}>
|
|
101
|
+
{loading ? t('billing.canceling') : t('billing.cancelSubscription')}
|
|
102
|
+
</Button>
|
|
103
|
+
)
|
|
104
|
+
)}
|
|
105
|
+
</CardFooter>
|
|
106
|
+
</Card>
|
|
107
|
+
)
|
|
108
|
+
}
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { MessageCircle, X } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { useChatStore } from '../../stores/chat.store'
|
|
5
|
+
import { useOrganizationStore } from '../../stores/organization.store'
|
|
6
|
+
import { useAITools, type ResolvedSuggestion } from '../../hooks/useAITools'
|
|
7
|
+
import { useChat } from '../../hooks/useChat'
|
|
8
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
9
|
+
|
|
10
|
+
interface ChatFabProps {
|
|
11
|
+
className?: string
|
|
12
|
+
apiEndpoint?: string
|
|
13
|
+
systemPrompt?: string
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
function TypewriterText({ text, speed = 28, onDone }: { text: string; speed?: number; onDone?: () => void }) {
|
|
17
|
+
const [displayed, setDisplayed] = React.useState('')
|
|
18
|
+
const indexRef = React.useRef(0)
|
|
19
|
+
|
|
20
|
+
React.useEffect(() => {
|
|
21
|
+
setDisplayed('')
|
|
22
|
+
indexRef.current = 0
|
|
23
|
+
const interval = setInterval(() => {
|
|
24
|
+
indexRef.current++
|
|
25
|
+
if (indexRef.current > text.length) {
|
|
26
|
+
clearInterval(interval)
|
|
27
|
+
onDone?.()
|
|
28
|
+
return
|
|
29
|
+
}
|
|
30
|
+
setDisplayed(text.slice(0, indexRef.current))
|
|
31
|
+
}, speed)
|
|
32
|
+
return () => clearInterval(interval)
|
|
33
|
+
}, [text, speed])
|
|
34
|
+
|
|
35
|
+
return (
|
|
36
|
+
<span>
|
|
37
|
+
{displayed}
|
|
38
|
+
{displayed.length < text.length && (
|
|
39
|
+
<span className="inline-block w-[2px] h-3 bg-current/60 animate-pulse ml-0.5 -mb-px" />
|
|
40
|
+
)}
|
|
41
|
+
</span>
|
|
42
|
+
)
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/** Injected once — keyframes for pill grow/shrink */
|
|
46
|
+
const STYLE_ID = 'chat-fab-anims'
|
|
47
|
+
function ensureStyles() {
|
|
48
|
+
if (typeof document === 'undefined') return
|
|
49
|
+
if (document.getElementById(STYLE_ID)) return
|
|
50
|
+
const style = document.createElement('style')
|
|
51
|
+
style.id = STYLE_ID
|
|
52
|
+
style.textContent = `
|
|
53
|
+
@keyframes fabPillGrow {
|
|
54
|
+
0% { max-width: 2.75rem; opacity: 0.6; }
|
|
55
|
+
40% { opacity: 1; }
|
|
56
|
+
100% { max-width: 22rem; opacity: 1; }
|
|
57
|
+
}
|
|
58
|
+
@keyframes fabPillShrink {
|
|
59
|
+
0% { max-width: 22rem; opacity: 1; }
|
|
60
|
+
60% { opacity: 0.6; }
|
|
61
|
+
100% { max-width: 2.75rem; opacity: 0; }
|
|
62
|
+
}
|
|
63
|
+
`
|
|
64
|
+
document.head.appendChild(style)
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export function ChatFab({ className, apiEndpoint, systemPrompt }: ChatFabProps) {
|
|
68
|
+
const { isOpen, toggleOpen, setOpen } = useChatStore()
|
|
69
|
+
const currentOrg = useOrganizationStore((s) => s.currentOrg)
|
|
70
|
+
const { contextualSuggestions } = useAITools()
|
|
71
|
+
const { sendMessage } = useChat({ apiEndpoint, systemPrompt })
|
|
72
|
+
const { t } = useTranslation()
|
|
73
|
+
|
|
74
|
+
const [phase, setPhase] = React.useState<'idle' | 'expanding' | 'typing' | 'visible' | 'collapsing'>('idle')
|
|
75
|
+
const [activeSuggestion, setActiveSuggestion] = React.useState<ResolvedSuggestion | null>(null)
|
|
76
|
+
const timersRef = React.useRef<ReturnType<typeof setTimeout>[]>([])
|
|
77
|
+
const prevSuggestionRef = React.useRef<string | null>(null)
|
|
78
|
+
|
|
79
|
+
React.useEffect(ensureStyles, [])
|
|
80
|
+
|
|
81
|
+
const clearTimers = () => { timersRef.current.forEach(clearTimeout); timersRef.current = [] }
|
|
82
|
+
const addTimer = (fn: () => void, ms: number) => { timersRef.current.push(setTimeout(fn, ms)) }
|
|
83
|
+
|
|
84
|
+
const topSuggestion = contextualSuggestions[0] ?? null
|
|
85
|
+
const topLabel = topSuggestion?.label ?? null
|
|
86
|
+
|
|
87
|
+
React.useEffect(() => {
|
|
88
|
+
if (isOpen || !topSuggestion || !topLabel) {
|
|
89
|
+
if (phase !== 'idle') { clearTimers(); setPhase('collapsing'); addTimer(() => setPhase('idle'), 450) }
|
|
90
|
+
prevSuggestionRef.current = null
|
|
91
|
+
return
|
|
92
|
+
}
|
|
93
|
+
if (prevSuggestionRef.current === topLabel) return
|
|
94
|
+
prevSuggestionRef.current = topLabel
|
|
95
|
+
clearTimers()
|
|
96
|
+
|
|
97
|
+
const startExpand = () => {
|
|
98
|
+
setActiveSuggestion(topSuggestion)
|
|
99
|
+
setPhase('expanding')
|
|
100
|
+
// Start typing while pill is still growing
|
|
101
|
+
addTimer(() => setPhase('typing'), 350)
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
if (phase !== 'idle') {
|
|
105
|
+
setPhase('collapsing')
|
|
106
|
+
addTimer(startExpand, 500)
|
|
107
|
+
} else {
|
|
108
|
+
addTimer(startExpand, 800)
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
addTimer(() => {
|
|
112
|
+
setPhase('collapsing')
|
|
113
|
+
addTimer(() => setPhase('idle'), 450)
|
|
114
|
+
}, 9000)
|
|
115
|
+
|
|
116
|
+
return clearTimers
|
|
117
|
+
}, [topLabel, isOpen])
|
|
118
|
+
|
|
119
|
+
React.useEffect(() => {
|
|
120
|
+
if (isOpen && phase !== 'idle') { clearTimers(); setPhase('idle') }
|
|
121
|
+
}, [isOpen])
|
|
122
|
+
|
|
123
|
+
if (!currentOrg) return null
|
|
124
|
+
|
|
125
|
+
// Resolve translated label for a suggestion (same logic as ChatSuggestions)
|
|
126
|
+
const resolveLabel = (s: ResolvedSuggestion, index: number) => {
|
|
127
|
+
const key = `chat.suggestion.${s.toolId}.${index}`
|
|
128
|
+
const translated = t(key)
|
|
129
|
+
return translated === key ? s.label : translated
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
const activeLabel = activeSuggestion ? resolveLabel(activeSuggestion, 0) : ''
|
|
133
|
+
|
|
134
|
+
const handleSuggestionClick = () => {
|
|
135
|
+
if (!activeSuggestion) return
|
|
136
|
+
clearTimers()
|
|
137
|
+
setPhase('idle')
|
|
138
|
+
setOpen(true)
|
|
139
|
+
setTimeout(() => { sendMessage(activeSuggestion.prompt ?? activeLabel) }, 150)
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
const showPill = phase === 'expanding' || phase === 'typing' || phase === 'visible'
|
|
143
|
+
|
|
144
|
+
return (
|
|
145
|
+
<div className={cn('fixed bottom-4 right-4 z-50 hidden md:block', className)}>
|
|
146
|
+
{/* Pill — grows from FAB circle size outward */}
|
|
147
|
+
{(showPill || phase === 'collapsing') && !isOpen && (
|
|
148
|
+
<button
|
|
149
|
+
onClick={handleSuggestionClick}
|
|
150
|
+
className="absolute right-0 bottom-0 flex flex-col items-start justify-center rounded-full bg-foreground text-background shadow-lg hover:opacity-95 cursor-pointer pl-5 overflow-hidden"
|
|
151
|
+
style={{
|
|
152
|
+
paddingRight: '3.25rem',
|
|
153
|
+
height: '2.75rem',
|
|
154
|
+
animation: showPill
|
|
155
|
+
? 'fabPillGrow 650ms cubic-bezier(0.22, 1, 0.36, 1) forwards'
|
|
156
|
+
: 'fabPillShrink 400ms cubic-bezier(0.55, 0, 1, 0.45) forwards',
|
|
157
|
+
}}
|
|
158
|
+
>
|
|
159
|
+
<span className="text-[7px] font-semibold uppercase tracking-widest opacity-40">
|
|
160
|
+
{t('chat.fab.tryIt')}
|
|
161
|
+
</span>
|
|
162
|
+
<span className={cn(
|
|
163
|
+
'text-[13px] font-medium whitespace-nowrap min-h-[1.15em] transition-opacity',
|
|
164
|
+
phase === 'expanding' ? 'opacity-0 duration-200' : 'opacity-100 duration-300',
|
|
165
|
+
)}>
|
|
166
|
+
{phase === 'typing' && activeSuggestion ? (
|
|
167
|
+
<TypewriterText
|
|
168
|
+
text={`"${activeLabel}"`}
|
|
169
|
+
onDone={() => setPhase('visible')}
|
|
170
|
+
/>
|
|
171
|
+
) : (phase === 'visible') && activeSuggestion ? (
|
|
172
|
+
`"${activeLabel}"`
|
|
173
|
+
) : (
|
|
174
|
+
'\u00A0'
|
|
175
|
+
)}
|
|
176
|
+
</span>
|
|
177
|
+
</button>
|
|
178
|
+
)}
|
|
179
|
+
|
|
180
|
+
{/* FAB circle */}
|
|
181
|
+
<button
|
|
182
|
+
onClick={toggleOpen}
|
|
183
|
+
className={cn(
|
|
184
|
+
'relative z-10 h-11 w-11 flex items-center justify-center transition-all hover:opacity-90 focus:outline-none',
|
|
185
|
+
isOpen
|
|
186
|
+
? 'rounded-tl-none rounded-tr-lg rounded-br-full rounded-bl-full bg-card text-muted-foreground border border-border/50 border-t-0 shadow-lg'
|
|
187
|
+
: 'rounded-full bg-foreground text-background shadow-lg',
|
|
188
|
+
)}
|
|
189
|
+
aria-label={isOpen ? 'Close chat' : 'Open chat'}
|
|
190
|
+
>
|
|
191
|
+
{isOpen ? <X className="h-4 w-4" /> : <MessageCircle className="h-4 w-4" />}
|
|
192
|
+
</button>
|
|
193
|
+
</div>
|
|
194
|
+
)
|
|
195
|
+
}
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { ArrowUp, Settings2 } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
5
|
+
import { useChatStore, type ChatMessage } from '../../stores/chat.store'
|
|
6
|
+
import { useChat } from '../../hooks/useChat'
|
|
7
|
+
import { useAITools } from '../../hooks/useAITools'
|
|
8
|
+
import { ChatSuggestions, ChatToolsPanel } from './ChatSuggestions'
|
|
9
|
+
|
|
10
|
+
interface ChatPanelProps {
|
|
11
|
+
title?: string
|
|
12
|
+
apiEndpoint?: string
|
|
13
|
+
systemPrompt?: string
|
|
14
|
+
className?: string
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
export function ChatPanel({
|
|
18
|
+
title = 'Assistant',
|
|
19
|
+
apiEndpoint,
|
|
20
|
+
systemPrompt,
|
|
21
|
+
className,
|
|
22
|
+
}: ChatPanelProps) {
|
|
23
|
+
const { isOpen, messages, isStreaming } = useChatStore()
|
|
24
|
+
const { sendMessage } = useChat({ apiEndpoint, systemPrompt })
|
|
25
|
+
const { t } = useTranslation()
|
|
26
|
+
const { suggestions, toolGroups } = useAITools()
|
|
27
|
+
const [input, setInput] = React.useState('')
|
|
28
|
+
const [showTools, setShowTools] = React.useState(false)
|
|
29
|
+
const scrollRef = React.useRef<HTMLDivElement>(null)
|
|
30
|
+
const inputRef = React.useRef<HTMLInputElement>(null)
|
|
31
|
+
|
|
32
|
+
const hasMessages = messages.length > 0
|
|
33
|
+
const totalTools = toolGroups.reduce((sum, g) => sum + g.tools.length, 0)
|
|
34
|
+
|
|
35
|
+
React.useEffect(() => {
|
|
36
|
+
scrollRef.current?.scrollTo({ top: scrollRef.current.scrollHeight, behavior: 'smooth' })
|
|
37
|
+
}, [messages])
|
|
38
|
+
|
|
39
|
+
React.useEffect(() => {
|
|
40
|
+
if (isOpen) {
|
|
41
|
+
const timer = setTimeout(() => inputRef.current?.focus(), 100)
|
|
42
|
+
return () => clearTimeout(timer)
|
|
43
|
+
}
|
|
44
|
+
setShowTools(false)
|
|
45
|
+
}, [isOpen])
|
|
46
|
+
|
|
47
|
+
if (!isOpen) return null
|
|
48
|
+
|
|
49
|
+
const handleSubmit = (e: React.FormEvent) => {
|
|
50
|
+
e.preventDefault()
|
|
51
|
+
const text = input.trim()
|
|
52
|
+
if (!text || isStreaming) return
|
|
53
|
+
setInput('')
|
|
54
|
+
setShowTools(false)
|
|
55
|
+
sendMessage(text)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
return (
|
|
59
|
+
<div
|
|
60
|
+
className={cn(
|
|
61
|
+
'fixed z-40 flex flex-col overflow-hidden bg-card shadow-2xl',
|
|
62
|
+
// Mobile: below header, above bottom nav
|
|
63
|
+
'inset-x-0 top-12 bottom-16 rounded-none border-0',
|
|
64
|
+
// Desktop: floating card — sits just above the FAB, touching it
|
|
65
|
+
'md:inset-auto md:bottom-[3.75rem] md:right-4 md:w-[22rem] md:max-h-[min(70vh,520px)] md:rounded-2xl md:rounded-br-none md:border md:border-border/50',
|
|
66
|
+
'animate-in slide-in-from-bottom-2 fade-in-0 duration-150',
|
|
67
|
+
className
|
|
68
|
+
)}
|
|
69
|
+
>
|
|
70
|
+
{/* Header */}
|
|
71
|
+
<div className="flex items-center justify-between border-b border-border/40 px-4 py-2.5">
|
|
72
|
+
<div className="flex items-center gap-2">
|
|
73
|
+
<div className="h-1.5 w-1.5 rounded-full bg-success" />
|
|
74
|
+
<span className="text-xs font-semibold text-foreground">{title}</span>
|
|
75
|
+
</div>
|
|
76
|
+
{totalTools > 0 && (
|
|
77
|
+
<button
|
|
78
|
+
onClick={() => setShowTools(!showTools)}
|
|
79
|
+
className={cn(
|
|
80
|
+
'inline-flex items-center gap-1 rounded-md px-1.5 py-1 text-[10px] transition-colors',
|
|
81
|
+
showTools
|
|
82
|
+
? 'bg-muted text-foreground'
|
|
83
|
+
: 'text-muted-foreground hover:bg-muted/50 hover:text-foreground'
|
|
84
|
+
)}
|
|
85
|
+
title="Available tools"
|
|
86
|
+
>
|
|
87
|
+
<Settings2 className="h-3 w-3" />
|
|
88
|
+
<span className="font-medium">{totalTools}</span>
|
|
89
|
+
</button>
|
|
90
|
+
)}
|
|
91
|
+
</div>
|
|
92
|
+
|
|
93
|
+
{/* Tools panel overlay */}
|
|
94
|
+
{showTools ? (
|
|
95
|
+
<ChatToolsPanel toolGroups={toolGroups} onClose={() => setShowTools(false)} />
|
|
96
|
+
) : (
|
|
97
|
+
/* Messages / suggestions area */
|
|
98
|
+
<div
|
|
99
|
+
ref={scrollRef}
|
|
100
|
+
className={cn(
|
|
101
|
+
'min-h-0 flex-1 overflow-y-auto',
|
|
102
|
+
hasMessages && 'space-y-2 px-3 py-3'
|
|
103
|
+
)}
|
|
104
|
+
>
|
|
105
|
+
{!hasMessages && (
|
|
106
|
+
<ChatSuggestions
|
|
107
|
+
suggestions={suggestions}
|
|
108
|
+
onSelect={(suggestion) => {
|
|
109
|
+
setInput('')
|
|
110
|
+
sendMessage(suggestion.prompt ?? suggestion.label)
|
|
111
|
+
}}
|
|
112
|
+
/>
|
|
113
|
+
)}
|
|
114
|
+
{messages.map((msg) => (
|
|
115
|
+
<MessageBubble key={msg.id} message={msg} />
|
|
116
|
+
))}
|
|
117
|
+
{isStreaming && messages[messages.length - 1]?.content === '' && (
|
|
118
|
+
<div className="flex items-center gap-1 px-2 py-1.5">
|
|
119
|
+
<span className="h-1 w-1 rounded-full bg-muted-foreground/50 animate-pulse" />
|
|
120
|
+
<span className="h-1 w-1 rounded-full bg-muted-foreground/50 animate-pulse [animation-delay:150ms]" />
|
|
121
|
+
<span className="h-1 w-1 rounded-full bg-muted-foreground/50 animate-pulse [animation-delay:300ms]" />
|
|
122
|
+
</div>
|
|
123
|
+
)}
|
|
124
|
+
</div>
|
|
125
|
+
)}
|
|
126
|
+
|
|
127
|
+
{/* Input */}
|
|
128
|
+
<form onSubmit={handleSubmit} className="border-t border-border/40 p-2">
|
|
129
|
+
<div className="flex items-center gap-1.5 rounded-full border border-border bg-background py-1 pl-3.5 pr-1 transition-colors focus-within:border-foreground/20">
|
|
130
|
+
<input
|
|
131
|
+
ref={inputRef}
|
|
132
|
+
type="text"
|
|
133
|
+
value={input}
|
|
134
|
+
onChange={(e) => setInput(e.target.value)}
|
|
135
|
+
placeholder={t('chat.messagePlaceholder')}
|
|
136
|
+
className="min-w-0 flex-1 bg-transparent text-sm text-foreground placeholder:text-muted-foreground/50 focus:outline-none"
|
|
137
|
+
disabled={isStreaming}
|
|
138
|
+
/>
|
|
139
|
+
<button
|
|
140
|
+
type="submit"
|
|
141
|
+
disabled={!input.trim() || isStreaming}
|
|
142
|
+
className="inline-flex h-7 w-7 shrink-0 items-center justify-center rounded-full bg-foreground text-background transition-opacity hover:opacity-80 disabled:opacity-20"
|
|
143
|
+
aria-label="Send message"
|
|
144
|
+
>
|
|
145
|
+
<ArrowUp className="h-3.5 w-3.5" strokeWidth={2.5} />
|
|
146
|
+
</button>
|
|
147
|
+
</div>
|
|
148
|
+
</form>
|
|
149
|
+
</div>
|
|
150
|
+
)
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
function MessageBubble({ message }: { message: ChatMessage }) {
|
|
154
|
+
const isUser = message.role === 'user'
|
|
155
|
+
|
|
156
|
+
return (
|
|
157
|
+
<div className={cn('flex', isUser ? 'justify-end' : 'justify-start')}>
|
|
158
|
+
<div
|
|
159
|
+
className={cn(
|
|
160
|
+
'max-w-[85%] rounded-2xl px-3 py-1.5 text-[13px] leading-relaxed',
|
|
161
|
+
isUser
|
|
162
|
+
? 'rounded-br-sm bg-foreground text-background'
|
|
163
|
+
: 'rounded-bl-sm bg-muted text-foreground'
|
|
164
|
+
)}
|
|
165
|
+
>
|
|
166
|
+
{message.content}
|
|
167
|
+
</div>
|
|
168
|
+
</div>
|
|
169
|
+
)
|
|
170
|
+
}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Sparkles, BookOpen, Pencil } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
5
|
+
import type { ResolvedSuggestion, ResolvedToolGroup } from '../../hooks/useAITools'
|
|
6
|
+
|
|
7
|
+
interface ChatSuggestionsProps {
|
|
8
|
+
suggestions: ResolvedSuggestion[]
|
|
9
|
+
onSelect: (suggestion: ResolvedSuggestion) => void
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
const MAX_SUGGESTIONS = 5
|
|
13
|
+
|
|
14
|
+
export function ChatSuggestions({ suggestions, onSelect }: ChatSuggestionsProps) {
|
|
15
|
+
const { t } = useTranslation()
|
|
16
|
+
const visible = suggestions.slice(0, MAX_SUGGESTIONS)
|
|
17
|
+
|
|
18
|
+
return (
|
|
19
|
+
<div className="flex flex-col gap-1 px-3 py-3">
|
|
20
|
+
<div className="flex items-center gap-1.5 px-1 pb-1">
|
|
21
|
+
<Sparkles className="h-3 w-3 text-muted-foreground/60" />
|
|
22
|
+
<span className="text-[11px] font-medium text-muted-foreground/70">{t('chat.tryAsking')}</span>
|
|
23
|
+
</div>
|
|
24
|
+
{visible.map((suggestion, i) => {
|
|
25
|
+
const labelKey = `chat.suggestion.${suggestion.toolId}.${i}`
|
|
26
|
+
const translated = t(labelKey)
|
|
27
|
+
const label = translated === labelKey ? suggestion.label : translated
|
|
28
|
+
return (
|
|
29
|
+
<button
|
|
30
|
+
key={`${suggestion.toolId}-${i}`}
|
|
31
|
+
onClick={() => onSelect(suggestion)}
|
|
32
|
+
className={cn(
|
|
33
|
+
'rounded-md px-2.5 py-1.5 text-left text-xs text-muted-foreground',
|
|
34
|
+
'transition-colors hover:bg-muted hover:text-foreground'
|
|
35
|
+
)}
|
|
36
|
+
>
|
|
37
|
+
“{label}”
|
|
38
|
+
</button>
|
|
39
|
+
)
|
|
40
|
+
})}
|
|
41
|
+
</div>
|
|
42
|
+
)
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
// --- Tools panel (shown via config button) ---
|
|
46
|
+
|
|
47
|
+
interface ChatToolsPanelProps {
|
|
48
|
+
toolGroups: ResolvedToolGroup[]
|
|
49
|
+
onClose: () => void
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
export function ChatToolsPanel({ toolGroups, onClose }: ChatToolsPanelProps) {
|
|
53
|
+
const { t } = useTranslation()
|
|
54
|
+
const totalTools = toolGroups.reduce((sum, g) => sum + g.tools.length, 0)
|
|
55
|
+
|
|
56
|
+
return (
|
|
57
|
+
<div className="min-h-0 flex-1 overflow-y-auto">
|
|
58
|
+
<div className="sticky top-0 z-10 flex items-center justify-between border-b border-border/40 bg-card px-4 py-2">
|
|
59
|
+
<span className="text-[11px] font-semibold text-foreground">
|
|
60
|
+
{t('chat.toolsAvailable', { count: totalTools })}
|
|
61
|
+
</span>
|
|
62
|
+
<button
|
|
63
|
+
onClick={onClose}
|
|
64
|
+
className="text-[11px] text-muted-foreground hover:text-foreground transition-colors"
|
|
65
|
+
>
|
|
66
|
+
{t('common.done')}
|
|
67
|
+
</button>
|
|
68
|
+
</div>
|
|
69
|
+
<div className="flex flex-col gap-3 px-3 py-3">
|
|
70
|
+
{toolGroups.map((group) => (
|
|
71
|
+
<div key={group.category} className="flex flex-col gap-0.5">
|
|
72
|
+
<span className="px-1 text-[10px] font-semibold uppercase tracking-wider text-muted-foreground/50">
|
|
73
|
+
{group.category}
|
|
74
|
+
</span>
|
|
75
|
+
{group.tools.map(({ tool, signature }) => {
|
|
76
|
+
const ModeIcon = tool.mode === 'persist' ? Pencil : BookOpen
|
|
77
|
+
return (
|
|
78
|
+
<div
|
|
79
|
+
key={tool.id}
|
|
80
|
+
className="flex items-start gap-1.5 rounded-md px-2 py-1 transition-colors hover:bg-muted/50"
|
|
81
|
+
title={tool.description}
|
|
82
|
+
>
|
|
83
|
+
<ModeIcon className="mt-0.5 h-3 w-3 shrink-0 text-muted-foreground/40" />
|
|
84
|
+
<div className="min-w-0">
|
|
85
|
+
<code className="text-[11px] font-mono text-muted-foreground">
|
|
86
|
+
{signature}
|
|
87
|
+
</code>
|
|
88
|
+
{tool.description && (
|
|
89
|
+
<p className="text-[10px] leading-tight text-muted-foreground/40 truncate">
|
|
90
|
+
{tool.description}
|
|
91
|
+
</p>
|
|
92
|
+
)}
|
|
93
|
+
</div>
|
|
94
|
+
</div>
|
|
95
|
+
)
|
|
96
|
+
})}
|
|
97
|
+
</div>
|
|
98
|
+
))}
|
|
99
|
+
</div>
|
|
100
|
+
</div>
|
|
101
|
+
)
|
|
102
|
+
}
|