This comprehensive guide covers performance optimization techniques and best practices for the Architect Resume Portfolio to ensure fast loading times, smooth animations, and excellent user experience.
The Architect Resume Portfolio is optimized for:
LCP measures loading performance and should occur within 2.5 seconds.
// 1. Optimize Hero Image Loading
// app/components/Hero.tsx
import Image from 'next/image'
export default function Hero() {
return (
<div className="relative min-h-screen">
<Image
src="/images/hero-background.jpg"
alt="Architectural Design Background"
fill
priority // Load immediately
quality={85}
placeholder="blur"
blurDataURL="..."
sizes="100vw"
className="object-cover"
/>
{/* Hero content */}
</div>
)
}
// 2. Preload Critical Resources
// app/layout.tsx
export default function RootLayout({ children }: { children: React.ReactNode }) {
return (
<html lang="en">
<head>
<link
rel="preload"
href="/images/hero-background.jpg"
as="image"
type="image/jpeg"
/>
<link
rel="preload"
href="/fonts/playfair-display.woff2"
as="font"
type="font/woff2"
crossOrigin="anonymous"
/>
</head>
<body>{children}</body>
</html>
)
}
FID measures interactivity and should be less than 100ms.
// 1. Code Splitting for Interactive Components
import dynamic from 'next/dynamic'
import { Suspense } from 'react'
// Lazy load non-critical interactive components
const ContactForm = dynamic(() => import('./ContactForm'), {
loading: () => <div className="h-96 bg-gray-100 animate-pulse rounded-lg" />,
ssr: false // Disable SSR for client-only components
})
const Portfolio = dynamic(() => import('./Portfolio'), {
loading: () => <PortfolioSkeleton />
})
export default function HomePage() {
return (
<main>
<Hero /> {/* Critical, loaded immediately */}
<Suspense fallback={<div>Loading...</div>}>
<Portfolio />
</Suspense>
<Suspense fallback={<div>Loading...</div>}>
<ContactForm />
</Suspense>
</main>
)
}
// 2. Optimize Event Handlers
import { useCallback, useMemo } from 'react'
export default function Navigation() {
// Memoize expensive calculations
const navigationItems = useMemo(() => [
{ href: '#home', label: 'Home' },
{ href: '#portfolio', label: 'Portfolio' },
{ href: '#contact', label: 'Contact' }
], [])
// Memoize event handlers
const handleMenuToggle = useCallback(() => {
setIsMenuOpen(prev => !prev)
}, [])
const handleSmoothScroll = useCallback((targetId: string) => {
const element = document.getElementById(targetId)
if (element) {
element.scrollIntoView({ behavior: 'smooth' })
}
}, [])
return (
<nav>
{navigationItems.map(item => (
<button
key={item.href}
onClick={() => handleSmoothScroll(item.href.slice(1))}
className="nav-link"
>
{item.label}
</button>
))}
</nav>
)
}
CLS measures visual stability and should be less than 0.1.
// 1. Reserve Space for Dynamic Content
// app/components/Portfolio.tsx
export default function Portfolio() {
const [projects, setProjects] = useState<Project[]>([])
const [loading, setLoading] = useState(true)
return (
<section className="py-20">
<div className="grid md:grid-cols-2 lg:grid-cols-3 gap-8">
{loading ? (
// Maintain layout during loading
Array.from({ length: 6 }).map((_, index) => (
<div
key={index}
className="aspect-[4/3] bg-gray-200 animate-pulse rounded-lg"
style= // Fixed height prevents CLS
/>
))
) : (
projects.map(project => (
<ProjectCard key={project.id} project={project} />
))
)}
</div>
</section>
)
}
// 2. Use CSS Aspect Ratios
// globals.css
.project-card-image {
aspect-ratio: 4 / 3;
width: 100%;
object-fit: cover;
}
.hero-background {
aspect-ratio: 16 / 9;
min-height: 100vh;
}
@media (max-width: 768px) {
.hero-background {
aspect-ratio: 3 / 4;
}
}
// lib/imageOptimization.ts
import Image, { ImageProps } from 'next/image'
interface OptimizedImageProps extends Omit<ImageProps, 'src'> {
src: string
alt: string
priority?: boolean
quality?: number
}
export function OptimizedImage({
src,
alt,
priority = false,
quality = 85,
...props
}: OptimizedImageProps) {
return (
<Image
src={src}
alt={alt}
priority={priority}
quality={quality}
placeholder="blur"
blurDataURL={generateBlurDataURL()}
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
{...props}
/>
)
}
function generateBlurDataURL(): string {
return ''
}
// next.config.js
/** @type {import('next').NextConfig} */
const nextConfig = {
images: {
formats: ['image/avif', 'image/webp'], // Serve modern formats first
domains: [
'images.unsplash.com',
'res.cloudinary.com',
'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 * 365, // 1 year
},
// Enable experimental image optimization
experimental: {
optimizeCss: true,
optimizePackageImports: ['lucide-react', 'framer-motion']
}
}
module.exports = nextConfig
// components/ResponsiveImage.tsx
interface ResponsiveImageProps {
src: string
alt: string
className?: string
priority?: boolean
}
export default function ResponsiveImage({
src,
alt,
className,
priority = false
}: ResponsiveImageProps) {
return (
<picture>
<source
srcSet={`${src}?format=avif&w=320 320w, ${src}?format=avif&w=640 640w, ${src}?format=avif&w=1280 1280w`}
type="image/avif"
/>
<source
srcSet={`${src}?format=webp&w=320 320w, ${src}?format=webp&w=640 640w, ${src}?format=webp&w=1280 1280w`}
type="image/webp"
/>
<Image
src={src}
alt={alt}
width={1280}
height={720}
priority={priority}
className={className}
sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
/>
</picture>
)
}
// app/page.tsx
import dynamic from 'next/dynamic'
import { Suspense } from 'react'
// Critical components - loaded immediately
import Hero from './components/Hero'
import Navigation from './components/Navigation'
// Non-critical components - lazy loaded
const Portfolio = dynamic(() => import('./components/Portfolio'), {
loading: () => <PortfolioSkeleton />,
})
const Experience = dynamic(() => import('./components/Experience'), {
loading: () => <ExperienceSkeleton />,
})
const Skills = dynamic(() => import('./components/Skills'), {
loading: () => <SkillsSkeleton />,
})
const Contact = dynamic(() => import('./components/Contact'), {
loading: () => <ContactSkeleton />,
ssr: false // Client-side only for form interactions
})
export default function HomePage() {
return (
<main>
<Navigation />
<Hero />
<Suspense fallback={<div>Loading portfolio...</div>}>
<Portfolio />
</Suspense>
<Suspense fallback={<div>Loading experience...</div>}>
<Experience />
</Suspense>
<Suspense fallback={<div>Loading skills...</div>}>
<Skills />
</Suspense>
<Suspense fallback={<div>Loading contact form...</div>}>
<Contact />
</Suspense>
</main>
)
}
// app/layout.tsx
import { Suspense } from 'react'
import LoadingSpinner from './components/LoadingSpinner'
export default function RootLayout({
children,
}: {
children: React.ReactNode
}) {
return (
<html lang="en">
<body>
<Suspense fallback={<LoadingSpinner />}>
{children}
</Suspense>
</body>
</html>
)
}
// next.config.js
const withBundleAnalyzer = require('@next/bundle-analyzer')({
enabled: process.env.ANALYZE === 'true',
})
module.exports = withBundleAnalyzer({
// Your Next.js config
experimental: {
optimizePackageImports: [
'lucide-react',
'framer-motion',
'@radix-ui/react-dialog',
'@radix-ui/react-tabs'
]
},
webpack: (config, { dev, isServer }) => {
// Production optimizations
if (!dev && !isServer) {
config.optimization.splitChunks.chunks = 'all'
config.optimization.splitChunks.cacheGroups = {
vendor: {
test: /[\\/]node_modules[\\/]/,
name: 'vendors',
chunks: 'all',
},
common: {
name: 'common',
minChunks: 2,
chunks: 'all',
enforce: true,
}
}
}
return config
}
})
// lib/utils.ts - Use targeted imports
// ❌ Bad - imports entire library
import * as Icons from 'lucide-react'
// ✅ Good - imports only needed icons
import { Mail, Phone, MapPin, Github, Linkedin } from 'lucide-react'
// ❌ Bad - imports entire Framer Motion
import * as Motion from 'framer-motion'
// ✅ Good - imports only needed parts
import { motion, AnimatePresence, useAnimation } from 'framer-motion'
// package.json
{
"scripts": {
"analyze": "ANALYZE=true npm run build",
"build:production": "NODE_ENV=production npm run build"
},
"dependencies": {
"framer-motion": "^11.0.0",
"lucide-react": "^0.400.0"
},
"devDependencies": {
"@next/bundle-analyzer": "^15.0.0"
}
}
// app/layout.tsx
import { Inter, Playfair_Display } from 'next/font/google'
// Optimize font loading
const inter = Inter({
subsets: ['latin'],
variable: '--font-inter',
display: 'swap', // Improve font loading performance
preload: true,
fallback: ['system-ui', 'arial']
})
const playfair = Playfair_Display({
subsets: ['latin'],
variable: '--font-playfair',
display: 'swap',
preload: true,
fallback: ['serif']
})
export default function RootLayout({
children,
}: {
children: React.ReactNode
}) {
return (
<html lang="en" className={`${inter.variable} ${playfair.variable}`}>
<body className="font-sans">
{children}
</body>
</html>
)
}
// For custom/local fonts
import localFont from 'next/font/local'
const customFont = localFont({
src: [
{
path: '../public/fonts/CustomFont-Regular.woff2',
weight: '400',
style: 'normal',
},
{
path: '../public/fonts/CustomFont-Bold.woff2',
weight: '700',
style: 'normal',
}
],
variable: '--font-custom',
display: 'swap',
preload: true,
fallback: ['Arial', 'sans-serif']
})
/* globals.css */
/* Font fallback stack to prevent layout shift */
:root {
--font-inter: 'Inter', system-ui, -apple-system, sans-serif;
--font-playfair: 'Playfair Display', Georgia, serif;
}
/* Prevent flash of unstyled text */
.font-serif {
font-family: var(--font-playfair);
font-display: swap;
}
.font-sans {
font-family: var(--font-inter);
font-display: swap;
}
/* Size adjust to match fallback fonts */
@font-face {
font-family: 'Inter';
size-adjust: 107%;
}
// tailwind.config.ts
import type { Config } from 'tailwindcss'
const config: Config = {
content: [
'./pages/**/*.{js,ts,jsx,tsx,mdx}',
'./components/**/*.{js,ts,jsx,tsx,mdx}',
'./app/**/*.{js,ts,jsx,tsx,mdx}',
],
theme: {
extend: {
// Only include custom styles you actually use
colors: {
architect: {
// Custom color palette
}
},
fontFamily: {
// Custom fonts
}
},
},
plugins: [
// Only include plugins you use
],
// Remove unused CSS
purge: {
content: ['./app/**/*.{js,ts,jsx,tsx}'],
options: {
safelist: [
// Add classes that might be added dynamically
'animate-pulse',
'animate-spin'
]
}
}
}
export default config
// app/layout.tsx
export default function RootLayout({
children,
}: {
children: React.ReactNode
}) {
return (
<html lang="en">
<head>
{/* Critical CSS inline */}
<style jsx>{`
.hero-section {
min-height: 100vh;
display: flex;
align-items: center;
justify-content: center;
}
.nav-fixed {
position: fixed;
top: 0;
width: 100%;
z-index: 50;
}
`}</style>
</head>
<body>
{children}
</body>
</html>
)
}
// components/Portfolio.tsx
import { memo, useMemo, useCallback } from 'react'
interface Project {
id: number
title: string
category: string
image: string
}
interface PortfolioProps {
projects: Project[]
}
const Portfolio = memo(function Portfolio({ projects }: PortfolioProps) {
// Memoize filtered projects
const [activeFilter, setActiveFilter] = useState('all')
const filteredProjects = useMemo(() => {
if (activeFilter === 'all') return projects
return projects.filter(project => project.category === activeFilter)
}, [projects, activeFilter])
// Memoize filter handler
const handleFilterChange = useCallback((filter: string) => {
setActiveFilter(filter)
}, [])
return (
<section>
<FilterButtons
activeFilter={activeFilter}
onFilterChange={handleFilterChange}
/>
<ProjectGrid projects={filteredProjects} />
</section>
)
})
// Memoize project cards
const ProjectCard = memo(function ProjectCard({ project }: { project: Project }) {
return (
<div className="project-card">
<OptimizedImage
src={project.image}
alt={project.title}
width={400}
height={300}
/>
<h3>{project.title}</h3>
</div>
)
})
// hooks/useThrottledScroll.ts
import { useCallback, useEffect, useRef } from 'react'
export function useThrottledScroll(callback: () => void, delay: number = 100) {
const lastRun = useRef(Date.now())
const throttledCallback = useCallback(() => {
if (Date.now() - lastRun.current >= delay) {
callback()
lastRun.current = Date.now()
}
}, [callback, delay])
useEffect(() => {
window.addEventListener('scroll', throttledCallback)
return () => window.removeEventListener('scroll', throttledCallback)
}, [throttledCallback])
}
// Usage
function Navigation() {
const [isScrolled, setIsScrolled] = useState(false)
const handleScroll = useCallback(() => {
setIsScrolled(window.scrollY > 100)
}, [])
useThrottledScroll(handleScroll, 100)
return (
<nav className={isScrolled ? 'nav-scrolled' : 'nav-top'}>
{/* Navigation content */}
</nav>
)
}
// lib/animations.ts
export const optimizedVariants = {
// Use transform and opacity for better performance
fadeInUp: {
initial: { opacity: 0, y: 20 },
animate: { opacity: 1, y: 0 },
transition: { duration: 0.5, ease: [0.25, 0.46, 0.45, 0.94] }
},
// Avoid animating layout properties
slideIn: {
initial: { opacity: 0, transform: 'translateX(-20px)' },
animate: { opacity: 1, transform: 'translateX(0px)' },
transition: { duration: 0.3 }
},
// Use will-change sparingly
hover: {
whileHover: {
scale: 1.05,
transition: { duration: 0.2 }
},
whileTap: { scale: 0.95 }
}
}
// Use layout animations only when necessary
export const layoutAnimation = {
layout: true,
transition: { duration: 0.3, ease: 'easeOut' }
}
// components/AnimatedSection.tsx
import { motion } from 'framer-motion'
import { useInView } from 'framer-motion'
import { useRef } from 'react'
export default function AnimatedSection({ children }: { children: React.ReactNode }) {
const ref = useRef(null)
const isInView = useInView(ref, { once: true, margin: '-100px' })
return (
<motion.section
ref={ref}
initial=
animate={isInView ? { opacity: 1, y: 0 } : { opacity: 0, y: 50 }}
transition=
>
{children}
</motion.section>
)
}
/* Use CSS animations for simple, repeated animations */
@keyframes pulse {
0%, 100% { opacity: 1; }
50% { opacity: 0.5; }
}
@keyframes slideIn {
from {
opacity: 0;
transform: translateY(20px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
.animate-pulse {
animation: pulse 2s infinite;
}
.animate-slide-in {
animation: slideIn 0.5s ease-out;
}
/* Use CSS custom properties for dynamic values */
.animated-element {
--duration: 0.3s;
--easing: cubic-bezier(0.25, 0.46, 0.45, 0.94);
transition: transform var(--duration) var(--easing);
}
// app/layout.tsx
export default function RootLayout({
children,
}: {
children: React.ReactNode
}) {
return (
<html lang="en">
<head>
{/* DNS prefetch for external domains */}
<link rel="dns-prefetch" href="//fonts.googleapis.com" />
<link rel="dns-prefetch" href="//images.unsplash.com" />
{/* Preconnect to critical domains */}
<link rel="preconnect" href="https://fonts.gstatic.com" crossOrigin="" />
{/* Preload critical resources */}
<link
rel="preload"
href="/fonts/inter-var.woff2"
as="font"
type="font/woff2"
crossOrigin=""
/>
{/* Prefetch likely next pages */}
<link rel="prefetch" href="/portfolio" />
<link rel="prefetch" href="/contact" />
</head>
<body>{children}</body>
</html>
)
}
// public/sw.js
const CACHE_NAME = 'architect-portfolio-v1'
const urlsToCache = [
'/',
'/static/js/bundle.js',
'/static/css/main.css',
'/images/hero-bg.jpg',
'/fonts/inter-var.woff2'
]
self.addEventListener('install', (event) => {
event.waitUntil(
caches.open(CACHE_NAME)
.then((cache) => cache.addAll(urlsToCache))
)
})
self.addEventListener('fetch', (event) => {
event.respondWith(
caches.match(event.request)
.then((response) => {
// Return cached version or fetch from network
return response || fetch(event.request)
})
)
})
// next.config.js
module.exports = {
async headers() {
return [
{
source: '/images/:path*',
headers: [
{
key: 'Cache-Control',
value: 'public, max-age=31536000, immutable'
}
]
},
{
source: '/fonts/:path*',
headers: [
{
key: 'Cache-Control',
value: 'public, max-age=31536000, immutable'
}
]
},
{
source: '/_next/static/:path*',
headers: [
{
key: 'Cache-Control',
value: 'public, max-age=31536000, immutable'
}
]
}
]
}
}
// lib/cache.ts
class CacheManager {
private cache = new Map()
set(key: string, value: any, ttl: number = 300000) { // 5 minutes default
const expires = Date.now() + ttl
this.cache.set(key, { value, expires })
}
get(key: string) {
const item = this.cache.get(key)
if (!item) return null
if (Date.now() > item.expires) {
this.cache.delete(key)
return null
}
return item.value
}
clear() {
this.cache.clear()
}
}
export const cacheManager = new CacheManager()
// Usage in components
export function useProjectsCache() {
const getCachedProjects = useCallback(() => {
const cached = cacheManager.get('projects')
if (cached) return cached
// Fetch and cache
return fetchProjects().then(data => {
cacheManager.set('projects', data, 600000) // 10 minutes
return data
})
}, [])
return { getCachedProjects }
}
// lib/analytics.ts
import { getCLS, getFID, getFCP, getLCP, getTTFB } from 'web-vitals'
function sendToAnalytics(metric: any) {
// Send to your analytics service
if (typeof window !== 'undefined' && window.gtag) {
window.gtag('event', metric.name, {
event_category: 'Web Vitals',
event_label: metric.id,
value: Math.round(metric.value),
non_interaction: true,
})
}
}
export function initPerformanceMonitoring() {
getCLS(sendToAnalytics)
getFID(sendToAnalytics)
getFCP(sendToAnalytics)
getLCP(sendToAnalytics)
getTTFB(sendToAnalytics)
}
// Performance observer for custom metrics
export function trackCustomMetric(name: string, value: number) {
if (typeof window !== 'undefined' && window.gtag) {
window.gtag('event', 'custom_metric', {
event_category: 'Performance',
event_label: name,
value: Math.round(value),
})
}
}
# .github/workflows/lighthouse.yml
name: Lighthouse CI
on:
push:
branches: [main]
pull_request:
branches: [main]
jobs:
lighthouse:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Setup Node.js
uses: actions/setup-node@v3
with:
node-version: '18'
- name: Install dependencies
run: npm ci
- name: Build app
run: npm run build
- name: Run Lighthouse CI
run: |
npm install -g @lhci/cli@0.12.x
lhci autorun
env:
LHCI_GITHUB_APP_TOKEN: $
// lighthouserc.js
module.exports = {
ci: {
collect: {
url: ['http://localhost:3000/'],
startServerCommand: 'npm start',
},
assert: {
assertions: {
'categories:performance': ['warn', { minScore: 0.9 }],
'categories:accessibility': ['error', { minScore: 0.95 }],
'categories:best-practices': ['warn', { minScore: 0.9 }],
'categories:seo': ['warn', { minScore: 0.9 }],
'first-contentful-paint': ['warn', { maxNumericValue: 2000 }],
'largest-contentful-paint': ['error', { maxNumericValue: 2500 }],
'cumulative-layout-shift': ['error', { maxNumericValue: 0.1 }],
},
},
upload: {
target: 'temporary-public-storage',
},
},
}
/* Optimize mobile-specific styles */
@media (max-width: 768px) {
/* Reduce animations on mobile */
.animate-on-desktop {
animation: none;
}
/* Optimize images for mobile */
.hero-image {
content: url('/images/hero-mobile.jpg');
}
/* Reduce motion for users who prefer it */
@media (prefers-reduced-motion: reduce) {
* {
animation-duration: 0.01ms !important;
animation-iteration-count: 1 !important;
transition-duration: 0.01ms !important;
}
}
}
// hooks/useTouch.ts
import { useCallback, useEffect, useRef } from 'react'
export function useTouch() {
const touchStartX = useRef(0)
const touchStartY = useRef(0)
const handleTouchStart = useCallback((e: TouchEvent) => {
touchStartX.current = e.touches[0].clientX
touchStartY.current = e.touches[0].clientY
}, [])
const handleTouchEnd = useCallback((e: TouchEvent) => {
if (!touchStartX.current || !touchStartY.current) return
const touchEndX = e.changedTouches[0].clientX
const touchEndY = e.changedTouches[0].clientY
const deltaX = touchStartX.current - touchEndX
const deltaY = touchStartY.current - touchEndY
// Handle swipe gestures
if (Math.abs(deltaX) > Math.abs(deltaY)) {
if (deltaX > 50) {
// Swipe left
} else if (deltaX < -50) {
// Swipe right
}
}
}, [])
useEffect(() => {
document.addEventListener('touchstart', handleTouchStart, { passive: true })
document.addEventListener('touchend', handleTouchEnd, { passive: true })
return () => {
document.removeEventListener('touchstart', handleTouchStart)
document.removeEventListener('touchend', handleTouchEnd)
}
}, [handleTouchStart, handleTouchEnd])
}
// lib/featureDetection.ts
export const features = {
webp: typeof window !== 'undefined' && (() => {
const canvas = document.createElement('canvas')
return canvas.toDataURL('image/webp').indexOf('data:image/webp') === 0
})(),
avif: typeof window !== 'undefined' && (() => {
const canvas = document.createElement('canvas')
return canvas.toDataURL('image/avif').indexOf('data:image/avif') === 0
})(),
intersectionObserver: typeof window !== 'undefined' && 'IntersectionObserver' in window,
webGL: typeof window !== 'undefined' && (() => {
try {
const canvas = document.createElement('canvas')
return !!(canvas.getContext('webgl') || canvas.getContext('experimental-webgl'))
} catch (e) {
return false
}
})(),
}
// Use features conditionally
export function useOptimalImageFormat() {
if (features.avif) return 'avif'
if (features.webp) return 'webp'
return 'jpg'
}
// components/EnhancedImage.tsx
export default function EnhancedImage({ src, alt, ...props }: ImageProps) {
const [imageError, setImageError] = useState(false)
const [format, setFormat] = useState('jpg')
useEffect(() => {
// Detect optimal format
const optimalFormat = useOptimalImageFormat()
setFormat(optimalFormat)
}, [])
if (imageError) {
return (
<div className="bg-gray-200 flex items-center justify-center">
<span>Image unavailable</span>
</div>
)
}
return (
<picture>
<source srcSet={`${src}.avif`} type="image/avif" />
<source srcSet={`${src}.webp`} type="image/webp" />
<img
src={`${src}.jpg`}
alt={alt}
onError={() => setImageError(true)}
loading="lazy"
{...props}
/>
</picture>
)
}
This comprehensive performance optimization guide provides all the tools and techniques needed to create a fast, efficient, and user-friendly architect portfolio that performs excellently across all devices and network conditions.