@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,170 @@
|
|
|
1
|
+
import React, { useState } from 'react'
|
|
2
|
+
import { cn } from '../../lib/cn'
|
|
3
|
+
import { useAuth } from '../../hooks/useAuth'
|
|
4
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
5
|
+
|
|
6
|
+
interface SignupFormProps {
|
|
7
|
+
onSuccess?: () => void
|
|
8
|
+
termsUrl?: string
|
|
9
|
+
privacyUrl?: string
|
|
10
|
+
className?: string
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function SignupForm({
|
|
14
|
+
onSuccess,
|
|
15
|
+
termsUrl,
|
|
16
|
+
privacyUrl,
|
|
17
|
+
className,
|
|
18
|
+
}: SignupFormProps) {
|
|
19
|
+
const { signUp, loading } = useAuth()
|
|
20
|
+
const { t } = useTranslation()
|
|
21
|
+
const [fullName, setFullName] = useState('')
|
|
22
|
+
const [email, setEmail] = useState('')
|
|
23
|
+
const [password, setPassword] = useState('')
|
|
24
|
+
const [confirmPassword, setConfirmPassword] = useState('')
|
|
25
|
+
const [error, setError] = useState<string | null>(null)
|
|
26
|
+
const [success, setSuccess] = useState<string | null>(null)
|
|
27
|
+
|
|
28
|
+
const handleSubmit = async (e: React.FormEvent) => {
|
|
29
|
+
e.preventDefault()
|
|
30
|
+
setError(null)
|
|
31
|
+
setSuccess(null)
|
|
32
|
+
|
|
33
|
+
if (password !== confirmPassword) {
|
|
34
|
+
setError(t('auth.signup.passwordsMismatch'))
|
|
35
|
+
return
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
if (password.length < 8) {
|
|
39
|
+
setError(t('auth.signup.passwordMinLength'))
|
|
40
|
+
return
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
try {
|
|
44
|
+
const result = await signUp(email, password, fullName)
|
|
45
|
+
if (result.requiresEmailVerification) {
|
|
46
|
+
setSuccess(t('auth.signup.emailVerification', { email }))
|
|
47
|
+
setPassword('')
|
|
48
|
+
setConfirmPassword('')
|
|
49
|
+
return
|
|
50
|
+
}
|
|
51
|
+
onSuccess?.()
|
|
52
|
+
} catch (err: any) {
|
|
53
|
+
setError(err.message ?? t('auth.signup.failedDefault'))
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
const inputClass =
|
|
58
|
+
'flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50'
|
|
59
|
+
|
|
60
|
+
return (
|
|
61
|
+
<form
|
|
62
|
+
onSubmit={handleSubmit}
|
|
63
|
+
className={cn('flex flex-col gap-4 w-full max-w-sm', className)}
|
|
64
|
+
>
|
|
65
|
+
{error && (
|
|
66
|
+
<div className="rounded-md bg-destructive/10 border border-destructive/20 px-4 py-3 text-sm text-destructive">
|
|
67
|
+
{error}
|
|
68
|
+
</div>
|
|
69
|
+
)}
|
|
70
|
+
|
|
71
|
+
{success && (
|
|
72
|
+
<div className="rounded-md border border-success/20 bg-success/10 px-4 py-3 text-sm text-success">
|
|
73
|
+
{success}
|
|
74
|
+
</div>
|
|
75
|
+
)}
|
|
76
|
+
|
|
77
|
+
<div className="flex flex-col gap-1.5">
|
|
78
|
+
<label htmlFor="signup-name" className="text-sm font-medium text-foreground">
|
|
79
|
+
{t('auth.signup.fullName')}
|
|
80
|
+
</label>
|
|
81
|
+
<input
|
|
82
|
+
id="signup-name"
|
|
83
|
+
type="text"
|
|
84
|
+
value={fullName}
|
|
85
|
+
onChange={(e) => setFullName(e.target.value)}
|
|
86
|
+
placeholder={t('auth.signup.placeholder.name')}
|
|
87
|
+
required
|
|
88
|
+
autoComplete="name"
|
|
89
|
+
className={inputClass}
|
|
90
|
+
/>
|
|
91
|
+
</div>
|
|
92
|
+
|
|
93
|
+
<div className="flex flex-col gap-1.5">
|
|
94
|
+
<label htmlFor="signup-email" className="text-sm font-medium text-foreground">
|
|
95
|
+
{t('auth.email')}
|
|
96
|
+
</label>
|
|
97
|
+
<input
|
|
98
|
+
id="signup-email"
|
|
99
|
+
type="email"
|
|
100
|
+
value={email}
|
|
101
|
+
onChange={(e) => setEmail(e.target.value)}
|
|
102
|
+
placeholder={t('auth.login.placeholder.email')}
|
|
103
|
+
required
|
|
104
|
+
autoComplete="email"
|
|
105
|
+
className={inputClass}
|
|
106
|
+
/>
|
|
107
|
+
</div>
|
|
108
|
+
|
|
109
|
+
<div className="flex flex-col gap-1.5">
|
|
110
|
+
<label htmlFor="signup-password" className="text-sm font-medium text-foreground">
|
|
111
|
+
{t('auth.password')}
|
|
112
|
+
</label>
|
|
113
|
+
<input
|
|
114
|
+
id="signup-password"
|
|
115
|
+
type="password"
|
|
116
|
+
value={password}
|
|
117
|
+
onChange={(e) => setPassword(e.target.value)}
|
|
118
|
+
placeholder={t('auth.signup.placeholder.password')}
|
|
119
|
+
required
|
|
120
|
+
minLength={8}
|
|
121
|
+
autoComplete="new-password"
|
|
122
|
+
className={inputClass}
|
|
123
|
+
/>
|
|
124
|
+
</div>
|
|
125
|
+
|
|
126
|
+
<div className="flex flex-col gap-1.5">
|
|
127
|
+
<label htmlFor="signup-confirm" className="text-sm font-medium text-foreground">
|
|
128
|
+
{t('auth.signup.confirmPassword')}
|
|
129
|
+
</label>
|
|
130
|
+
<input
|
|
131
|
+
id="signup-confirm"
|
|
132
|
+
type="password"
|
|
133
|
+
value={confirmPassword}
|
|
134
|
+
onChange={(e) => setConfirmPassword(e.target.value)}
|
|
135
|
+
placeholder={t('auth.signup.placeholder.confirm')}
|
|
136
|
+
required
|
|
137
|
+
minLength={8}
|
|
138
|
+
autoComplete="new-password"
|
|
139
|
+
className={inputClass}
|
|
140
|
+
/>
|
|
141
|
+
</div>
|
|
142
|
+
|
|
143
|
+
{(termsUrl || privacyUrl) && (
|
|
144
|
+
<p className="text-xs text-muted-foreground">
|
|
145
|
+
{t('auth.signup.termsAgreement')}{' '}
|
|
146
|
+
{termsUrl && (
|
|
147
|
+
<a href={termsUrl} target="_blank" rel="noopener noreferrer" className="underline hover:text-foreground">
|
|
148
|
+
{t('auth.signup.termsOfService')}
|
|
149
|
+
</a>
|
|
150
|
+
)}
|
|
151
|
+
{termsUrl && privacyUrl && ' and '}
|
|
152
|
+
{privacyUrl && (
|
|
153
|
+
<a href={privacyUrl} target="_blank" rel="noopener noreferrer" className="underline hover:text-foreground">
|
|
154
|
+
{t('auth.signup.privacyPolicy')}
|
|
155
|
+
</a>
|
|
156
|
+
)}
|
|
157
|
+
.
|
|
158
|
+
</p>
|
|
159
|
+
)}
|
|
160
|
+
|
|
161
|
+
<button
|
|
162
|
+
type="submit"
|
|
163
|
+
disabled={loading}
|
|
164
|
+
className="inline-flex items-center justify-center h-10 px-4 py-2 rounded-md bg-primary border border-primary text-primary-foreground font-medium text-sm hover:bg-primary/90 shadow-button-primary active:shadow-button-inset transition-colors disabled:opacity-50 disabled:pointer-events-none"
|
|
165
|
+
>
|
|
166
|
+
{loading ? t('auth.signup.creating') : t('auth.signup.title')}
|
|
167
|
+
</button>
|
|
168
|
+
</form>
|
|
169
|
+
)
|
|
170
|
+
}
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
import React, { useState } from 'react'
|
|
2
|
+
import { LoginForm } from './LoginForm'
|
|
3
|
+
import { SignupForm } from './SignupForm'
|
|
4
|
+
import { RecoveryForm } from './RecoveryForm'
|
|
5
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
6
|
+
import type { AuthProvider } from '../../types'
|
|
7
|
+
|
|
8
|
+
interface SplitLoginProps {
|
|
9
|
+
appName: string
|
|
10
|
+
logo?: React.ReactNode
|
|
11
|
+
tagline?: string
|
|
12
|
+
description?: string
|
|
13
|
+
showOAuth?: boolean
|
|
14
|
+
oauthProviders?: Exclude<AuthProvider, 'email'>[]
|
|
15
|
+
onSuccess?: () => void
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
type View = 'login' | 'signup' | 'recovery'
|
|
19
|
+
|
|
20
|
+
export function SplitLogin({ appName, logo, tagline, description, showOAuth, oauthProviders, onSuccess }: SplitLoginProps) {
|
|
21
|
+
const [view, setView] = useState<View>('login')
|
|
22
|
+
const { t } = useTranslation()
|
|
23
|
+
|
|
24
|
+
const heading = view === 'login' ? t('auth.login.title') : view === 'signup' ? t('auth.signup.title') : t('auth.recovery.title')
|
|
25
|
+
const subtitle = view === 'login'
|
|
26
|
+
? t('auth.login.enterCredentials')
|
|
27
|
+
: view === 'signup'
|
|
28
|
+
? t('auth.signup.getStarted')
|
|
29
|
+
: t('auth.recovery.resetSubtitle')
|
|
30
|
+
|
|
31
|
+
return (
|
|
32
|
+
<div className="flex min-h-screen bg-sidebar">
|
|
33
|
+
{/* Left — Brand Panel (hidden on mobile) */}
|
|
34
|
+
<div className="relative hidden w-1/2 overflow-hidden lg:block">
|
|
35
|
+
{/* Gradient overlay */}
|
|
36
|
+
<div className="absolute inset-0 bg-gradient-to-br from-sidebar-accent/50 via-transparent to-sidebar/80" />
|
|
37
|
+
|
|
38
|
+
{/* Decorative circles */}
|
|
39
|
+
<div className="absolute -left-20 -top-20 h-80 w-80 rounded-full bg-primary/10" />
|
|
40
|
+
<div className="absolute -bottom-10 -right-10 h-60 w-60 rounded-full bg-primary/5" />
|
|
41
|
+
<div className="absolute right-1/4 top-1/3 h-40 w-40 rounded-full bg-primary/5" />
|
|
42
|
+
|
|
43
|
+
{/* Content */}
|
|
44
|
+
<div className="relative flex h-full flex-col justify-end p-12">
|
|
45
|
+
<div className="space-y-4">
|
|
46
|
+
<p className="text-sm font-medium text-sidebar-muted">{appName}</p>
|
|
47
|
+
<h2 className="text-3xl font-bold text-sidebar-foreground">
|
|
48
|
+
{tagline ?? t('auth.login.welcomeTo', { appName })}
|
|
49
|
+
</h2>
|
|
50
|
+
{description && (
|
|
51
|
+
<p className="max-w-md text-sm leading-relaxed text-sidebar-muted">
|
|
52
|
+
{description}
|
|
53
|
+
</p>
|
|
54
|
+
)}
|
|
55
|
+
|
|
56
|
+
{/* Social proof */}
|
|
57
|
+
<div className="mt-8 flex items-center gap-3 rounded-xl bg-sidebar-accent/40 p-4">
|
|
58
|
+
<div className="flex -space-x-2">
|
|
59
|
+
{['bg-primary', 'bg-success', 'bg-warning'].map((bg, i) => (
|
|
60
|
+
<div key={i} className={`h-8 w-8 rounded-full ${bg} border-2 border-sidebar`} />
|
|
61
|
+
))}
|
|
62
|
+
</div>
|
|
63
|
+
<p className="text-xs text-sidebar-foreground/80">
|
|
64
|
+
{t('auth.login.socialProof', { appName })}
|
|
65
|
+
</p>
|
|
66
|
+
</div>
|
|
67
|
+
</div>
|
|
68
|
+
</div>
|
|
69
|
+
</div>
|
|
70
|
+
|
|
71
|
+
{/* Right — Form in rounded frame */}
|
|
72
|
+
<div className="flex flex-1 flex-col p-3 pl-0 lg:pl-3">
|
|
73
|
+
<div className="flex flex-1 flex-col justify-center rounded-[1.25rem] bg-background px-6 py-12 lg:px-16">
|
|
74
|
+
<div className="mx-auto w-full max-w-sm">
|
|
75
|
+
{/* Logo */}
|
|
76
|
+
<div className="mb-8">
|
|
77
|
+
{logo}
|
|
78
|
+
</div>
|
|
79
|
+
|
|
80
|
+
{/* Heading */}
|
|
81
|
+
<div className="mb-8">
|
|
82
|
+
<h1 className="text-2xl font-bold text-foreground">{heading}</h1>
|
|
83
|
+
<p className="mt-1 text-sm text-muted-foreground">{subtitle}</p>
|
|
84
|
+
</div>
|
|
85
|
+
|
|
86
|
+
{/* Form */}
|
|
87
|
+
{view === 'login' && (
|
|
88
|
+
<div className="space-y-6">
|
|
89
|
+
<LoginForm
|
|
90
|
+
onSuccess={onSuccess}
|
|
91
|
+
onForgotPassword={() => setView('recovery')}
|
|
92
|
+
showOAuth={showOAuth}
|
|
93
|
+
oauthProviders={oauthProviders}
|
|
94
|
+
/>
|
|
95
|
+
<p className="text-center text-sm text-muted-foreground">
|
|
96
|
+
{t('auth.login.noAccount')}{' '}
|
|
97
|
+
<button type="button" onClick={() => setView('signup')} className="font-medium text-primary hover:underline">
|
|
98
|
+
{t('auth.signUp')}
|
|
99
|
+
</button>
|
|
100
|
+
</p>
|
|
101
|
+
</div>
|
|
102
|
+
)}
|
|
103
|
+
|
|
104
|
+
{view === 'signup' && (
|
|
105
|
+
<div className="space-y-6">
|
|
106
|
+
<SignupForm onSuccess={onSuccess} />
|
|
107
|
+
<p className="text-center text-sm text-muted-foreground">
|
|
108
|
+
{t('auth.signup.hasAccount')}{' '}
|
|
109
|
+
<button type="button" onClick={() => setView('login')} className="font-medium text-primary hover:underline">
|
|
110
|
+
{t('auth.signIn')}
|
|
111
|
+
</button>
|
|
112
|
+
</p>
|
|
113
|
+
</div>
|
|
114
|
+
)}
|
|
115
|
+
|
|
116
|
+
{view === 'recovery' && (
|
|
117
|
+
<RecoveryForm onBackToLogin={() => setView('login')} />
|
|
118
|
+
)}
|
|
119
|
+
</div>
|
|
120
|
+
</div>
|
|
121
|
+
</div>
|
|
122
|
+
</div>
|
|
123
|
+
)
|
|
124
|
+
}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export { LoginForm } from './LoginForm'
|
|
2
|
+
export { SignupForm } from './SignupForm'
|
|
3
|
+
export { RecoveryForm } from './RecoveryForm'
|
|
4
|
+
export { OAuthButtons } from './OAuthButtons'
|
|
5
|
+
export { ProtectedRoute } from './ProtectedRoute'
|
|
6
|
+
export { LoginPage } from './LoginPage'
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { PlanSelector } from './PlanSelector'
|
|
3
|
+
import { SubscriptionCard } from './SubscriptionCard'
|
|
4
|
+
import { useBillingStore } from '../../stores/billing.store'
|
|
5
|
+
import { useBilling } from '../../hooks/useBilling'
|
|
6
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
7
|
+
import type { PlanInterval } from '../../types'
|
|
8
|
+
|
|
9
|
+
interface BillingPageProps {
|
|
10
|
+
className?: string
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
export function BillingPage({ className }: BillingPageProps) {
|
|
14
|
+
const { plans, subscription, loading } = useBillingStore()
|
|
15
|
+
const billing = useBilling()
|
|
16
|
+
const { t } = useTranslation()
|
|
17
|
+
const [interval, setInterval] = React.useState<PlanInterval>('monthly')
|
|
18
|
+
|
|
19
|
+
// Fetch subscription on mount
|
|
20
|
+
React.useEffect(() => {
|
|
21
|
+
billing.fetchSubscription()
|
|
22
|
+
}, [])
|
|
23
|
+
|
|
24
|
+
const currentPlan = plans.find((p) => p.id === subscription?.planId)
|
|
25
|
+
|
|
26
|
+
return (
|
|
27
|
+
<div className={className ?? 'mx-auto max-w-5xl space-y-8'}>
|
|
28
|
+
<div>
|
|
29
|
+
<h1 className="text-2xl font-bold tracking-tight">{t('billing.title')}</h1>
|
|
30
|
+
<p className="text-sm text-muted-foreground">
|
|
31
|
+
{t('billing.subtitle')}
|
|
32
|
+
</p>
|
|
33
|
+
</div>
|
|
34
|
+
|
|
35
|
+
{subscription && (
|
|
36
|
+
<SubscriptionCard
|
|
37
|
+
subscription={subscription as any}
|
|
38
|
+
plan={currentPlan as any}
|
|
39
|
+
onCancel={billing.cancelSubscription}
|
|
40
|
+
loading={loading}
|
|
41
|
+
/>
|
|
42
|
+
)}
|
|
43
|
+
|
|
44
|
+
{plans.length > 0 && (
|
|
45
|
+
<PlanSelector
|
|
46
|
+
plans={plans as any}
|
|
47
|
+
currentPlanId={subscription?.planId}
|
|
48
|
+
interval={interval}
|
|
49
|
+
onIntervalChange={setInterval}
|
|
50
|
+
onSelectPlan={(planId) => billing.subscribe(planId, interval)}
|
|
51
|
+
loading={loading}
|
|
52
|
+
/>
|
|
53
|
+
)}
|
|
54
|
+
</div>
|
|
55
|
+
)
|
|
56
|
+
}
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Download, FileText } from 'lucide-react'
|
|
3
|
+
import { cn } from '../../lib/cn'
|
|
4
|
+
import { Badge } from '@fayz-ai/ui'
|
|
5
|
+
import { Button } from '@fayz-ai/ui'
|
|
6
|
+
import { Skeleton } from '@fayz-ai/ui'
|
|
7
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
8
|
+
import type { Invoice } from '../../types'
|
|
9
|
+
|
|
10
|
+
interface InvoiceListProps {
|
|
11
|
+
invoices: Invoice[]
|
|
12
|
+
loading?: boolean
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
const STATUS_VARIANT: Record<string, 'default' | 'secondary' | 'destructive' | 'outline'> = {
|
|
16
|
+
paid: 'default',
|
|
17
|
+
open: 'secondary',
|
|
18
|
+
draft: 'outline',
|
|
19
|
+
void: 'outline',
|
|
20
|
+
uncollectible: 'destructive',
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
function formatCurrency(amount: number, currency: string): string {
|
|
24
|
+
return new Intl.NumberFormat('en-US', {
|
|
25
|
+
style: 'currency',
|
|
26
|
+
currency: currency.toUpperCase(),
|
|
27
|
+
}).format(amount / 100)
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
function formatDate(dateString: string): string {
|
|
31
|
+
return new Date(dateString).toLocaleDateString('en-US', {
|
|
32
|
+
month: 'short',
|
|
33
|
+
day: 'numeric',
|
|
34
|
+
year: 'numeric',
|
|
35
|
+
})
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
export function InvoiceList({ invoices, loading }: InvoiceListProps) {
|
|
39
|
+
const { t } = useTranslation()
|
|
40
|
+
if (loading) {
|
|
41
|
+
return (
|
|
42
|
+
<div className="space-y-3">
|
|
43
|
+
{Array.from({ length: 3 }).map((_, i) => (
|
|
44
|
+
<Skeleton key={i} className="h-16 w-full" />
|
|
45
|
+
))}
|
|
46
|
+
</div>
|
|
47
|
+
)
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
if (invoices.length === 0) {
|
|
51
|
+
return (
|
|
52
|
+
<div className="flex flex-col items-center justify-center py-12 text-center">
|
|
53
|
+
<FileText className="mb-3 h-10 w-10 text-muted-foreground/50" />
|
|
54
|
+
<p className="text-sm text-muted-foreground">{t('billing.invoice.noInvoices')}</p>
|
|
55
|
+
</div>
|
|
56
|
+
)
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
return (
|
|
60
|
+
<div className="overflow-hidden rounded-md border">
|
|
61
|
+
<table className="w-full">
|
|
62
|
+
<thead>
|
|
63
|
+
<tr className="border-b bg-muted/50">
|
|
64
|
+
<th className="px-4 py-3 text-left text-xs font-medium uppercase tracking-wider text-muted-foreground">
|
|
65
|
+
{t('billing.invoice.date')}
|
|
66
|
+
</th>
|
|
67
|
+
<th className="px-4 py-3 text-left text-xs font-medium uppercase tracking-wider text-muted-foreground">
|
|
68
|
+
{t('billing.invoice.amount')}
|
|
69
|
+
</th>
|
|
70
|
+
<th className="px-4 py-3 text-left text-xs font-medium uppercase tracking-wider text-muted-foreground">
|
|
71
|
+
{t('billing.invoice.status')}
|
|
72
|
+
</th>
|
|
73
|
+
<th className="px-4 py-3 text-right text-xs font-medium uppercase tracking-wider text-muted-foreground">
|
|
74
|
+
{t('billing.invoice.invoice')}
|
|
75
|
+
</th>
|
|
76
|
+
</tr>
|
|
77
|
+
</thead>
|
|
78
|
+
<tbody className="divide-y">
|
|
79
|
+
{invoices.map((invoice) => (
|
|
80
|
+
<tr
|
|
81
|
+
key={invoice.id}
|
|
82
|
+
className={cn(
|
|
83
|
+
'transition-colors hover:bg-muted/30'
|
|
84
|
+
)}
|
|
85
|
+
>
|
|
86
|
+
<td className="px-4 py-3 text-sm">
|
|
87
|
+
{formatDate(invoice.createdAt)}
|
|
88
|
+
</td>
|
|
89
|
+
<td className="px-4 py-3 text-sm font-medium">
|
|
90
|
+
{formatCurrency(invoice.amount, invoice.currency)}
|
|
91
|
+
</td>
|
|
92
|
+
<td className="px-4 py-3">
|
|
93
|
+
<Badge variant={STATUS_VARIANT[invoice.status] ?? 'outline'}>
|
|
94
|
+
{invoice.status.charAt(0).toUpperCase() + invoice.status.slice(1)}
|
|
95
|
+
</Badge>
|
|
96
|
+
</td>
|
|
97
|
+
<td className="px-4 py-3 text-right">
|
|
98
|
+
{invoice.pdfUrl ? (
|
|
99
|
+
<Button
|
|
100
|
+
variant="ghost"
|
|
101
|
+
size="sm"
|
|
102
|
+
asChild
|
|
103
|
+
>
|
|
104
|
+
<a
|
|
105
|
+
href={invoice.pdfUrl}
|
|
106
|
+
target="_blank"
|
|
107
|
+
rel="noopener noreferrer"
|
|
108
|
+
>
|
|
109
|
+
<Download className="mr-1.5 h-3.5 w-3.5" />
|
|
110
|
+
PDF
|
|
111
|
+
</a>
|
|
112
|
+
</Button>
|
|
113
|
+
) : (
|
|
114
|
+
<span className="text-xs text-muted-foreground">--</span>
|
|
115
|
+
)}
|
|
116
|
+
</td>
|
|
117
|
+
</tr>
|
|
118
|
+
))}
|
|
119
|
+
</tbody>
|
|
120
|
+
</table>
|
|
121
|
+
</div>
|
|
122
|
+
)
|
|
123
|
+
}
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Lock } from 'lucide-react'
|
|
3
|
+
import { Button } from '@fayz-ai/ui'
|
|
4
|
+
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@fayz-ai/ui'
|
|
5
|
+
import { useBillingStore } from '../../stores/billing.store'
|
|
6
|
+
import { useTranslation } from '../../hooks/useTranslation'
|
|
7
|
+
|
|
8
|
+
interface PaywallGateProps {
|
|
9
|
+
requiredPlan?: string
|
|
10
|
+
requiredFeature?: string
|
|
11
|
+
children: React.ReactNode
|
|
12
|
+
fallback?: React.ReactNode
|
|
13
|
+
onUpgrade?: () => void
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export function PaywallGate({
|
|
17
|
+
requiredPlan,
|
|
18
|
+
requiredFeature,
|
|
19
|
+
children,
|
|
20
|
+
fallback,
|
|
21
|
+
onUpgrade,
|
|
22
|
+
}: PaywallGateProps) {
|
|
23
|
+
const { subscription, plans } = useBillingStore()
|
|
24
|
+
|
|
25
|
+
const hasAccess = React.useMemo(() => {
|
|
26
|
+
if (!requiredPlan && !requiredFeature) return true
|
|
27
|
+
if (!subscription || subscription.status === 'canceled') return false
|
|
28
|
+
|
|
29
|
+
if (requiredPlan) {
|
|
30
|
+
const currentPlan = plans.find((p) => p.id === subscription.planId)
|
|
31
|
+
const requiredPlanData = plans.find((p) => p.id === requiredPlan)
|
|
32
|
+
if (!currentPlan || !requiredPlanData) return false
|
|
33
|
+
|
|
34
|
+
const currentIndex = plans.indexOf(currentPlan)
|
|
35
|
+
const requiredIndex = plans.indexOf(requiredPlanData)
|
|
36
|
+
return currentIndex >= requiredIndex
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
if (requiredFeature) {
|
|
40
|
+
const currentPlan = plans.find((p) => p.id === subscription.planId)
|
|
41
|
+
if (!currentPlan) return false
|
|
42
|
+
return currentPlan.features.some(
|
|
43
|
+
(f) => f.toLowerCase().includes(requiredFeature.toLowerCase())
|
|
44
|
+
)
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
return true
|
|
48
|
+
}, [subscription, plans, requiredPlan, requiredFeature])
|
|
49
|
+
|
|
50
|
+
if (hasAccess) {
|
|
51
|
+
return <>{children}</>
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
if (fallback) {
|
|
55
|
+
return <>{fallback}</>
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
const { t } = useTranslation()
|
|
59
|
+
|
|
60
|
+
return (
|
|
61
|
+
<Card className="border-dashed">
|
|
62
|
+
<CardHeader className="text-center">
|
|
63
|
+
<div className="mx-auto mb-2 flex h-12 w-12 items-center justify-center rounded-full bg-muted">
|
|
64
|
+
<Lock className="h-6 w-6 text-muted-foreground" />
|
|
65
|
+
</div>
|
|
66
|
+
<CardTitle className="text-lg">{t('billing.paywall.upgradeRequired')}</CardTitle>
|
|
67
|
+
<CardDescription>
|
|
68
|
+
{requiredPlan
|
|
69
|
+
? t('billing.paywall.requiresPlan', { plan: requiredPlan })
|
|
70
|
+
: t('billing.paywall.notAvailable')}
|
|
71
|
+
</CardDescription>
|
|
72
|
+
</CardHeader>
|
|
73
|
+
<CardContent className="flex justify-center">
|
|
74
|
+
<Button onClick={onUpgrade}>
|
|
75
|
+
{t('billing.paywall.viewPlans')}
|
|
76
|
+
</Button>
|
|
77
|
+
</CardContent>
|
|
78
|
+
</Card>
|
|
79
|
+
)
|
|
80
|
+
}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
import * as React from 'react'
|
|
2
|
+
import { Check } 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 { Plan, PlanInterval } from '../../types'
|
|
9
|
+
|
|
10
|
+
interface PlanSelectorProps {
|
|
11
|
+
plans: Plan[]
|
|
12
|
+
currentPlanId?: string
|
|
13
|
+
interval: PlanInterval
|
|
14
|
+
onIntervalChange: (interval: PlanInterval) => void
|
|
15
|
+
onSelectPlan: (planId: string) => void
|
|
16
|
+
loading?: boolean
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
export function PlanSelector({
|
|
20
|
+
plans,
|
|
21
|
+
currentPlanId,
|
|
22
|
+
interval,
|
|
23
|
+
onIntervalChange,
|
|
24
|
+
onSelectPlan,
|
|
25
|
+
loading,
|
|
26
|
+
}: PlanSelectorProps) {
|
|
27
|
+
const { t } = useTranslation()
|
|
28
|
+
|
|
29
|
+
return (
|
|
30
|
+
<div className="space-y-8">
|
|
31
|
+
<div className="flex items-center justify-center gap-3">
|
|
32
|
+
<button
|
|
33
|
+
type="button"
|
|
34
|
+
className={cn(
|
|
35
|
+
'rounded-full px-4 py-2 text-sm font-medium transition-colors',
|
|
36
|
+
interval === 'monthly'
|
|
37
|
+
? 'bg-primary text-primary-foreground'
|
|
38
|
+
: 'text-muted-foreground hover:text-foreground'
|
|
39
|
+
)}
|
|
40
|
+
onClick={() => onIntervalChange('monthly')}
|
|
41
|
+
>
|
|
42
|
+
{t('billing.monthly')}
|
|
43
|
+
</button>
|
|
44
|
+
<button
|
|
45
|
+
type="button"
|
|
46
|
+
className={cn(
|
|
47
|
+
'rounded-full px-4 py-2 text-sm font-medium transition-colors',
|
|
48
|
+
interval === 'yearly'
|
|
49
|
+
? 'bg-primary text-primary-foreground'
|
|
50
|
+
: 'text-muted-foreground hover:text-foreground'
|
|
51
|
+
)}
|
|
52
|
+
onClick={() => onIntervalChange('yearly')}
|
|
53
|
+
>
|
|
54
|
+
{t('billing.yearly')}
|
|
55
|
+
<span className="ml-1.5 text-xs opacity-80">{t('billing.yearlySave')}</span>
|
|
56
|
+
</button>
|
|
57
|
+
</div>
|
|
58
|
+
|
|
59
|
+
<div className="grid gap-6 md:grid-cols-2 lg:grid-cols-3">
|
|
60
|
+
{plans.map((plan) => {
|
|
61
|
+
const isCurrent = plan.id === currentPlanId
|
|
62
|
+
const price = interval === 'monthly' ? (plan.prices?.monthly ?? plan.priceMonthly) : (plan.prices?.yearly ?? plan.priceYearly)
|
|
63
|
+
|
|
64
|
+
return (
|
|
65
|
+
<Card
|
|
66
|
+
key={plan.id}
|
|
67
|
+
className={cn(
|
|
68
|
+
'relative flex flex-col',
|
|
69
|
+
plan.popular && 'border-primary shadow-md'
|
|
70
|
+
)}
|
|
71
|
+
>
|
|
72
|
+
{plan.popular && (
|
|
73
|
+
<div className="absolute -top-3 left-1/2 -translate-x-1/2">
|
|
74
|
+
<Badge>{t('billing.mostPopular')}</Badge>
|
|
75
|
+
</div>
|
|
76
|
+
)}
|
|
77
|
+
|
|
78
|
+
<CardHeader>
|
|
79
|
+
<CardTitle className="text-xl">{plan.name}</CardTitle>
|
|
80
|
+
<CardDescription>{plan.description}</CardDescription>
|
|
81
|
+
</CardHeader>
|
|
82
|
+
|
|
83
|
+
<CardContent className="flex-1">
|
|
84
|
+
<div className="mb-6">
|
|
85
|
+
<span className="text-4xl font-bold">
|
|
86
|
+
${price}
|
|
87
|
+
</span>
|
|
88
|
+
<span className="text-muted-foreground">
|
|
89
|
+
/{interval === 'monthly' ? 'mo' : 'yr'}
|
|
90
|
+
</span>
|
|
91
|
+
</div>
|
|
92
|
+
|
|
93
|
+
<ul className="space-y-3">
|
|
94
|
+
{plan.features.map((feature) => (
|
|
95
|
+
<li key={feature} className="flex items-start gap-2 text-sm">
|
|
96
|
+
<Check className="mt-0.5 h-4 w-4 shrink-0 text-primary" />
|
|
97
|
+
<span>{feature}</span>
|
|
98
|
+
</li>
|
|
99
|
+
))}
|
|
100
|
+
</ul>
|
|
101
|
+
</CardContent>
|
|
102
|
+
|
|
103
|
+
<CardFooter>
|
|
104
|
+
{isCurrent ? (
|
|
105
|
+
<Button variant="outline" className="w-full" disabled>
|
|
106
|
+
{t('billing.currentPlan')}
|
|
107
|
+
</Button>
|
|
108
|
+
) : (
|
|
109
|
+
<Button
|
|
110
|
+
className="w-full"
|
|
111
|
+
variant={plan.popular ? 'default' : 'outline'}
|
|
112
|
+
disabled={loading}
|
|
113
|
+
onClick={() => onSelectPlan(plan.id)}
|
|
114
|
+
>
|
|
115
|
+
{loading ? t('billing.processing') : t('billing.getStarted')}
|
|
116
|
+
</Button>
|
|
117
|
+
)}
|
|
118
|
+
</CardFooter>
|
|
119
|
+
</Card>
|
|
120
|
+
)
|
|
121
|
+
})}
|
|
122
|
+
</div>
|
|
123
|
+
</div>
|
|
124
|
+
)
|
|
125
|
+
}
|