Add detailed landing page development plan in docs/frontend_landing_plan.md: - Complete landing page structure (Hero, Problem/Solution, Features, Demo, CTA) - Design guidelines from downloaded skills (typography, color, motion, composition) - Security considerations (XSS prevention, input sanitization, CSP) - Performance targets (LCP <2.5s, bundle <150KB, Lighthouse >90) - Responsiveness and accessibility requirements (WCAG 2.1 AA) - Success KPIs and monitoring setup - 3-week development timeline with daily tasks - Definition of Done checklist Download 10+ frontend/UI/UX skills via universal-skills-manager: - frontend-ui-ux: UI/UX design without mockups - frontend-design-guidelines: Production-grade interface guidelines - frontend-developer: React best practices (40+ rules) - frontend-engineer: Next.js 14 App Router patterns - ui-ux-master: Comprehensive design systems and accessibility - ui-ux-systems-designer: Information architecture and interaction - ui-ux-design-user-experience: Platform-specific guidelines - Plus additional reference materials and validation scripts Configure universal-skills MCP with SkillsMP API key for curated skill access. Safety first: All skills validated before installation, no project code modified. Refs: Universal Skills Manager (github:jacob-bd/universal-skills-manager) Next: Begin Sprint 3 landing page development
81 lines
1.9 KiB
Markdown
81 lines
1.9 KiB
Markdown
---
|
|
title: Cache Repeated Function Calls
|
|
impact: MEDIUM
|
|
impactDescription: avoid redundant computation
|
|
tags: javascript, cache, memoization, performance
|
|
---
|
|
|
|
## Cache Repeated Function Calls
|
|
|
|
Use a module-level Map to cache function results when the same function is called repeatedly with the same inputs during render.
|
|
|
|
**Incorrect (redundant computation):**
|
|
|
|
```typescript
|
|
function ProjectList({ projects }: { projects: Project[] }) {
|
|
return (
|
|
<div>
|
|
{projects.map(project => {
|
|
// slugify() called 100+ times for same project names
|
|
const slug = slugify(project.name)
|
|
|
|
return <ProjectCard key={project.id} slug={slug} />
|
|
})}
|
|
</div>
|
|
)
|
|
}
|
|
```
|
|
|
|
**Correct (cached results):**
|
|
|
|
```typescript
|
|
// Module-level cache
|
|
const slugifyCache = new Map<string, string>()
|
|
|
|
function cachedSlugify(text: string): string {
|
|
if (slugifyCache.has(text)) {
|
|
return slugifyCache.get(text)!
|
|
}
|
|
const result = slugify(text)
|
|
slugifyCache.set(text, result)
|
|
return result
|
|
}
|
|
|
|
function ProjectList({ projects }: { projects: Project[] }) {
|
|
return (
|
|
<div>
|
|
{projects.map(project => {
|
|
// Computed only once per unique project name
|
|
const slug = cachedSlugify(project.name)
|
|
|
|
return <ProjectCard key={project.id} slug={slug} />
|
|
})}
|
|
</div>
|
|
)
|
|
}
|
|
```
|
|
|
|
**Simpler pattern for single-value functions:**
|
|
|
|
```typescript
|
|
let isLoggedInCache: boolean | null = null
|
|
|
|
function isLoggedIn(): boolean {
|
|
if (isLoggedInCache !== null) {
|
|
return isLoggedInCache
|
|
}
|
|
|
|
isLoggedInCache = document.cookie.includes('auth=')
|
|
return isLoggedInCache
|
|
}
|
|
|
|
// Clear cache when auth changes
|
|
function onAuthChange() {
|
|
isLoggedInCache = null
|
|
}
|
|
```
|
|
|
|
Use a Map (not a hook) so it works everywhere: utilities, event handlers, not just React components.
|
|
|
|
Reference: [How we made the Vercel Dashboard twice as fast](https://vercel.com/blog/how-we-made-the-vercel-dashboard-twice-as-fast)
|