Performance optimization, applicant portal, and missing DB migration

Performance:
- Convert admin dashboard from SSR to client-side tRPC (fixes 503/ChunkLoadError)
- New dashboard.getStats tRPC endpoint batches 16 queries into single response
- Parallelize jury dashboard queries (assignments + gracePeriods via Promise.all)
- Add project.getFullDetail combined endpoint (project + assignments + stats)
- Configure Prisma connection pool (connection_limit=20, pool_timeout=10)
- Add optimizePackageImports for lucide-react tree-shaking
- Increase React Query staleTime from 1min to 5min

Applicant portal:
- Add applicant layout, nav, dashboard, documents, team, and mentor pages
- Add applicant router with document and team management endpoints
- Add chunk error recovery utility
- Update role nav and auth redirect for applicant role

Database:
- Add migration for missing schema elements (SpecialAward job tracking
  columns, WizardTemplate table, missing indexes)

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-02-11 11:04:26 +01:00
parent 09091d7c08
commit 98f4a957cc
32 changed files with 3002 additions and 1121 deletions

View File

@@ -4,6 +4,7 @@ import { useCallback, useEffect, useRef, useState } from 'react'
import Link from 'next/link'
import type { Route } from 'next'
import { usePathname } from 'next/navigation'
import { useSession } from 'next-auth/react'
import { trpc } from '@/lib/trpc/client'
import { cn, formatRelativeTime } from '@/lib/utils'
import { Button } from '@/components/ui/button'
@@ -212,6 +213,8 @@ function NotificationItem({
export function NotificationBell() {
const [open, setOpen] = useState(false)
const pathname = usePathname()
const { status: sessionStatus } = useSession()
const isAuthenticated = sessionStatus === 'authenticated'
// Derive the role-based path prefix from the current route
const pathPrefix = pathname.startsWith('/admin')
@@ -222,16 +225,20 @@ export function NotificationBell() {
? '/mentor'
: pathname.startsWith('/observer')
? '/observer'
: ''
: pathname.startsWith('/applicant')
? '/applicant'
: ''
const { data: countData } = trpc.notification.getUnreadCount.useQuery(
undefined,
{
enabled: isAuthenticated,
refetchInterval: 30000, // Refetch every 30 seconds
}
)
const { data: hasUrgent } = trpc.notification.hasUrgent.useQuery(undefined, {
enabled: isAuthenticated,
refetchInterval: 30000,
})
@@ -241,7 +248,7 @@ export function NotificationBell() {
limit: 20,
},
{
enabled: open, // Only fetch when popover is open
enabled: open && isAuthenticated, // Only fetch when popover is open and authenticated
}
)