@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,171 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { MapPin, Phone, Plus, Loader2 } from 'lucide-react'
|
|
3
|
+
import { Card, CardContent, CardDescription, CardHeader, CardTitle } 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 { useTranslation } from '../../hooks/useTranslation'
|
|
8
|
+
import type { Location } from '../../types/tenant'
|
|
9
|
+
|
|
10
|
+
interface LocationsOverviewProps {
|
|
11
|
+
locations: Location[]
|
|
12
|
+
loading?: boolean
|
|
13
|
+
canManage?: boolean
|
|
14
|
+
onAdd?: (data: { name: string; address?: string; city?: string; state?: string; country?: string; phone?: string; isHeadquarters?: boolean }) => Promise<void>
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
function AddLocationForm({ onAdd, onCancel }: { onAdd: LocationsOverviewProps['onAdd']; onCancel: () => void }) {
|
|
18
|
+
const { t } = useTranslation()
|
|
19
|
+
const [name, setName] = React.useState('')
|
|
20
|
+
const [address, setAddress] = React.useState('')
|
|
21
|
+
const [city, setCity] = React.useState('')
|
|
22
|
+
const [state, setState] = React.useState('')
|
|
23
|
+
const [country, setCountry] = React.useState('BR')
|
|
24
|
+
const [phone, setPhone] = React.useState('')
|
|
25
|
+
const [isHQ, setIsHQ] = React.useState(false)
|
|
26
|
+
const [saving, setSaving] = React.useState(false)
|
|
27
|
+
|
|
28
|
+
const handleSubmit = async (e: React.FormEvent) => {
|
|
29
|
+
e.preventDefault()
|
|
30
|
+
if (!name.trim() || !onAdd) return
|
|
31
|
+
setSaving(true)
|
|
32
|
+
try {
|
|
33
|
+
await onAdd({
|
|
34
|
+
name: name.trim(),
|
|
35
|
+
address: address.trim() || undefined,
|
|
36
|
+
city: city.trim() || undefined,
|
|
37
|
+
state: state.trim() || undefined,
|
|
38
|
+
country: country.trim() || undefined,
|
|
39
|
+
phone: phone.trim() || undefined,
|
|
40
|
+
isHeadquarters: isHQ,
|
|
41
|
+
})
|
|
42
|
+
onCancel()
|
|
43
|
+
} catch {
|
|
44
|
+
// handled via toast in connected wrapper
|
|
45
|
+
} finally {
|
|
46
|
+
setSaving(false)
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
return (
|
|
51
|
+
<form onSubmit={handleSubmit} className="border rounded-lg p-4 space-y-3">
|
|
52
|
+
<p className="text-sm font-medium">{t('settings.locations.newLocation')}</p>
|
|
53
|
+
<Input value={name} onChange={(e) => setName(e.target.value)} placeholder={t('settings.locations.namePlaceholder')} required />
|
|
54
|
+
<Input value={address} onChange={(e) => setAddress(e.target.value)} placeholder={t('settings.locations.addressPlaceholder')} />
|
|
55
|
+
<div className="grid gap-3 sm:grid-cols-3">
|
|
56
|
+
<Input value={city} onChange={(e) => setCity(e.target.value)} placeholder={t('settings.locations.cityPlaceholder')} />
|
|
57
|
+
<Input value={state} onChange={(e) => setState(e.target.value)} placeholder={t('settings.locations.statePlaceholder')} />
|
|
58
|
+
<Input value={country} onChange={(e) => setCountry(e.target.value)} placeholder={t('settings.locations.countryPlaceholder')} />
|
|
59
|
+
</div>
|
|
60
|
+
<Input value={phone} onChange={(e) => setPhone(e.target.value)} placeholder={t('settings.locations.phonePlaceholder')} />
|
|
61
|
+
<label className="flex items-center gap-2.5 cursor-pointer text-sm">
|
|
62
|
+
<Checkbox checked={isHQ} onChange={setIsHQ} />
|
|
63
|
+
{t('settings.locations.headquarters')}
|
|
64
|
+
</label>
|
|
65
|
+
<div className="flex justify-end gap-2 pt-1">
|
|
66
|
+
<Button type="button" variant="outline" size="sm" onClick={onCancel} disabled={saving}>{t('common.cancel')}</Button>
|
|
67
|
+
<Button type="submit" size="sm" disabled={saving || !name.trim()}>
|
|
68
|
+
{saving && <Loader2 className="mr-2 h-3.5 w-3.5 animate-spin" />}
|
|
69
|
+
{saving ? t('settings.locations.adding') : t('settings.locations.addLocation')}
|
|
70
|
+
</Button>
|
|
71
|
+
</div>
|
|
72
|
+
</form>
|
|
73
|
+
)
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
export function LocationsOverview({ locations, loading, canManage, onAdd }: LocationsOverviewProps) {
|
|
77
|
+
const { t } = useTranslation()
|
|
78
|
+
const [showForm, setShowForm] = React.useState(false)
|
|
79
|
+
|
|
80
|
+
if (loading) {
|
|
81
|
+
return (
|
|
82
|
+
<Card>
|
|
83
|
+
<CardHeader>
|
|
84
|
+
<CardTitle className="text-lg">{t('settings.locations.title')}</CardTitle>
|
|
85
|
+
<CardDescription>{t('settings.locations.subtitle')}</CardDescription>
|
|
86
|
+
</CardHeader>
|
|
87
|
+
<CardContent>
|
|
88
|
+
<div className="flex items-center justify-center py-8">
|
|
89
|
+
<div className="h-6 w-6 animate-spin rounded-full border-2 border-muted border-t-primary" />
|
|
90
|
+
</div>
|
|
91
|
+
</CardContent>
|
|
92
|
+
</Card>
|
|
93
|
+
)
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
return (
|
|
97
|
+
<Card>
|
|
98
|
+
<CardHeader>
|
|
99
|
+
<div className="flex items-center justify-between">
|
|
100
|
+
<div>
|
|
101
|
+
<CardTitle className="text-lg">Locations</CardTitle>
|
|
102
|
+
<CardDescription>
|
|
103
|
+
{locations.length === 0
|
|
104
|
+
? t('settings.locations.noLocations')
|
|
105
|
+
: t('settings.locations.countDescription', { count: String(locations.length), plural: locations.length === 1 ? '' : 's' })}
|
|
106
|
+
</CardDescription>
|
|
107
|
+
</div>
|
|
108
|
+
{canManage && !showForm && (
|
|
109
|
+
<Button size="sm" variant="outline" onClick={() => setShowForm(true)}>
|
|
110
|
+
<Plus className="h-4 w-4 mr-1" />
|
|
111
|
+
{t('settings.locations.addLocation')}
|
|
112
|
+
</Button>
|
|
113
|
+
)}
|
|
114
|
+
</div>
|
|
115
|
+
</CardHeader>
|
|
116
|
+
<CardContent className="space-y-4">
|
|
117
|
+
{showForm && (
|
|
118
|
+
<AddLocationForm onAdd={onAdd} onCancel={() => setShowForm(false)} />
|
|
119
|
+
)}
|
|
120
|
+
|
|
121
|
+
{locations.length === 0 && !showForm ? (
|
|
122
|
+
<div className="flex flex-col items-center justify-center py-8 text-center">
|
|
123
|
+
<MapPin className="h-8 w-8 text-muted-foreground/50 mb-2" />
|
|
124
|
+
<p className="text-sm text-muted-foreground mb-3">
|
|
125
|
+
{t('settings.locations.empty')}
|
|
126
|
+
</p>
|
|
127
|
+
{canManage && (
|
|
128
|
+
<Button size="sm" onClick={() => setShowForm(true)}>
|
|
129
|
+
<Plus className="h-4 w-4 mr-1" />
|
|
130
|
+
{t('settings.locations.addFirst')}
|
|
131
|
+
</Button>
|
|
132
|
+
)}
|
|
133
|
+
</div>
|
|
134
|
+
) : (
|
|
135
|
+
<div className="divide-y">
|
|
136
|
+
{locations.map((loc) => (
|
|
137
|
+
<div key={loc.id} className="flex items-center justify-between py-3 first:pt-0 last:pb-0">
|
|
138
|
+
<div className="space-y-0.5">
|
|
139
|
+
<div className="flex items-center gap-2">
|
|
140
|
+
<span className="text-sm font-medium">{loc.name}</span>
|
|
141
|
+
{loc.isHeadquarters && (
|
|
142
|
+
<span className="inline-flex items-center rounded-full bg-primary/10 px-2 py-0.5 text-[10px] font-semibold text-primary">
|
|
143
|
+
HQ
|
|
144
|
+
</span>
|
|
145
|
+
)}
|
|
146
|
+
{!loc.isActive && (
|
|
147
|
+
<span className="inline-flex items-center rounded-full bg-muted px-2 py-0.5 text-[10px] font-semibold text-muted-foreground">
|
|
148
|
+
{t('common.inactive')}
|
|
149
|
+
</span>
|
|
150
|
+
)}
|
|
151
|
+
</div>
|
|
152
|
+
{(loc.city || loc.state) && (
|
|
153
|
+
<p className="text-xs text-muted-foreground">
|
|
154
|
+
{[loc.city, loc.state].filter(Boolean).join(', ')}
|
|
155
|
+
</p>
|
|
156
|
+
)}
|
|
157
|
+
</div>
|
|
158
|
+
{loc.phone && (
|
|
159
|
+
<div className="flex items-center gap-1.5 text-xs text-muted-foreground">
|
|
160
|
+
<Phone className="h-3 w-3" />
|
|
161
|
+
{loc.phone}
|
|
162
|
+
</div>
|
|
163
|
+
)}
|
|
164
|
+
</div>
|
|
165
|
+
))}
|
|
166
|
+
</div>
|
|
167
|
+
)}
|
|
168
|
+
</CardContent>
|
|
169
|
+
</Card>
|
|
170
|
+
)
|
|
171
|
+
}
|
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { KeyRound, Loader2, Clock, ShieldCheck, UserPlus } from 'lucide-react'
|
|
3
|
+
import { Button } from '@fayz-ai/ui'
|
|
4
|
+
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@fayz-ai/ui'
|
|
5
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
6
|
+
import type { AuthUser } from '../../types'
|
|
7
|
+
|
|
8
|
+
export interface TenantSecuritySettings {
|
|
9
|
+
sessionTimeoutMinutes?: number
|
|
10
|
+
require2FA?: boolean
|
|
11
|
+
inviteOnly?: boolean
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
interface SecuritySettingsProps {
|
|
15
|
+
user?: AuthUser | null
|
|
16
|
+
tenantSecurity?: TenantSecuritySettings
|
|
17
|
+
isAdmin?: boolean
|
|
18
|
+
onResetPassword?: () => Promise<void>
|
|
19
|
+
onTenantSecurityChange?: (settings: TenantSecuritySettings) => Promise<void>
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const SESSION_TIMEOUT_OPTIONS = [
|
|
23
|
+
{ label: '30 minutes', value: 30 },
|
|
24
|
+
{ label: '1 hour', value: 60 },
|
|
25
|
+
{ label: '4 hours', value: 240 },
|
|
26
|
+
{ label: '8 hours', value: 480 },
|
|
27
|
+
{ label: '24 hours', value: 1440 },
|
|
28
|
+
{ label: 'No limit', value: 0 },
|
|
29
|
+
]
|
|
30
|
+
|
|
31
|
+
function ToggleRow({
|
|
32
|
+
icon,
|
|
33
|
+
label,
|
|
34
|
+
description,
|
|
35
|
+
checked,
|
|
36
|
+
onChange,
|
|
37
|
+
disabled,
|
|
38
|
+
}: {
|
|
39
|
+
icon: React.ReactNode
|
|
40
|
+
label: string
|
|
41
|
+
description: string
|
|
42
|
+
checked: boolean
|
|
43
|
+
onChange: (val: boolean) => void
|
|
44
|
+
disabled?: boolean
|
|
45
|
+
}) {
|
|
46
|
+
return (
|
|
47
|
+
<div className="flex items-start justify-between gap-4 py-3">
|
|
48
|
+
<div className="flex gap-3">
|
|
49
|
+
<div className="mt-0.5 text-muted-foreground">{icon}</div>
|
|
50
|
+
<div>
|
|
51
|
+
<p className="text-sm font-medium">{label}</p>
|
|
52
|
+
<p className="text-xs text-muted-foreground">{description}</p>
|
|
53
|
+
</div>
|
|
54
|
+
</div>
|
|
55
|
+
<button
|
|
56
|
+
type="button"
|
|
57
|
+
role="switch"
|
|
58
|
+
aria-checked={checked}
|
|
59
|
+
disabled={disabled}
|
|
60
|
+
onClick={() => onChange(!checked)}
|
|
61
|
+
className={`relative inline-flex h-5 w-9 shrink-0 cursor-pointer items-center rounded-full transition-colors disabled:cursor-not-allowed disabled:opacity-50 ${
|
|
62
|
+
checked ? 'bg-primary' : 'bg-muted-foreground/30'
|
|
63
|
+
}`}
|
|
64
|
+
>
|
|
65
|
+
<span
|
|
66
|
+
className={`pointer-events-none block h-4 w-4 rounded-full bg-white shadow-sm transition-transform ${
|
|
67
|
+
checked ? 'translate-x-[18px]' : 'translate-x-[2px]'
|
|
68
|
+
}`}
|
|
69
|
+
/>
|
|
70
|
+
</button>
|
|
71
|
+
</div>
|
|
72
|
+
)
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
export function SecuritySettings({
|
|
76
|
+
user,
|
|
77
|
+
tenantSecurity,
|
|
78
|
+
isAdmin,
|
|
79
|
+
onResetPassword,
|
|
80
|
+
onTenantSecurityChange,
|
|
81
|
+
}: SecuritySettingsProps) {
|
|
82
|
+
const { t } = useTranslation()
|
|
83
|
+
const [resetting, setResetting] = React.useState(false)
|
|
84
|
+
const [resetSent, setResetSent] = React.useState(false)
|
|
85
|
+
const [savingTenant, setSavingTenant] = React.useState(false)
|
|
86
|
+
|
|
87
|
+
const handleResetPassword = async () => {
|
|
88
|
+
if (!onResetPassword) return
|
|
89
|
+
setResetting(true)
|
|
90
|
+
try {
|
|
91
|
+
await onResetPassword()
|
|
92
|
+
setResetSent(true)
|
|
93
|
+
} catch {
|
|
94
|
+
// handled via toast in connected wrapper
|
|
95
|
+
} finally {
|
|
96
|
+
setResetting(false)
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
const updateTenantSecurity = async (partial: Partial<TenantSecuritySettings>) => {
|
|
101
|
+
if (!onTenantSecurityChange) return
|
|
102
|
+
setSavingTenant(true)
|
|
103
|
+
try {
|
|
104
|
+
await onTenantSecurityChange({ ...tenantSecurity, ...partial })
|
|
105
|
+
} catch {
|
|
106
|
+
// handled via toast
|
|
107
|
+
} finally {
|
|
108
|
+
setSavingTenant(false)
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
const baseClass =
|
|
113
|
+
'flex h-10 w-full rounded-input border border-input bg-background px-3 py-2 text-sm shadow-sm ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50'
|
|
114
|
+
|
|
115
|
+
return (
|
|
116
|
+
<div className="space-y-6">
|
|
117
|
+
{/* Password reset */}
|
|
118
|
+
<Card>
|
|
119
|
+
<CardHeader>
|
|
120
|
+
<CardTitle className="text-lg">{t('settings.security.passwordTitle')}</CardTitle>
|
|
121
|
+
<CardDescription>
|
|
122
|
+
{t('settings.security.passwordSubtitle')}
|
|
123
|
+
</CardDescription>
|
|
124
|
+
</CardHeader>
|
|
125
|
+
<CardContent>
|
|
126
|
+
<div className="flex items-center justify-between rounded-lg border p-4">
|
|
127
|
+
<div className="space-y-0.5">
|
|
128
|
+
<p className="text-sm font-medium">{user?.email}</p>
|
|
129
|
+
<p className="text-xs text-muted-foreground">
|
|
130
|
+
{resetSent
|
|
131
|
+
? t('settings.security.resetLinkSent')
|
|
132
|
+
: t('settings.security.resetLinkDescription')}
|
|
133
|
+
</p>
|
|
134
|
+
</div>
|
|
135
|
+
<Button
|
|
136
|
+
variant="outline"
|
|
137
|
+
size="sm"
|
|
138
|
+
disabled={resetting || resetSent}
|
|
139
|
+
onClick={handleResetPassword}
|
|
140
|
+
>
|
|
141
|
+
{resetting && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
|
|
142
|
+
<KeyRound className="mr-2 h-4 w-4" />
|
|
143
|
+
{resetSent ? t('settings.security.linkSent') : t('settings.security.resetPassword')}
|
|
144
|
+
</Button>
|
|
145
|
+
</div>
|
|
146
|
+
</CardContent>
|
|
147
|
+
</Card>
|
|
148
|
+
|
|
149
|
+
{/* Tenant security — only visible to admins */}
|
|
150
|
+
{isAdmin && (
|
|
151
|
+
<Card>
|
|
152
|
+
<CardHeader>
|
|
153
|
+
<CardTitle className="text-lg">{t('settings.security.orgSecurityTitle')}</CardTitle>
|
|
154
|
+
<CardDescription>
|
|
155
|
+
{t('settings.security.orgSecuritySubtitle')}
|
|
156
|
+
</CardDescription>
|
|
157
|
+
</CardHeader>
|
|
158
|
+
<CardContent className="space-y-1 divide-y">
|
|
159
|
+
<ToggleRow
|
|
160
|
+
icon={<ShieldCheck className="h-4 w-4" />}
|
|
161
|
+
label={t('settings.security.require2FA')}
|
|
162
|
+
description={t('settings.security.require2FADescription')}
|
|
163
|
+
checked={tenantSecurity?.require2FA ?? false}
|
|
164
|
+
onChange={(val) => updateTenantSecurity({ require2FA: val })}
|
|
165
|
+
disabled={savingTenant}
|
|
166
|
+
/>
|
|
167
|
+
<ToggleRow
|
|
168
|
+
icon={<UserPlus className="h-4 w-4" />}
|
|
169
|
+
label={t('settings.security.inviteOnly')}
|
|
170
|
+
description={t('settings.security.inviteOnlyDescription')}
|
|
171
|
+
checked={tenantSecurity?.inviteOnly ?? true}
|
|
172
|
+
onChange={(val) => updateTenantSecurity({ inviteOnly: val })}
|
|
173
|
+
disabled={savingTenant}
|
|
174
|
+
/>
|
|
175
|
+
|
|
176
|
+
<div className="flex items-start justify-between gap-4 py-3">
|
|
177
|
+
<div className="flex gap-3">
|
|
178
|
+
<div className="mt-0.5 text-muted-foreground">
|
|
179
|
+
<Clock className="h-4 w-4" />
|
|
180
|
+
</div>
|
|
181
|
+
<div>
|
|
182
|
+
<p className="text-sm font-medium">{t('settings.security.sessionTimeout')}</p>
|
|
183
|
+
<p className="text-xs text-muted-foreground">
|
|
184
|
+
{t('settings.security.sessionTimeoutDescription')}
|
|
185
|
+
</p>
|
|
186
|
+
</div>
|
|
187
|
+
</div>
|
|
188
|
+
<select
|
|
189
|
+
value={tenantSecurity?.sessionTimeoutMinutes ?? 0}
|
|
190
|
+
onChange={(e) =>
|
|
191
|
+
updateTenantSecurity({ sessionTimeoutMinutes: Number(e.target.value) })
|
|
192
|
+
}
|
|
193
|
+
disabled={savingTenant}
|
|
194
|
+
className={`${baseClass} w-36`}
|
|
195
|
+
>
|
|
196
|
+
{SESSION_TIMEOUT_OPTIONS.map((opt) => (
|
|
197
|
+
<option key={opt.value} value={opt.value}>
|
|
198
|
+
{opt.label}
|
|
199
|
+
</option>
|
|
200
|
+
))}
|
|
201
|
+
</select>
|
|
202
|
+
</div>
|
|
203
|
+
</CardContent>
|
|
204
|
+
</Card>
|
|
205
|
+
)}
|
|
206
|
+
</div>
|
|
207
|
+
)
|
|
208
|
+
}
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Building2, User, Shield, Palette, ArrowLeft, ChevronRight } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
5
|
+
import { ConnectedCompanySettings } from './ConnectedCompanySettings'
|
|
6
|
+
import { ConnectedUserProfile } from './ConnectedUserProfile'
|
|
7
|
+
import { ConnectedSecuritySettings } from './ConnectedSecuritySettings'
|
|
8
|
+
import { ConnectedBrandingSettings } from './ConnectedBrandingSettings'
|
|
9
|
+
|
|
10
|
+
interface SettingsTab {
|
|
11
|
+
id: string
|
|
12
|
+
label: string
|
|
13
|
+
icon?: React.ReactNode
|
|
14
|
+
component: React.ReactNode
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
interface SettingsPageProps {
|
|
18
|
+
tabs?: SettingsTab[]
|
|
19
|
+
extraTabs?: SettingsTab[]
|
|
20
|
+
defaultTab?: string
|
|
21
|
+
className?: string
|
|
22
|
+
beforeContent?: React.ReactNode
|
|
23
|
+
afterContent?: React.ReactNode
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export function SettingsPage({
|
|
27
|
+
tabs,
|
|
28
|
+
extraTabs,
|
|
29
|
+
defaultTab,
|
|
30
|
+
className,
|
|
31
|
+
beforeContent,
|
|
32
|
+
afterContent,
|
|
33
|
+
}: SettingsPageProps) {
|
|
34
|
+
const { t } = useTranslation()
|
|
35
|
+
|
|
36
|
+
const defaultTabs: SettingsTab[] = React.useMemo(() => [
|
|
37
|
+
{ id: 'general', label: t('settings.general'), icon: <Building2 className="h-4 w-4" />, component: <ConnectedCompanySettings /> },
|
|
38
|
+
{ id: 'profile', label: t('settings.profile'), icon: <User className="h-4 w-4" />, component: <ConnectedUserProfile /> },
|
|
39
|
+
{ id: 'security', label: t('settings.security'), icon: <Shield className="h-4 w-4" />, component: <ConnectedSecuritySettings /> },
|
|
40
|
+
{ id: 'branding', label: t('settings.branding'), icon: <Palette className="h-4 w-4" />, component: <ConnectedBrandingSettings /> },
|
|
41
|
+
], [t])
|
|
42
|
+
|
|
43
|
+
const baseTabs = tabs ?? defaultTabs
|
|
44
|
+
const resolvedTabs = extraTabs ? [...baseTabs, ...extraTabs] : baseTabs
|
|
45
|
+
|
|
46
|
+
// Detect active tab from URL: /settings/financial/... → 'financial'
|
|
47
|
+
function getTabFromHash(): string | null {
|
|
48
|
+
const hash = typeof window !== 'undefined' ? window.location.hash.slice(1) || '/' : '/'
|
|
49
|
+
if (hash.startsWith('/settings/')) {
|
|
50
|
+
const tabId = hash.slice('/settings/'.length).split('/')[0]
|
|
51
|
+
if (resolvedTabs.find((t) => t.id === tabId)) return tabId
|
|
52
|
+
}
|
|
53
|
+
return null
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
const [activeTab, setActiveTab] = React.useState<string | null>(() => {
|
|
57
|
+
return getTabFromHash() ?? defaultTab ?? null
|
|
58
|
+
})
|
|
59
|
+
|
|
60
|
+
// Sync tab from hash changes
|
|
61
|
+
React.useEffect(() => {
|
|
62
|
+
const handler = () => {
|
|
63
|
+
const tab = getTabFromHash()
|
|
64
|
+
if (tab) setActiveTab(tab)
|
|
65
|
+
}
|
|
66
|
+
window.addEventListener('hashchange', handler)
|
|
67
|
+
return () => window.removeEventListener('hashchange', handler)
|
|
68
|
+
}, [resolvedTabs])
|
|
69
|
+
|
|
70
|
+
// Desktop always shows a selected tab; when none is chosen yet (landing on
|
|
71
|
+
// /settings), default to the first tab — 'general'. Mobile keeps its
|
|
72
|
+
// list-first master/detail behaviour (driven by the raw activeTab below).
|
|
73
|
+
const desktopTab = activeTab ?? resolvedTabs[0]?.id ?? null
|
|
74
|
+
const activeContent = desktopTab ? resolvedTabs.find((t) => t.id === desktopTab)?.component : null
|
|
75
|
+
const activeLabel = activeTab ? resolvedTabs.find((t) => t.id === activeTab)?.label : null
|
|
76
|
+
|
|
77
|
+
// On mobile: show list OR content. On desktop: show both.
|
|
78
|
+
const showingContent = activeTab !== null
|
|
79
|
+
|
|
80
|
+
const handleSelectTab = (tabId: string) => {
|
|
81
|
+
setActiveTab(tabId)
|
|
82
|
+
window.location.hash = `/settings/${tabId}`
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
const handleBack = () => {
|
|
86
|
+
setActiveTab(null)
|
|
87
|
+
window.location.hash = '/settings'
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
const renderNav = (activeId: string | null) => (
|
|
91
|
+
<ul className="space-y-0.5">
|
|
92
|
+
{resolvedTabs.map((tab, i) => (
|
|
93
|
+
<React.Fragment key={tab.id}>
|
|
94
|
+
{(tab as any).isPlugin && (i === 0 || !(resolvedTabs[i - 1] as any)?.isPlugin) && (
|
|
95
|
+
<li className="py-2">
|
|
96
|
+
<div className="border-t" />
|
|
97
|
+
<p className="px-3 pt-2 text-[10px] font-semibold uppercase tracking-wider text-muted-foreground">{t('common.plugins')}</p>
|
|
98
|
+
</li>
|
|
99
|
+
)}
|
|
100
|
+
<li>
|
|
101
|
+
<button
|
|
102
|
+
onClick={() => handleSelectTab(tab.id)}
|
|
103
|
+
className={cn(
|
|
104
|
+
'flex w-full items-center gap-2 rounded-md px-3 py-2 text-sm font-medium transition-colors',
|
|
105
|
+
activeId === tab.id
|
|
106
|
+
? 'bg-muted text-foreground'
|
|
107
|
+
: 'text-muted-foreground hover:bg-muted/50 hover:text-foreground'
|
|
108
|
+
)}
|
|
109
|
+
>
|
|
110
|
+
{tab.icon && <span className="shrink-0">{tab.icon}</span>}
|
|
111
|
+
<span className="flex-1 text-left">{tab.label}</span>
|
|
112
|
+
{/* Mobile: show chevron */}
|
|
113
|
+
<ChevronRight className="h-4 w-4 text-muted-foreground/50 md:hidden" />
|
|
114
|
+
</button>
|
|
115
|
+
</li>
|
|
116
|
+
</React.Fragment>
|
|
117
|
+
))}
|
|
118
|
+
</ul>
|
|
119
|
+
)
|
|
120
|
+
|
|
121
|
+
return (
|
|
122
|
+
<div className={cn('space-y-6', className)}>
|
|
123
|
+
{/* Mobile: back button + tab title when viewing content */}
|
|
124
|
+
{showingContent && (
|
|
125
|
+
<button
|
|
126
|
+
onClick={handleBack}
|
|
127
|
+
className="flex items-center gap-1.5 text-sm text-muted-foreground hover:text-foreground transition-colors md:hidden"
|
|
128
|
+
>
|
|
129
|
+
<ArrowLeft className="h-4 w-4" />
|
|
130
|
+
{t('settings.title')}
|
|
131
|
+
</button>
|
|
132
|
+
)}
|
|
133
|
+
|
|
134
|
+
{/* Header — hidden on mobile when viewing content */}
|
|
135
|
+
<div className={cn(showingContent && 'hidden md:block')}>
|
|
136
|
+
<h1 className="text-2xl font-bold tracking-tight">{t('settings.title')}</h1>
|
|
137
|
+
<p className="text-sm text-muted-foreground">
|
|
138
|
+
{t('settings.subtitle')}
|
|
139
|
+
</p>
|
|
140
|
+
</div>
|
|
141
|
+
|
|
142
|
+
{beforeContent}
|
|
143
|
+
|
|
144
|
+
{/* Desktop: side-by-side layout */}
|
|
145
|
+
<div className="hidden md:flex md:gap-6">
|
|
146
|
+
<nav className="w-52 shrink-0">{renderNav(desktopTab)}</nav>
|
|
147
|
+
<div className="min-w-0 flex-1">{activeContent ?? resolvedTabs[0]?.component}</div>
|
|
148
|
+
</div>
|
|
149
|
+
|
|
150
|
+
{/* Mobile: list or content, not both */}
|
|
151
|
+
<div className="md:hidden">
|
|
152
|
+
{showingContent ? (
|
|
153
|
+
<div>{activeContent}</div>
|
|
154
|
+
) : (
|
|
155
|
+
<nav>{renderNav(activeTab)}</nav>
|
|
156
|
+
)}
|
|
157
|
+
</div>
|
|
158
|
+
|
|
159
|
+
{afterContent}
|
|
160
|
+
</div>
|
|
161
|
+
)
|
|
162
|
+
}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Camera, Save } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { Button } from '@fayz-ai/ui'
|
|
5
|
+
import { Input } from '@fayz-ai/ui'
|
|
6
|
+
import { Avatar, AvatarFallback, AvatarImage } from '@fayz-ai/ui'
|
|
7
|
+
import { Card, CardContent, CardDescription, CardFooter, CardHeader, CardTitle } from '@fayz-ai/ui'
|
|
8
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
9
|
+
import type { AuthUser } from '../../types'
|
|
10
|
+
|
|
11
|
+
interface UserProfileProps {
|
|
12
|
+
user?: AuthUser | null
|
|
13
|
+
onSave?: (data: { fullName: string }) => void
|
|
14
|
+
onAvatarChange?: (file: File) => void
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
function getInitials(name: string): string {
|
|
18
|
+
return name
|
|
19
|
+
.split(' ')
|
|
20
|
+
.map((part) => part[0])
|
|
21
|
+
.filter(Boolean)
|
|
22
|
+
.slice(0, 2)
|
|
23
|
+
.join('')
|
|
24
|
+
.toUpperCase()
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export function UserProfile({ user, onSave, onAvatarChange }: UserProfileProps) {
|
|
28
|
+
const { t } = useTranslation()
|
|
29
|
+
const [fullName, setFullName] = React.useState(user?.fullName ?? '')
|
|
30
|
+
const [saving, setSaving] = React.useState(false)
|
|
31
|
+
const fileInputRef = React.useRef<HTMLInputElement>(null)
|
|
32
|
+
|
|
33
|
+
React.useEffect(() => {
|
|
34
|
+
if (user) {
|
|
35
|
+
setFullName(user.fullName)
|
|
36
|
+
}
|
|
37
|
+
}, [user])
|
|
38
|
+
|
|
39
|
+
const handleAvatarClick = () => {
|
|
40
|
+
fileInputRef.current?.click()
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
const handleFileChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
44
|
+
const file = e.target.files?.[0]
|
|
45
|
+
if (file && onAvatarChange) {
|
|
46
|
+
onAvatarChange(file)
|
|
47
|
+
}
|
|
48
|
+
if (e.target) {
|
|
49
|
+
e.target.value = ''
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
const handleSubmit = async (e: React.FormEvent) => {
|
|
54
|
+
e.preventDefault()
|
|
55
|
+
if (!onSave) return
|
|
56
|
+
|
|
57
|
+
setSaving(true)
|
|
58
|
+
try {
|
|
59
|
+
await onSave({ fullName })
|
|
60
|
+
} finally {
|
|
61
|
+
setSaving(false)
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
return (
|
|
66
|
+
<form onSubmit={handleSubmit} className="space-y-6">
|
|
67
|
+
<Card>
|
|
68
|
+
<CardHeader>
|
|
69
|
+
<CardTitle className="text-lg">{t('settings.profile.title')}</CardTitle>
|
|
70
|
+
<CardDescription>
|
|
71
|
+
{t('settings.profile.subtitle')}
|
|
72
|
+
</CardDescription>
|
|
73
|
+
</CardHeader>
|
|
74
|
+
|
|
75
|
+
<CardContent className="space-y-6">
|
|
76
|
+
<div className="flex items-center gap-6">
|
|
77
|
+
<div className="relative">
|
|
78
|
+
<Avatar className="h-20 w-20">
|
|
79
|
+
<AvatarImage src={user?.avatarUrl} alt={user?.fullName} />
|
|
80
|
+
<AvatarFallback className="text-lg">
|
|
81
|
+
{user?.fullName ? getInitials(user.fullName) : '?'}
|
|
82
|
+
</AvatarFallback>
|
|
83
|
+
</Avatar>
|
|
84
|
+
{onAvatarChange && (
|
|
85
|
+
<button
|
|
86
|
+
type="button"
|
|
87
|
+
onClick={handleAvatarClick}
|
|
88
|
+
className={cn(
|
|
89
|
+
'absolute -bottom-1 -right-1 flex h-8 w-8 items-center justify-center rounded-full',
|
|
90
|
+
'border-2 border-background bg-primary text-primary-foreground',
|
|
91
|
+
'transition-colors hover:bg-primary/90'
|
|
92
|
+
)}
|
|
93
|
+
aria-label="Change avatar"
|
|
94
|
+
>
|
|
95
|
+
<Camera className="h-3.5 w-3.5" />
|
|
96
|
+
</button>
|
|
97
|
+
)}
|
|
98
|
+
<input
|
|
99
|
+
ref={fileInputRef}
|
|
100
|
+
type="file"
|
|
101
|
+
accept="image/*"
|
|
102
|
+
className="hidden"
|
|
103
|
+
onChange={handleFileChange}
|
|
104
|
+
/>
|
|
105
|
+
</div>
|
|
106
|
+
|
|
107
|
+
<div className="space-y-1">
|
|
108
|
+
<p className="font-medium">{user?.fullName ?? 'User'}</p>
|
|
109
|
+
<p className="text-sm text-muted-foreground">{user?.email}</p>
|
|
110
|
+
</div>
|
|
111
|
+
</div>
|
|
112
|
+
|
|
113
|
+
<div className="space-y-2">
|
|
114
|
+
<label htmlFor="full-name" className="text-sm font-medium">
|
|
115
|
+
{t('settings.profile.fullName')}
|
|
116
|
+
</label>
|
|
117
|
+
<Input
|
|
118
|
+
id="full-name"
|
|
119
|
+
value={fullName}
|
|
120
|
+
onChange={(e) => setFullName(e.target.value)}
|
|
121
|
+
placeholder="John Doe"
|
|
122
|
+
/>
|
|
123
|
+
</div>
|
|
124
|
+
|
|
125
|
+
<div className="space-y-2">
|
|
126
|
+
<label htmlFor="email" className="text-sm font-medium">
|
|
127
|
+
{t('settings.profile.email')}
|
|
128
|
+
</label>
|
|
129
|
+
<Input
|
|
130
|
+
id="email"
|
|
131
|
+
type="email"
|
|
132
|
+
value={user?.email ?? ''}
|
|
133
|
+
disabled
|
|
134
|
+
className="bg-muted"
|
|
135
|
+
/>
|
|
136
|
+
<p className="text-xs text-muted-foreground">
|
|
137
|
+
{t('settings.profile.emailHelp')}
|
|
138
|
+
</p>
|
|
139
|
+
</div>
|
|
140
|
+
</CardContent>
|
|
141
|
+
|
|
142
|
+
<CardFooter className="justify-end">
|
|
143
|
+
<Button type="submit" disabled={saving}>
|
|
144
|
+
<Save className="mr-2 h-4 w-4" />
|
|
145
|
+
{saving ? t('common.saving') : t('settings.profile.saveProfile')}
|
|
146
|
+
</Button>
|
|
147
|
+
</CardFooter>
|
|
148
|
+
</Card>
|
|
149
|
+
</form>
|
|
150
|
+
)
|
|
151
|
+
}
|