From 2bca1a356956f03d72cf8b9c6a5eafd7c6594c60 Mon Sep 17 00:00:00 2001 From: fxi Date: Mon, 2 Dec 2024 14:07:45 +0100 Subject: [PATCH] Fetch indicator data : work around the 200 item max limitation using loop and chunk --- src/components/data/DataExplorer.tsx | 64 +++++++++++++++--- src/components/data/map/MapPanel.tsx | 97 +++++++++++++++++----------- 2 files changed, 113 insertions(+), 48 deletions(-) diff --git a/src/components/data/DataExplorer.tsx b/src/components/data/DataExplorer.tsx index e2fddd9..3d3346e 100644 --- a/src/components/data/DataExplorer.tsx +++ b/src/components/data/DataExplorer.tsx @@ -14,12 +14,18 @@ const initialFilters: FilterState = { }; const getInitialLanguage = (): Language => { - if (typeof window === 'undefined') return DEFAULT_LANGUAGE; - + if (typeof window === 'undefined') return DEFAULT_LANGUAGE; const storedLang = window.localStorage?.getItem('language') as Language; return storedLang && SUPPORTED_LANGUAGES.includes(storedLang) ? storedLang : DEFAULT_LANGUAGE; }; +// Utility function to chunk array into smaller arrays +const chunk = (arr: T[], size: number): T[][] => { + return Array.from({ length: Math.ceil(arr.length / size) }, (_, i) => + arr.slice(i * size, i * size + size) + ); +}; + export function DataExplorer() { const [language, setLanguage] = useState(DEFAULT_LANGUAGE); const [indicators, setIndicators] = useState([]); @@ -81,21 +87,59 @@ export function DataExplorer() { // Fetch indicator data when selection changes useEffect(() => { - if (!selectedIndicator) return; + if (!selectedIndicator) { + setIndicatorData([]); + return; + } + + const indicatorId = selectedIndicator.id; - async function fetchIndicatorData() { + async function fetchIndicatorDataChunk(offset: number): Promise { + const response = await fetch( + `https://api.unepgrid.ch/stats/v1/indicatorsData?id=eq.${indicatorId}&limit=200&offset=${offset}` + ); + return response.json(); + } + + async function fetchAllIndicatorData() { try { - const response = await fetch( - `https://api.unepgrid.ch/stats/v1/indicatorsData?id=eq.${selectedIndicator.id}` - ); - const data = await response.json(); - setIndicatorData(data); + const CHUNK_SIZE = 200; + const PARALLEL_REQUESTS = 5; // Number of parallel requests to make + let allData: IndicatorData[] = []; + let offset = 0; + let hasMore = true; + + while (hasMore) { + // Create an array of offsets for parallel requests + const offsets = Array.from({ length: PARALLEL_REQUESTS }, (_, i) => offset + i * CHUNK_SIZE); + + // Fetch multiple chunks in parallel + const chunks = await Promise.all( + offsets.map(currentOffset => fetchIndicatorDataChunk(currentOffset)) + ); + + // Process the results + const newData = chunks.flat(); + + // If any chunk is empty or we got less data than expected, we've reached the end + hasMore = chunks.every(chunk => chunk.length === CHUNK_SIZE); + + if (newData.length > 0) { + allData = [...allData, ...newData]; + offset += PARALLEL_REQUESTS * CHUNK_SIZE; + } else { + hasMore = false; + } + } + + setIndicatorData(allData); } catch (err) { console.error("Error fetching indicator data:", err); + setError("Failed to load indicator data"); } } - fetchIndicatorData(); + fetchAllIndicatorData(); }, [selectedIndicator]); // Calculate remaining counts for categories and keywords diff --git a/src/components/data/map/MapPanel.tsx b/src/components/data/map/MapPanel.tsx index e60f9ee..14d6108 100644 --- a/src/components/data/map/MapPanel.tsx +++ b/src/components/data/map/MapPanel.tsx @@ -1,4 +1,6 @@ -{/* Previous imports remain unchanged */} +{ + /* Previous imports remain unchanged */ +} import { useEffect, useRef, useMemo, useState, useCallback } from "react"; import * as d3 from "d3"; import { feature } from "topojson-client"; @@ -8,7 +10,11 @@ import { useTheme } from "../../layout/ThemeProvider"; import { Legend } from "./Legend"; import { MapToolbar } from "./MapToolbar"; import { MapTooltip } from "./MapTooltip"; -import { t, type Language, DEFAULT_LANGUAGE } from '../../../lib/utils/translations'; +import { + t, + type Language, + DEFAULT_LANGUAGE, +} from "../../../lib/utils/translations"; import { throttle, processRegionData, @@ -36,10 +42,8 @@ interface HoveredRegion { } // Shared size scale configuration -const createSizeScale = (extent: [number, number]) => - d3.scaleSqrt() - .domain(extent) - .range([3, 15]); // Reduced maximum size +const createSizeScale = (extent: [number, number]) => + d3.scaleSqrt().domain(extent).range([3, 15]); // Reduced maximum size // Custom Legend for proportional symbols function ProportionalSymbolLegend({ @@ -53,27 +57,28 @@ function ProportionalSymbolLegend({ }) { const [minValue, maxValue] = globalExtent; const format = d3.format(".2~s"); // Use d3 SI-prefix formatting with 2 significant digits - + // Calculate intermediate values const steps = [ minValue, minValue + (maxValue - minValue) * 0.25, minValue + (maxValue - minValue) * 0.5, minValue + (maxValue - minValue) * 0.75, - maxValue + maxValue, ]; - + // SVG dimensions and layout const width = 120; const height = 160; // Increased height to accommodate more circles const margin = { top: 20, right: 40, bottom: 10, left: 10 }; const centerX = margin.left + (width - margin.left - margin.right) / 3; - + // Use shared size scale const sizeScale = createSizeScale(globalExtent); // Calculate vertical spacing - const verticalSpacing = (height - margin.top - margin.bottom) / (steps.length - 1); + const verticalSpacing = + (height - margin.top - margin.bottom) / (steps.length - 1); return (
@@ -129,9 +134,13 @@ export function MapPanel({ data, language }: MapPanelProps) { const currentTransformRef = useRef(d3.zoomIdentity); const graticuleRef = useRef( - d3.geoGraticule() - .step([10, 10]) // Draw lines every 10 degrees for better density - .extent([[-180, -90], [180, 90]]) // Ensure full coverage + d3 + .geoGraticule() + .step([10, 10]) // Draw lines every 10 degrees for better density + .extent([ + [-180, -90], + [180, 90], + ]) // Ensure full coverage ); // All state @@ -140,7 +149,9 @@ export function MapPanel({ data, language }: MapPanelProps) { useState("Mollweide"); const [isLegendVisible, setIsLegendVisible] = useState(true); const [isLatestMode, setIsLatestMode] = useState(false); - const [hoveredRegion, setHoveredRegion] = useState(null); + const [hoveredRegion, setHoveredRegion] = useState( + null + ); // Use theme context for colors const { colors } = useTheme(); @@ -157,8 +168,8 @@ export function MapPanel({ data, language }: MapPanelProps) { // Prepare legend title with year/latest mode const legendTitle = useMemo(() => { - const baseTitle = t("dv.legend",language) // This should be translated based on language - const latest = t("dv.latest", language) + const baseTitle = t("dv.legend", language); // This should be translated based on language + const latest = t("dv.latest", language); return `${baseTitle} ${isLatestMode ? latest : selectedYear}`; }, [isLatestMode, selectedYear]); @@ -231,22 +242,25 @@ export function MapPanel({ data, language }: MapPanelProps) { }, [updateRegionPaths]); // Handle zoom - const handleZoom = useCallback((event: d3.D3ZoomEvent) => { - if (!projectionRef.current || !pathGeneratorRef.current) return; + const handleZoom = useCallback( + (event: d3.D3ZoomEvent) => { + if (!projectionRef.current || !pathGeneratorRef.current) return; - currentTransformRef.current = event.transform; - const container = svgRef.current?.parentElement; - if (!container) return; + currentTransformRef.current = event.transform; + const container = svgRef.current?.parentElement; + if (!container) return; - const width = container.clientWidth; - const baseScale = width / 6; + const width = container.clientWidth; + const baseScale = width / 6; - // Update projection scale based on zoom transform - const newScale = baseScale * event.transform.k; - projectionRef.current.scale(newScale); - pathGeneratorRef.current = d3.geoPath(projectionRef.current); - scheduleUpdate(); - }, [scheduleUpdate]); + // Update projection scale based on zoom transform + const newScale = baseScale * event.transform.k; + projectionRef.current.scale(newScale); + pathGeneratorRef.current = d3.geoPath(projectionRef.current); + scheduleUpdate(); + }, + [scheduleUpdate] + ); // Drag interaction handlers const handleDragStart = useCallback(() => { @@ -286,7 +300,12 @@ export function MapPanel({ data, language }: MapPanelProps) { // Handle projection change const handleProjectionChange = useCallback( (newProjection: ProjectionType) => { - if (!svgRef.current || !worldDataRef.current || !projectionRef.current || !zoomRef.current) + if ( + !svgRef.current || + !worldDataRef.current || + !projectionRef.current || + !zoomRef.current + ) return; const container = svgRef.current.parentElement; @@ -374,12 +393,12 @@ export function MapPanel({ data, language }: MapPanelProps) { const containerRect = containerRef.current.getBoundingClientRect(); const mouseX = event.clientX - containerRect.left; const mouseY = event.clientY - containerRect.top; - + setHoveredRegion({ name: regionInfo.name, value: regionInfo.value, x: mouseX, - y: mouseY + y: mouseY, }); } }; @@ -389,11 +408,11 @@ export function MapPanel({ data, language }: MapPanelProps) { const containerRect = containerRef.current.getBoundingClientRect(); const mouseX = event.clientX - containerRect.left; const mouseY = event.clientY - containerRect.top; - + setHoveredRegion({ ...hoveredRegion, x: mouseX, - y: mouseY + y: mouseY, }); } }; @@ -418,6 +437,7 @@ export function MapPanel({ data, language }: MapPanelProps) { const response = await d3.json( "/grid_stat/world-110m.json" ); + debugger; if (!response) { throw new Error("Failed to load world map data"); } @@ -435,13 +455,14 @@ export function MapPanel({ data, language }: MapPanelProps) { .style("height", "auto"); // Initialize zoom behavior - const zoom = d3.zoom() + const zoom = d3 + .zoom() .scaleExtent([0.5, 8]) .on("zoom", handleZoom) - .filter(event => { + .filter((event) => { // Only handle wheel events for zooming // Ignore double-click zoom - return event.type === 'wheel' && !event.button; + return event.type === "wheel" && !event.button; }); zoomRef.current = zoom;