/** * HomeView Component * Client-side home page view with search and filtering */ 'use client'; import { useState, useEffect } from 'react'; import { useSearchParams } from 'next/navigation'; import { FilterBar } from './FilterBar'; import { SchoolRow } from './SchoolRow'; import { SchoolMap } from './SchoolMap'; import { Pagination } from './Pagination'; import { EmptyState } from './EmptyState'; import { useComparisonContext } from '@/context/ComparisonContext'; import type { SchoolsResponse, Filters, School } from '@/lib/types'; import styles from './HomeView.module.css'; interface HomeViewProps { initialSchools: SchoolsResponse; filters: Filters; totalSchools?: number | null; } export function HomeView({ initialSchools, filters, totalSchools }: HomeViewProps) { const searchParams = useSearchParams(); const { addSchool, removeSchool, selectedSchools } = useComparisonContext(); const [resultsView, setResultsView] = useState<'list' | 'map'>('list'); const [selectedMapSchool, setSelectedMapSchool] = useState(null); const [sortOrder, setSortOrder] = useState('default'); const hasSearch = searchParams.get('search') || searchParams.get('postcode'); const isLocationSearch = !!searchParams.get('postcode'); const isSearchActive = !!(hasSearch || searchParams.get('local_authority') || searchParams.get('school_type')); // Close bottom sheet if we change views or search useEffect(() => { setSelectedMapSchool(null); }, [resultsView, searchParams]); const sortedSchools = [...initialSchools.schools].sort((a, b) => { if (sortOrder === 'rwm_desc') return (b.rwm_expected_pct ?? -Infinity) - (a.rwm_expected_pct ?? -Infinity); if (sortOrder === 'rwm_asc') return (a.rwm_expected_pct ?? Infinity) - (b.rwm_expected_pct ?? Infinity); if (sortOrder === 'distance') return (a.distance ?? Infinity) - (b.distance ?? Infinity); if (sortOrder === 'name_asc') return a.school_name.localeCompare(b.school_name); return 0; }); return (
{/* Combined Hero + Search and Filters */} {!isSearchActive && (

Compare Primary School Performance

Search and compare KS2 results for thousands of schools across England

)} {/* Discovery section shown on landing page before any search */} {!isSearchActive && initialSchools.schools.length === 0 && (
{totalSchools &&

{totalSchools.toLocaleString()}+ primary schools across England

}

Try searching for a school name, or enter a postcode to find schools near you.

Quick searches: {['Manchester', 'Bristol', 'Leeds', 'Birmingham'].map(city => ( {city} ))}
)} {/* Location Info Banner with View Toggle */} {isLocationSearch && initialSchools.location_info && (
Showing schools within {(initialSchools.location_info.radius / 1.60934).toFixed(1)} miles of{' '} {initialSchools.location_info.postcode}
{initialSchools.schools.length > 0 && (
)}
)} {/* Results Section */}
{!hasSearch && initialSchools.schools.length > 0 && (

Featured Schools

Explore schools from across England

)} {hasSearch && resultsView === 'list' && (

{initialSchools.total.toLocaleString()} school {initialSchools.total !== 1 ? 's' : ''} found

)} {isSearchActive && (
{searchParams.get('search') && Search: {searchParams.get('search')} { e.preventDefault(); }}>×} {searchParams.get('local_authority') && {searchParams.get('local_authority')}} {searchParams.get('school_type') && {searchParams.get('school_type')}} {searchParams.get('postcode') && Near {searchParams.get('postcode')} ({(parseFloat(searchParams.get('radius') || '1.6') / 1.60934).toFixed(1)} mi)}
)} {initialSchools.schools.length === 0 && isSearchActive ? ( { window.location.href = '/'; }, }} /> ) : initialSchools.schools.length > 0 && resultsView === 'map' && isLocationSearch ? ( /* Map View Layout */
{initialSchools.schools.map((school) => (
s.urn === school.urn)} />
))}
{/* Mobile Bottom Sheet for Selected Map Pin */} {selectedMapSchool && (
s.urn === selectedMapSchool.urn)} />
)}
) : ( /* List View Layout */ <>
{sortedSchools.map((school) => ( s.urn === school.urn)} /> ))}
{initialSchools.total_pages > 1 && ( )} )}
); } /* Compact School Item for Map View */ interface CompactSchoolItemProps { school: School; onAddToCompare: (school: School) => void; isInCompare: boolean; } function CompactSchoolItem({ school, onAddToCompare, isInCompare }: CompactSchoolItemProps) { return (
{school.school_name} {school.distance !== undefined && school.distance !== null && ( {(school.distance / 1.60934).toFixed(1)} mi )}
{school.school_type && {school.school_type}} {school.local_authority && {school.local_authority}}
{school.rwm_expected_pct !== null ? `${school.rwm_expected_pct}%` : '-'} RWM {school.total_pupils || '-'} pupils
Details
); }