Frontend Performance Optimization

Frontend performance optimization techniques including bundle optimization, asset management, and runtime performance.

Frontend Performance Optimization

Frontend performance is critical for user experience in Smart Shelf. This section covers comprehensive strategies for optimizing client-side performance, from initial bundle loading to runtime efficiency.

Bundle Optimization

Code Splitting Strategy

Code splitting reduces initial bundle size by loading code on demand, improving initial page load times.

Route-Based Code Splitting

// Dynamic imports for route-based splitting
const Dashboard = dynamic(() => import('./dashboard'), {
  loading: () => <PageSkeleton />,
  ssr: false
})

const InventoryPage = dynamic(() => import('./inventory'), {
  loading: () => <InventoryPageSkeleton />,
  ssr: true // Enable SSR for critical pages
})

const ReportsPage = dynamic(() => import('./reports'), {
  loading: () => <ReportsSkeleton />,
  ssr: false // Disable SSR for heavy analytical pages
})

Component-Based Code Splitting

// Heavy components loaded on demand
const BarcodeScannerModal = dynamic(
  () => import('@/components/barcode/scanner-modal'),
  {
    loading: () => <div className="animate-pulse">Loading scanner...</div>,
    ssr: false // Camera features require client-side rendering
  }
)

const AdvancedReportBuilder = dynamic(
  () => import('@/components/reports/advanced-builder'),
  {
    loading: () => <ReportBuilderSkeleton />,
    ssr: false
  }
)

// Chart library splitting
const Chart = dynamic(() => import('recharts').then(mod => mod.Chart), {
  loading: () => <ChartSkeleton />
})

Library-Level Code Splitting

// Split large libraries
const ExcelExport = dynamic(
  () => import('@/lib/excel-export').then(mod => mod.ExcelExport),
  { ssr: false }
)

// Conditional library loading
const PDFGenerator = dynamic(
  () => import('@/lib/pdf-generator'),
  {
    loading: () => <div>Preparing PDF...</div>,
    ssr: false
  }
)

Tree Shaking Configuration

Eliminate unused code from the final bundle through proper tree shaking configuration.

Next.js Configuration

// next.config.mjs
const nextConfig = {
  experimental: {
    optimizePackageImports: [
      '@radix-ui/react-icons',
      'lucide-react',
      'recharts',
      'date-fns',
      'lodash-es'
    ]
  },
  webpack: (config, { dev, isServer }) => {
    // Enable tree shaking
    config.optimization.sideEffects = false
    
    // Optimize production builds
    if (!dev && !isServer) {
      config.optimization.splitChunks = {
        chunks: 'all',
        cacheGroups: {
          vendor: {
            test: /[\\/]node_modules[\\/]/,
            name: 'vendors',
            chunks: 'all',
          },
          common: {
            name: 'common',
            minChunks: 2,
            chunks: 'all',
          },
        },
      }
    }
    
    return config
  }
}

export default nextConfig

Package.json Optimizations

{
  "sideEffects": false,
  "dependencies": {
    "lodash-es": "^4.17.21",
    "date-fns": "^2.29.3"
  },
  "scripts": {
    "build:analyze": "ANALYZE=true next build",
    "bundle:analyze": "npx webpack-bundle-analyzer .next/static/chunks/*.js"
  }
}

Bundle Analysis and Monitoring

Regular bundle analysis helps identify optimization opportunities.

Bundle Analysis Script

#!/bin/bash
# scripts/analyze-bundle.sh

echo "Building production bundle..."
npm run build

echo "Analyzing bundle size..."
npx webpack-bundle-analyzer .next/static/chunks/*.js --mode static --report bundle-report.html

echo "Checking for duplicate dependencies..."
npx depcheck

echo "Bundle size summary:"
du -sh .next/static/chunks/

Bundle Size Monitoring

// lib/monitoring/bundle-monitoring.ts
export function trackBundleMetrics() {
  if (typeof window !== 'undefined' && 'performance' in window) {
    window.addEventListener('load', () => {
      const navigation = performance.getEntriesByType('navigation')[0] as PerformanceNavigationTiming
      
      const metrics = {
        domContentLoaded: navigation.domContentLoadedEventEnd - navigation.domContentLoadedEventStart,
        loadComplete: navigation.loadEventEnd - navigation.loadEventStart,
        resourceCount: performance.getEntriesByType('resource').length,
        transferSize: performance.getEntriesByType('resource').reduce(
          (total, resource: any) => total + (resource.transferSize || 0), 0
        )
      }
      
      console.log('Bundle Performance Metrics:', metrics)
      // Send to analytics service
    })
  }
}

Asset Optimization

Image Optimization

Smart Shelf implements comprehensive image optimization for better performance.

Optimized Image Component

// components/ui/optimized-image.tsx
import Image from 'next/image'
import { useState } from 'react'

interface OptimizedImageProps {
  src: string
  alt: string
  width: number
  height: number
  priority?: boolean
  className?: string
  onLoad?: () => void
}

export function OptimizedImage({ 
  src, 
  alt, 
  width, 
  height, 
  priority = false,
  className,
  onLoad
}: OptimizedImageProps) {
  const [isLoading, setIsLoading] = useState(true)

  return (
    <div className={`relative ${className}`}>
      <Image
        src={src}
        alt={alt}
        width={width}
        height={height}
        priority={priority}
        sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
        style={{
          width: '100%',
          height: 'auto',
        }}
        placeholder="blur"
        blurDataURL=""
        onLoad={() => {
          setIsLoading(false)
          onLoad?.()
        }}
        className={`
          transition-opacity duration-300
          ${isLoading ? 'opacity-0' : 'opacity-100'}
        `}
      />
      {isLoading && (
        <div className="absolute inset-0 bg-gray-200 animate-pulse rounded" />
      )}
    </div>
  )
}

Image Optimization Configuration

// next.config.mjs
const nextConfig = {
  images: {
    formats: ['image/avif', 'image/webp'],
    domains: ['your-cdn-domain.com'],
    deviceSizes: [640, 750, 828, 1080, 1200, 1920, 2048, 3840],
    imageSizes: [16, 32, 48, 64, 96, 128, 256, 384],
    minimumCacheTTL: 60 * 60 * 24 * 30, // 30 days
  }
}

Font Optimization

Optimize font loading for better performance and user experience.

Font Loading Strategy

// app/layout.tsx
import { Inter, JetBrains_Mono } from 'next/font/google'

const inter = Inter({
  subsets: ['latin'],
  display: 'swap',
  preload: true,
  variable: '--font-inter',
  fallback: ['system-ui', 'arial']
})

const jetbrainsMono = JetBrains_Mono({
  subsets: ['latin'],
  display: 'swap',
  preload: false, // Only load when needed
  variable: '--font-mono',
  fallback: ['Courier New', 'monospace']
})

export default function RootLayout({
  children,
}: {
  children: React.ReactNode
}) {
  return (
    <html lang="en" className={`${inter.variable} ${jetbrainsMono.variable}`}>
      <head>
        {/* Preload critical fonts */}
        <link
          rel="preload"
          href="/fonts/inter-var.woff2"
          as="font"
          type="font/woff2"
          crossOrigin="anonymous"
        />
      </head>
      <body className="font-sans">{children}</body>
    </html>
  )
}

Custom Font Loading Hook

// hooks/use-font-loading.ts
import { useState, useEffect } from 'react'

export function useFontLoading() {
  const [fontsLoaded, setFontsLoaded] = useState(false)

  useEffect(() => {
    if ('fonts' in document) {
      Promise.all([
        document.fonts.load('400 1em Inter'),
        document.fonts.load('500 1em Inter'),
        document.fonts.load('600 1em Inter'),
      ]).then(() => {
        setFontsLoaded(true)
      })
    } else {
      // Fallback for browsers without FontFace API
      setTimeout(() => setFontsLoaded(true), 1000)
    }
  }, [])

  return fontsLoaded
}

Runtime Performance

React Performance Optimization

Optimize React component rendering and updates for better runtime performance.

Memoization Strategies

// components/inventory/product-table.tsx
import React, { memo, useMemo, useCallback } from 'react'

interface Product {
  id: string
  name: string
  sku: string
  price: number
  quantity: number
}

interface ProductTableProps {
  products: Product[]
  onProductClick: (id: string) => void
  sortField: string
  sortDirection: 'asc' | 'desc'
}

// Memoize expensive components
const ProductRow = memo(({ 
  product, 
  onClick 
}: { 
  product: Product
  onClick: (id: string) => void 
}) => {
  const handleClick = useCallback(() => {
    onClick(product.id)
  }, [product.id, onClick])

  return (
    <tr onClick={handleClick} className="hover:bg-gray-50 cursor-pointer">
      <td>{product.name}</td>
      <td>{product.sku}</td>
      <td>${product.price.toFixed(2)}</td>
      <td>{product.quantity}</td>
    </tr>
  )
})

export const ProductTable = memo(({ 
  products, 
  onProductClick, 
  sortField, 
  sortDirection 
}: ProductTableProps) => {
  // Memoize expensive calculations
  const sortedProducts = useMemo(() => {
    return [...products].sort((a, b) => {
      const aValue = a[sortField as keyof Product]
      const bValue = b[sortField as keyof Product]
      
      if (sortDirection === 'asc') {
        return aValue > bValue ? 1 : -1
      }
      return aValue < bValue ? 1 : -1
    })
  }, [products, sortField, sortDirection])

  // Stable callback references
  const handleProductClick = useCallback((id: string) => {
    onProductClick(id)
  }, [onProductClick])

  return (
    <table className="w-full">
      <thead>
        <tr>
          <th>Name</th>
          <th>SKU</th>
          <th>Price</th>
          <th>Quantity</th>
        </tr>
      </thead>
      <tbody>
        {sortedProducts.map(product => (
          <ProductRow 
            key={product.id} 
            product={product}
            onClick={handleProductClick}
          />
        ))}
      </tbody>
    </table>
  )
})

Virtual Scrolling for Large Lists

Handle large datasets efficiently with virtual scrolling.

Virtual Scrolling Implementation

// components/ui/virtual-table.tsx
import { FixedSizeList as List } from 'react-window'
import { memo } from 'react'

interface VirtualTableProps<T> {
  items: T[]
  height: number
  itemHeight: number
  renderItem: (item: T, index: number) => React.ReactNode
  className?: string
}

export function VirtualTable<T>({ 
  items, 
  height, 
  itemHeight, 
  renderItem,
  className 
}: VirtualTableProps<T>) {
  const Row = memo(({ index, style }: { index: number; style: any }) => (
    <div style={style}>
      {renderItem(items[index], index)}
    </div>
  ))

  return (
    <div className={className}>
      <List
        height={height}
        itemCount={items.length}
        itemSize={itemHeight}
        width="100%"
        overscanCount={5} // Render 5 extra items for smooth scrolling
      >
        {Row}
      </List>
    </div>
  )
}

// Usage example
export function InventoryList({ inventory }: { inventory: InventoryItem[] }) {
  const renderInventoryItem = useCallback((item: InventoryItem, index: number) => (
    <div className="flex items-center p-4 border-b">
      <div className="flex-1">
        <h4 className="font-medium">{item.product.name}</h4>
        <p className="text-sm text-gray-500">{item.product.sku}</p>
      </div>
      <div className="text-right">
        <span className="font-semibold">{item.quantity}</span>
        <p className="text-sm text-gray-500">{item.warehouse.name}</p>
      </div>
    </div>
  ), [])

  return (
    <VirtualTable
      items={inventory}
      height={600}
      itemHeight={80}
      renderItem={renderInventoryItem}
      className="border rounded-lg"
    />
  )
}

Progressive Loading and Infinite Scroll

Implement progressive data loading for better perceived performance.

Progressive Loading Hook

// hooks/use-progressive-loading.ts
import { useState, useCallback, useRef, useEffect } from 'react'

interface UseProgressiveLoadingOptions<T> {
  fetchFunction: (page: number, pageSize: number) => Promise<T[]>
  pageSize?: number
  initialPage?: number
}

export function useProgressiveLoading<T>({
  fetchFunction,
  pageSize = 20,
  initialPage = 0
}: UseProgressiveLoadingOptions<T>) {
  const [items, setItems] = useState<T[]>([])
  const [loading, setLoading] = useState(false)
  const [hasMore, setHasMore] = useState(true)
  const [error, setError] = useState<string | null>(null)
  const currentPage = useRef(initialPage)

  const loadMore = useCallback(async () => {
    if (loading || !hasMore) return

    setLoading(true)
    setError(null)

    try {
      const newItems = await fetchFunction(currentPage.current, pageSize)
      
      setItems(prev => [...prev, ...newItems])
      setHasMore(newItems.length === pageSize)
      currentPage.current += 1
    } catch (err) {
      setError(err instanceof Error ? err.message : 'Failed to load data')
    } finally {
      setLoading(false)
    }
  }, [fetchFunction, loading, hasMore, pageSize])

  const reset = useCallback(() => {
    setItems([])
    setHasMore(true)
    setError(null)
    currentPage.current = initialPage
  }, [initialPage])

  // Load initial data
  useEffect(() => {
    loadMore()
  }, []) // Only run once

  return {
    items,
    loading,
    hasMore,
    error,
    loadMore,
    reset
  }
}

Infinite Scroll Component

// components/ui/infinite-scroll.tsx
import { useEffect, useRef } from 'react'

interface InfiniteScrollProps {
  children: React.ReactNode
  hasMore: boolean
  loading: boolean
  onLoadMore: () => void
  threshold?: number
  className?: string
}

export function InfiniteScroll({
  children,
  hasMore,
  loading,
  onLoadMore,
  threshold = 200,
  className
}: InfiniteScrollProps) {
  const sentinelRef = useRef<HTMLDivElement>(null)

  useEffect(() => {
    const sentinel = sentinelRef.current
    if (!sentinel) return

    const observer = new IntersectionObserver(
      (entries) => {
        const entry = entries[0]
        if (entry.isIntersecting && hasMore && !loading) {
          onLoadMore()
        }
      },
      {
        rootMargin: `${threshold}px`,
      }
    )

    observer.observe(sentinel)

    return () => {
      observer.unobserve(sentinel)
    }
  }, [hasMore, loading, onLoadMore, threshold])

  return (
    <div className={className}>
      {children}
      {hasMore && (
        <div ref={sentinelRef} className="h-4 flex items-center justify-center">
          {loading && (
            <div className="flex items-center space-x-2">
              <div className="animate-spin rounded-full h-4 w-4 border-b-2 border-blue-600"></div>
              <span className="text-sm text-gray-500">Loading more...</span>
            </div>
          )}
        </div>
      )}
    </div>
  )
}

Loading States & User Experience

Skeleton Components

Provide immediate visual feedback while content loads.

Comprehensive Skeleton System

// components/ui/skeleton.tsx
import { cn } from '@/lib/utils'

export function Skeleton({ 
  className, 
  ...props 
}: React.HTMLAttributes<HTMLDivElement>) {
  return (
    <div
      className={cn(
        "animate-pulse rounded-md bg-gray-200 dark:bg-gray-800",
        className
      )}
      {...props}
    />
  )
}

// Specific skeleton components
export function ProductTableSkeleton() {
  return (
    <div className="space-y-4">
      {/* Table header */}
      <div className="grid grid-cols-4 gap-4 pb-2 border-b">
        {Array.from({ length: 4 }).map((_, i) => (
          <Skeleton key={i} className="h-4 w-20" />
        ))}
      </div>
      
      {/* Table rows */}
      {Array.from({ length: 10 }).map((_, index) => (
        <div key={index} className="grid grid-cols-4 gap-4 py-3">
          <Skeleton className="h-4 w-32" />
          <Skeleton className="h-4 w-24" />
          <Skeleton className="h-4 w-16" />
          <Skeleton className="h-4 w-12" />
        </div>
      ))}
    </div>
  )
}

export function DashboardSkeleton() {
  return (
    <div className="space-y-6">
      {/* Stats cards */}
      <div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
        {Array.from({ length: 4 }).map((_, i) => (
          <div key={i} className="p-6 border rounded-lg">
            <div className="flex items-center justify-between">
              <Skeleton className="h-4 w-20" />
              <Skeleton className="h-6 w-6 rounded" />
            </div>
            <Skeleton className="h-8 w-16 mt-2" />
            <Skeleton className="h-3 w-24 mt-1" />
          </div>
        ))}
      </div>
      
      {/* Chart area */}
      <div className="p-6 border rounded-lg">
        <Skeleton className="h-6 w-40 mb-4" />
        <Skeleton className="h-64 w-full" />
      </div>
    </div>
  )
}

Smart Loading States

Implement intelligent loading states that improve perceived performance.

Loading State Manager

// hooks/use-loading-state.ts
import { useState, useCallback } from 'react'

interface LoadingState {
  loading: boolean
  error: string | null
  success: boolean
}

export function useLoadingState(initialState: Partial<LoadingState> = {}) {
  const [state, setState] = useState<LoadingState>({
    loading: false,
    error: null,
    success: false,
    ...initialState
  })

  const startLoading = useCallback(() => {
    setState({ loading: true, error: null, success: false })
  }, [])

  const setSuccess = useCallback(() => {
    setState({ loading: false, error: null, success: true })
  }, [])

  const setError = useCallback((error: string) => {
    setState({ loading: false, error, success: false })
  }, [])

  const reset = useCallback(() => {
    setState({ loading: false, error: null, success: false })
  }, [])

  const withLoading = useCallback(async <T>(
    promise: Promise<T>
  ): Promise<T> => {
    startLoading()
    try {
      const result = await promise
      setSuccess()
      return result
    } catch (error) {
      setError(error instanceof Error ? error.message : 'An error occurred')
      throw error
    }
  }, [startLoading, setSuccess, setError])

  return {
    ...state,
    startLoading,
    setSuccess,
    setError,
    reset,
    withLoading
  }
}

Frontend performance optimization requires continuous monitoring and improvement. These techniques provide a solid foundation for delivering fast, responsive user experiences in Smart Shelf.