name: data-to-ui description: Data-to-UI pipeline patterns. Use when transforming JSON data into React components, creating TypeScript types from schemas, building derived types, or creating data utilities.
Data to UI
Overview
Patterns for transforming static data into type-safe React components. This skill covers JSON → TypeScript → React pipelines with emphasis on semantic color systems, derived types, and formatting utilities.
Workflows
1. JSON Schema → TypeScript Types
- Read JSON schema/data structure
- Create base TypeScript interfaces matching JSON shape
- Export union types for enums (e.g.,
type Severity = 'safety_hazard' | 'repair_needed') - Use optional properties (
?) for nullable/missing fields - Add JSDoc comments for complex types
2. Derived Types for UI
- Create composed types extending base types with
extends - Use
Pick<T, K>andOmit<T, K>for component props - Build intersection types with
&for joined data (e.g.,FindingWithAsset) - Create aggregate interfaces for statistics/summaries
3. Color Mapping Systems
- Define
Record<EnumType, ColorValue>for semantic colors - Provide multiple color formats: badge, bg, text, border, dot
- Use Tailwind utility classes (e.g.,
'bg-red-500 text-red-600') - Export accessor functions (e.g.,
getSeverityColors()) - Document color choices with comments
4. Icon Mapping
- Create
Record<EnumType, string>mapping to lucide-react icon names - Use PascalCase icon names (e.g.,
'AlertTriangle','Thermometer') - Export accessor function (e.g.,
getSeverityIcon())
5. Formatting Utilities
- Currency: Use
Intl.NumberFormatwith USD, no decimals - Dates: Use
toLocaleDateStringwith short month format - Calculations: Create helpers for years, percentages, lifespans
- Labels: Create human-readable label maps
6. Aggregation & Grouping
- Implement
groupBypatterns using reduce or forEach - Sort with custom comparators using severity/priority order
- Calculate summary statistics (min, max, avg, count)
- Return strongly-typed aggregates
Reference Implementation
Color Mapping System
// Single source of truth for semantic colors
export interface SeverityColors {
badge: string; // 'text-red-600 bg-red-100'
bg: string; // 'bg-red-500'
text: string; // 'text-red-600'
border: string; // 'border-red-500'
dot: string; // 'bg-red-500'
}
const SEVERITY_COLOR_MAP: Record<Severity, SeverityColors> = {
safety_hazard: {
badge: 'text-red-600 bg-red-100',
bg: 'bg-red-500',
text: 'text-red-600',
dot: 'bg-red-500',
border: 'border-red-500',
},
// ... other severities
};
export function getSeverityColors(severity: Severity): SeverityColors {
return SEVERITY_COLOR_MAP[severity];
}
Icon Mapping
export function getSeverityIcon(severity: Severity): string {
const icons: Record<Severity, string> = {
safety_hazard: 'AlertTriangle',
repair_needed: 'Wrench',
maintenance_item: 'Settings',
monitor: 'Eye',
informational: 'Info'
};
return icons[severity];
}
Derived Types
// Base type
export interface Finding {
id: string;
assetId?: string | null;
severity: Severity;
title: string;
}
// Derived type with relationship
export interface FindingWithAsset extends Finding {
asset?: Asset;
}
// Aggregate type
export interface PropertyWithDetails {
property: Property;
inspectionReport: InspectionReport;
findings: Finding[];
assets: Asset[];
}
Formatting Utilities
export function formatCurrency(value: number): string {
return new Intl.NumberFormat('en-US', {
style: 'currency',
currency: 'USD',
minimumFractionDigits: 0,
maximumFractionDigits: 0
}).format(value);
}
export function formatDate(dateString: string): string {
return new Date(dateString).toLocaleDateString('en-US', {
year: 'numeric',
month: 'short',
day: 'numeric'
});
}
export function yearsSince(dateString: string): number {
const date = new Date(dateString);
const now = new Date();
return Math.floor((now.getTime() - date.getTime()) / (365.25 * 24 * 60 * 60 * 1000));
}
Aggregation Patterns
// Group by enum value
export function groupFindingsBySeverity(findings: Finding[]): Record<Severity, Finding[]> {
const grouped: Record<Severity, Finding[]> = {
safety_hazard: [],
repair_needed: [],
maintenance_item: [],
monitor: [],
informational: []
};
findings.forEach(f => grouped[f.severity].push(f));
return grouped;
}
// Sort by priority
export function sortFindingsBySeverity(findings: Finding[]): Finding[] {
const severityOrder: Record<Severity, number> = {
safety_hazard: 0,
repair_needed: 1,
maintenance_item: 2,
monitor: 3,
informational: 4
};
return [...findings].sort((a, b) =>
severityOrder[a.severity] - severityOrder[b.severity]
);
}
Best Practices
- Single Source of Truth: All color/icon mappings in one place with accessor functions
- Multi-Format Colors: Provide badge, bg, text, border, dot variants for flexibility
- Type Safety: Use
Record<EnumType, Value>instead of plain objects - Intl APIs: Use
Intl.NumberFormatandIntl.DateTimeFormatfor localization - Immutability: Use spread operator when sorting/filtering arrays
- Documentation: Add JSDoc comments explaining color choices and data structures
- Colocate Utilities: Keep types and utilities in same file for easy import
Anti-Patterns
- DO NOT use generic color names without semantic meaning (e.g.,
'red'instead of'safety_hazard') - DO NOT inline color classes in components; always use mapping functions
- DO NOT use
anytypes; preferunknownand type guards - DO NOT mutate input arrays in sort/filter functions; always create copies
- DO NOT hardcode date formats; use
Intl.DateTimeFormatfor consistency - DO NOT create separate files for simple utilities; colocate with types
- DO NOT forget to handle null/undefined in optional fields
- DO NOT use snake_case or kebab-case for TypeScript file names; use camelCase
Feedback Loops
- Type Checking: Run
tsc --noEmitto validate types - Runtime Validation: Consider Zod for JSON schema validation at runtime
- Visual Testing: Build Storybook stories to verify color systems
- Data Consistency: Compare aggregated stats with source data counts
- Import Verification: Ensure all utilities are exported and importable
Related Skills
interface-design- Use color systems in React componentsrefactoring-code- Consolidate duplicate color/formatting logic