mirror of
http://88.130.71.182:3000/BlitTech/contexta_fe.git
synced 2026-06-12 23:23:22 +00:00
121 lines
7.4 KiB
TypeScript
121 lines
7.4 KiB
TypeScript
import React, { lazy, Suspense } from 'react'
|
|
import { Routes, Route, Navigate } from 'react-router-dom'
|
|
import { useAuthStore } from '@/store/authStore'
|
|
import { AppLayout } from '@/components/Layout'
|
|
import { AdminLayout } from '@/components/AdminLayout'
|
|
import { PublicLayout } from '@/components/PublicLayout'
|
|
import { ErrorBoundary } from '@/components/ErrorBoundary'
|
|
import { Spinner } from '@/components/ui'
|
|
import './App.css'
|
|
|
|
// Route code splitting with lazy imports
|
|
const LandingPage = lazy(() => import('@/pages/LandingPage').then(m => ({ default: m.LandingPage })))
|
|
const LoginPage = lazy(() => import('@/pages/AuthPages').then(m => ({ default: m.LoginPage })))
|
|
const SignupPage = lazy(() => import('@/pages/AuthPages').then(m => ({ default: m.SignupPage })))
|
|
const ForgotPasswordPage = lazy(() => import('@/pages/ForgotPasswordPage').then(m => ({ default: m.ForgotPasswordPage })))
|
|
const ResetPasswordPage = lazy(() => import('@/pages/ResetPasswordPage').then(m => ({ default: m.ResetPasswordPage })))
|
|
const DashboardPage = lazy(() => import('@/pages/DashboardPage').then(m => ({ default: m.DashboardPage })))
|
|
const ChatbotBuilderPage = lazy(() => import('@/pages/ChatbotBuilderPage').then(m => ({ default: m.ChatbotBuilderPage })))
|
|
const MarketplacePage = lazy(() => import('@/pages/MarketplacePage').then(m => ({ default: m.MarketplacePage })))
|
|
const ChatbotDetailPage = lazy(() => import('@/pages/MarketplacePage').then(m => ({ default: m.ChatbotDetailPage })))
|
|
const PricingPage = lazy(() => import('@/pages/PricingPage').then(m => ({ default: m.PricingPage })))
|
|
const SettingsPage = lazy(() => import('@/pages/SettingsPage').then(m => ({ default: m.SettingsPage })))
|
|
const AnalyticsPage = lazy(() => import('@/pages/AnalyticsPage').then(m => ({ default: m.AnalyticsPage })))
|
|
const PublicChatPage = lazy(() => import('@/pages/PublicChatPage').then(m => ({ default: m.PublicChatPage })))
|
|
const InboxPage = lazy(() => import('@/pages/InboxPage').then(m => ({ default: m.InboxPage })))
|
|
const LeadsPage = lazy(() => import('@/pages/LeadsPage').then(m => ({ default: m.LeadsPage })))
|
|
const AppointmentsPage = lazy(() => import('@/pages/AppointmentsPage').then(m => ({ default: m.AppointmentsPage })))
|
|
const CampaignsPage = lazy(() => import('@/pages/CampaignsPage').then(m => ({ default: m.CampaignsPage })))
|
|
const PublicBookingPage = lazy(() => import('@/pages/PublicBookingPage').then(m => ({ default: m.PublicBookingPage })))
|
|
|
|
// Admin pages
|
|
const AdminDashboardPage = lazy(() => import('@/pages/admin/AdminDashboardPage').then(m => ({ default: m.AdminDashboardPage })))
|
|
const AdminUsersPage = lazy(() => import('@/pages/admin/AdminUsersPage').then(m => ({ default: m.AdminUsersPage })))
|
|
const AdminChatbotsPage = lazy(() => import('@/pages/admin/AdminChatbotsPage').then(m => ({ default: m.AdminChatbotsPage })))
|
|
const AdminConversationsPage = lazy(() => import('@/pages/admin/AdminConversationsPage').then(m => ({ default: m.AdminConversationsPage })))
|
|
const AdminSystemPage = lazy(() => import('@/pages/admin/AdminSystemPage').then(m => ({ default: m.AdminSystemPage })))
|
|
|
|
const PageLoader = () => (
|
|
<div className="flex items-center justify-center min-h-screen">
|
|
<Spinner className="text-primary-600" />
|
|
</div>
|
|
)
|
|
|
|
const PrivateRoute: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
|
const { isAuthenticated } = useAuthStore()
|
|
if (!isAuthenticated) return <Navigate to="/login" replace />
|
|
return <AppLayout>{children}</AppLayout>
|
|
}
|
|
|
|
const AdminRoute: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
|
const { isAuthenticated, user } = useAuthStore()
|
|
if (!isAuthenticated) return <Navigate to="/login" replace />
|
|
if (!user?.is_admin) return <Navigate to="/dashboard" replace />
|
|
return <AdminLayout>{children}</AdminLayout>
|
|
}
|
|
|
|
const PublicOnlyRoute: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
|
const { isAuthenticated } = useAuthStore()
|
|
if (isAuthenticated) return <Navigate to="/dashboard" replace />
|
|
return <>{children}</>
|
|
}
|
|
|
|
const SmartPublicRoute: React.FC<{ children: React.ReactNode }> = ({ children }) => {
|
|
const { isAuthenticated } = useAuthStore()
|
|
if (isAuthenticated) {
|
|
return <AppLayout>{children}</AppLayout>
|
|
}
|
|
return <PublicLayout>{children}</PublicLayout>
|
|
}
|
|
|
|
export const App: React.FC = () => (
|
|
<ErrorBoundary>
|
|
<Suspense fallback={<PageLoader />}>
|
|
<Routes>
|
|
{/* Public - Landing has its own nav */}
|
|
<Route path="/" element={<LandingPage />} />
|
|
|
|
{/* Public pages - wrapped in SmartPublicRoute for proper nav */}
|
|
<Route path="/pricing" element={<SmartPublicRoute><PricingPage /></SmartPublicRoute>} />
|
|
<Route path="/marketplace" element={<SmartPublicRoute><MarketplacePage /></SmartPublicRoute>} />
|
|
<Route path="/marketplace/:id" element={<SmartPublicRoute><ChatbotDetailPage /></SmartPublicRoute>} />
|
|
|
|
{/* Public chat - no auth, no layout */}
|
|
<Route path="/chat/:id" element={<PublicChatPage />} />
|
|
|
|
{/* Public booking - no auth, no layout */}
|
|
<Route path="/book/:chatbotId" element={<PublicBookingPage />} />
|
|
|
|
{/* Auth */}
|
|
<Route path="/login" element={<PublicOnlyRoute><LoginPage /></PublicOnlyRoute>} />
|
|
<Route path="/signup" element={<PublicOnlyRoute><SignupPage /></PublicOnlyRoute>} />
|
|
<Route path="/forgot-password" element={<PublicOnlyRoute><ForgotPasswordPage /></PublicOnlyRoute>} />
|
|
<Route path="/reset-password" element={<ResetPasswordPage />} />
|
|
|
|
{/* Protected */}
|
|
<Route path="/dashboard" element={<PrivateRoute><DashboardPage /></PrivateRoute>} />
|
|
<Route path="/analytics" element={<PrivateRoute><AnalyticsPage /></PrivateRoute>} />
|
|
<Route path="/chatbots/new" element={<PrivateRoute><ChatbotBuilderPage /></PrivateRoute>} />
|
|
<Route path="/chatbots/:id/edit" element={<PrivateRoute><ChatbotBuilderPage /></PrivateRoute>} />
|
|
<Route path="/chatbots/:id/preview" element={<PrivateRoute><ChatbotBuilderPage /></PrivateRoute>} />
|
|
<Route path="/inbox" element={<PrivateRoute><InboxPage /></PrivateRoute>} />
|
|
<Route path="/leads" element={<PrivateRoute><LeadsPage /></PrivateRoute>} />
|
|
<Route path="/appointments" element={<PrivateRoute><AppointmentsPage /></PrivateRoute>} />
|
|
<Route path="/campaigns" element={<PrivateRoute><CampaignsPage /></PrivateRoute>} />
|
|
<Route path="/settings" element={<PrivateRoute><SettingsPage /></PrivateRoute>} />
|
|
<Route path="/settings/billing" element={<PrivateRoute><SettingsPage /></PrivateRoute>} />
|
|
|
|
{/* Admin */}
|
|
<Route path="/admin" element={<AdminRoute><AdminDashboardPage /></AdminRoute>} />
|
|
<Route path="/admin/users" element={<AdminRoute><AdminUsersPage /></AdminRoute>} />
|
|
<Route path="/admin/chatbots" element={<AdminRoute><AdminChatbotsPage /></AdminRoute>} />
|
|
<Route path="/admin/conversations" element={<AdminRoute><AdminConversationsPage /></AdminRoute>} />
|
|
<Route path="/admin/system" element={<AdminRoute><AdminSystemPage /></AdminRoute>} />
|
|
|
|
{/* Fallback */}
|
|
<Route path="*" element={<Navigate to="/" replace />} />
|
|
</Routes>
|
|
</Suspense>
|
|
</ErrorBoundary>
|
|
)
|