Skip to content

Commit

Permalink
Simplify BlockMap, Use new datatype
Browse files Browse the repository at this point in the history
  • Loading branch information
hanbyul-here committed May 31, 2024
1 parent 4531c97 commit 899459c
Show file tree
Hide file tree
Showing 6 changed files with 150 additions and 175 deletions.
207 changes: 78 additions & 129 deletions app/scripts/components/common/blocks/block-map.tsx
Original file line number Diff line number Diff line change
@@ -1,11 +1,9 @@
import React, { useMemo, useState, useEffect } from 'react';
import styled from 'styled-components';
import { DatasetDatumFnResolverBag, ProjectionOptions, datasets } from 'veda';
import { ProjectionOptions } from 'veda';
import { MapboxOptions } from 'mapbox-gl';
import * as dateFns from 'date-fns';
import { DatasetLayerCompareInternal } from 'veda';
import {
convertProjectionToMapbox,
projectionDefault,
validateProjectionBlockProps
} from '../map/controls/map-options/projections';
import { Basemap } from '../map/style-generators/basemap';
Expand All @@ -14,14 +12,14 @@ import MapCoordsControl from '../map/controls/coords';
import MapMessage from '../map/map-message';
import {
formatCompareDate,
formatSingleDate,
resolveConfigFunctions
formatSingleDate
} from '../map/utils';
import {
BasemapId,
DEFAULT_MAP_STYLE_URL
} from '../map/controls/map-options/basemap';
import { utcString2userTzDate } from '$utils/date';
import { S_SUCCEEDED } from '$utils/status';
import Map, { Compare, MapControls } from '$components/common/map';
import { validateRangeNum } from '$utils/utils';
import { HintedError } from '$utils/hinted-error';
Expand All @@ -30,13 +28,12 @@ import {
ScaleControl
} from '$components/common/map/controls';
import { Layer } from '$components/exploration/components/map/layer';
import { S_SUCCEEDED } from '$utils/status';
import {
TimelineDataset,
TimelineDatasetSuccess
VizDataset,
VizDatasetSUCCESS
} from '$components/exploration/types.d.ts';

import { reconcileDatasets } from '$components/exploration/data-utils';
import { reconcileVizDataset } from '$components/exploration/data-utils';
import { datasetLayers } from '$components/exploration/data-utils';
import { useReconcileWithStacMetadata } from '$components/exploration/hooks/use-stac-metadata-datasets';

Expand Down Expand Up @@ -130,16 +127,12 @@ function MapBlock(props: MapBlockProps) {
const generatedId = useMemo(() => `map-${++mapInstanceId}`, []);

const {
datasetId,
layerId,
dateTime,
compareDateTime,
compareLabel,
center,
zoom,
projectionId,
projectionCenter,
projectionParallels,
basemapId
} = props;

Expand All @@ -148,98 +141,62 @@ function MapBlock(props: MapBlockProps) {
throw new HintedError('Malformed Map Block', errors);
}

const [baseLayers, setBaseLayers] = useState<TimelineDataset[] | undefined>();
const [compareLayers, setCompareLayers] = useState<
TimelineDataset[] | undefined
>();

const [baseMapStaticData] = reconcileDatasets([layerId], datasetLayers, []);
const baseMapStaticCompareData = baseMapStaticData.data.compare;

let compareLayerId: undefined | string;
if (baseMapStaticCompareData && 'layerId' in baseMapStaticCompareData) {
compareLayerId = baseMapStaticCompareData.layerId;
}

const [compareMapStaticData] = reconcileDatasets(
compareLayerId ? [compareLayerId] : [],
datasetLayers,
[]
);

useReconcileWithStacMetadata([baseMapStaticData], setBaseLayers);
useReconcileWithStacMetadata([compareMapStaticData], setCompareLayers);
const renderCompare = !!compareDateTime;
const [ layers, setLayers ] = useState<VizDatasetSUCCESS[] | undefined >();

const selectedDatetime = dateTime
? utcString2userTzDate(dateTime)
: undefined;

const selectedCompareDatetime = compareDateTime
? utcString2userTzDate(compareDateTime)
: undefined;

const projectionStart = useMemo(() => {
if (projectionId) {
// Ensure that the default center and parallels are used if none are
// provided.
const projection = convertProjectionToMapbox({
id: projectionId,
center: projectionCenter,
parallels: projectionParallels
});
return {
...projection,
id: projectionId
};
} else {
return projectionDefault;
}
}, [projectionId, projectionCenter, projectionParallels]);

const [, setProjection] = useState(projectionStart);

const dataset = datasetId ? datasets[datasetId] : null;

const resolverBag = useMemo<DatasetDatumFnResolverBag>(
() => ({
datetime: selectedDatetime,
compareDatetime: selectedCompareDatetime,
dateFns
}),
[selectedDatetime, selectedCompareDatetime]
);

const [baseLayerResolvedData] = useMemo(() => {
if (!baseLayers || baseLayers.length !== 1) return [null, null];
const baseLayer = baseLayers[0];

if (baseLayer.status !== S_SUCCEEDED) return [null, null];
const baseLayerMDXData = datasetLayers.find(e => e.id === layerId);

// @ REVIEW - Is there any risk not using useMemo for this line?
// const baseLayerMDXWConfig = resolveConfigFunctions(baseLayerMDXData, resolverBag);

const baseMapStaticData = reconcileVizDataset(baseLayerMDXData);
let dataToReconcileWithStac = [baseMapStaticData];

const baseMapStaticCompareData = baseMapStaticData.data.compare as DatasetLayerCompareInternal | null;

// compareData can be null
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
if (baseMapStaticCompareData?.layerId) {
const compareLayerId = baseMapStaticCompareData.layerId;
const compareLayerMDXData = datasetLayers.find(e => e.id === compareLayerId);
const compareMapStaticData = reconcileVizDataset(compareLayerMDXData);
dataToReconcileWithStac = [...dataToReconcileWithStac, compareMapStaticData];
} else {
if (renderCompare) throw Error('There is no compare layer defined.');
}

const bag = { ...resolverBag, raw: baseLayer.data };
const data = resolveConfigFunctions(baseLayer.data, bag);
return [data];
}, [baseLayers, resolverBag]);
useReconcileWithStacMetadata(dataToReconcileWithStac as VizDataset[], setLayers);

const baseDataLayer: TimelineDataset | null = {
data: baseLayerResolvedData
} as unknown as TimelineDataset;
const baseTimeDensity = baseLayerResolvedData?.timeDensity;
const baseLayer = layers?.[0];
const baseTimeDensity = baseLayer?.data.timeDensity;

const compareLayer = layers?.[1];
const compareTimeDensity = compareLayer?.data.timeDensity;

// Resolve data needed for the compare layer once it is loaded.
const [compareLayerResolvedData] = useMemo(() => {
if (!compareLayers || compareLayers.length !== 1) return [null, null];
const compareLayer = compareLayers[0];

if (compareLayer.status !== S_SUCCEEDED) return [null, null];
// Include access to raw data.
const bag = { ...resolverBag, raw: compareLayer.data };
const data = resolveConfigFunctions(compareLayer.data, bag);
return [data];
}, [compareLayers, resolverBag]);

const compareDataLayer: TimelineDataset | null = {
data: compareLayerResolvedData
} as unknown as TimelineDataset;
const compareTimeDensity = compareLayerResolvedData?.timeDensity;
// const [compareLayerResolvedData] = useMemo(() => {
// if (!compareLayers || compareLayers.length !== 1) return [null, null];
// const compareLayer = compareLayers[0];

// if (compareLayer.status !== S_SUCCEEDED) return [null, null];
// // Include access to raw data.
// const bag = { ...resolverBag, raw: compareLayer.data };
// const data = resolveConfigFunctions(compareLayer.data, bag);
// return [data];
// }, [compareLayers, resolverBag]);

// const compareDataLayer: TimelineDataset | null = {
// data: compareLayerResolvedData
// } as unknown as TimelineDataset;
// const compareTimeDensity = compareLayerResolvedData?.time_density;

const mapOptions: Partial<MapboxOptions> = {
style: DEFAULT_MAP_STYLE_URL,
Expand All @@ -263,15 +220,10 @@ function MapBlock(props: MapBlockProps) {
return opts;
};

useEffect(() => {
setProjection(projectionStart);
}, [projectionStart]);

const [mapBasemapId, setMapBasemapId] = useState(basemapId);

useEffect(() => {
if (!basemapId) return;

setMapBasemapId(basemapId);
}, [basemapId]);

Expand All @@ -284,8 +236,8 @@ function MapBlock(props: MapBlockProps) {

const computedCompareLabel = useMemo(() => {
// Use a provided label if it exist.
if (compareLabel && compareLayerResolvedData) {
const providedLabel = compareLayerResolvedData?.mapLabel as string;
if (compareLabel && compareLayer) {
const providedLabel = compareLayer.data.mapLabel as string;
return providedLabel;
}

Expand All @@ -300,7 +252,7 @@ function MapBlock(props: MapBlockProps) {
: null;
}, [
compareLabel,
compareLayerResolvedData,
compareLayer,
selectedDatetime,
compareToDate,
baseTimeDensity,
Expand All @@ -321,33 +273,33 @@ function MapBlock(props: MapBlockProps) {
}}
>
<Basemap basemapStyleId={mapBasemapId} />
{dataset && selectedDatetime && layerId && baseLayerResolvedData && (
{selectedDatetime && (baseLayer?.status === S_SUCCEEDED) && (
<Layer
key={baseLayerResolvedData?.id}
id={`base-${baseLayerResolvedData?.id}`}
dataset={baseDataLayer as unknown as TimelineDatasetSuccess}
key={baseLayer.data.id}
id={`base-${baseLayer.data.id}`}
dataset={baseLayer as VizDatasetSUCCESS}
selectedDay={selectedDatetime}
/>
)}
{baseLayerResolvedData?.legend && (
{baseLayer?.data.legend && (
// Map overlay element
// Layer legend for the active layer.
// @NOTE: LayerLegendContainer is in old mapbox directory, may want to move this over to /map directory once old directory is deprecated
<LayerLegendContainer>
<LayerLegend
id={`base-${baseLayerResolvedData.id}`}
title={baseLayerResolvedData.name}
description={baseLayerResolvedData.description}
{...baseLayerResolvedData.legend}
id={`base-${baseLayer.data.id}`}
title={baseLayer.data.name}
description={baseLayer.data.description}
{...baseLayer.data.legend}
/>
{compareLayerResolvedData?.legend &&
{compareLayer?.data.legend &&
!!selectedCompareDatetime &&
baseLayerResolvedData.id !== compareLayerResolvedData.id && (
baseLayer.data.id !== compareLayer.data.id && (
<LayerLegend
id={`compare-${compareLayerResolvedData.id}`}
title={compareLayerResolvedData.name}
description={compareLayerResolvedData.description}
{...compareLayerResolvedData.legend}
id={`compare-${compareLayer.data.id}`}
title={compareLayer.data.name}
description={compareLayer.data.description}
{...compareLayer.data.legend}
/>
)}
</LayerLegendContainer>
Expand All @@ -356,39 +308,36 @@ function MapBlock(props: MapBlockProps) {
{selectedDatetime && selectedCompareDatetime ? (
<MapMessage
id='compare-message'
active={!!(selectedCompareDatetime && compareLayerResolvedData)}
active={!!(selectedCompareDatetime && compareLayer)}
>
{computedCompareLabel}
</MapMessage>
) : (
<MapMessage
id='single-map-message'
active={!!(selectedDatetime && baseLayerResolvedData)}
active={!!(selectedDatetime && baseLayer)}
>
{selectedDatetime &&
formatSingleDate(
selectedDatetime,
baseLayerResolvedData?.timeDensity
baseLayer?.data.time_density
)}
</MapMessage>
)}
<ScaleControl />
<NavigationControl position='top-left' />
<MapCoordsControl />
</MapControls>
{selectedCompareDatetime && (
{selectedCompareDatetime && (compareLayer?.status === S_SUCCEEDED) && (
<Compare>
<Basemap basemapStyleId={mapBasemapId} />
{dataset &&
selectedCompareDatetime &&
{selectedCompareDatetime &&
layerId &&
compareLayerResolvedData && (
compareLayer && (
<Layer
key={compareLayerResolvedData.id}
id={`compare-${compareLayerResolvedData.id}`}
dataset={
compareDataLayer as unknown as TimelineDatasetSuccess
}
key={compareLayer.data.id}
id={`compare-${compareLayer.data.id}`}
dataset={compareLayer as VizDatasetSUCCESS}
selectedDay={selectedCompareDatetime}
/>
)}
Expand Down
32 changes: 6 additions & 26 deletions app/scripts/components/exploration/components/map/layer.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,8 @@ import React, { useMemo } from 'react';
// Avoid error: node_modules/date-fns/esm/index.js does not export 'default'
import * as dateFns from 'date-fns';

import { TimelineDatasetSuccess } from '../../types.d.ts';
import { TimelineDatasetSuccess, VizDatasetSUCCESS } from '../../types.d.ts';
import { getTimeDensityStartDate } from '../../data-utils';
import {
useTimelineDatasetAtom,
useTimelineDatasetSettings
} from '../../atoms/hooks';

import { resolveConfigFunctions } from '$components/common/map/utils';
import { RasterTimeseries } from '$components/common/map/style-generators/raster-timeseries';
Expand All @@ -17,36 +13,20 @@ import { CMRTimeseries } from '$components/common/map/style-generators/cmr-times

interface LayerProps {
id: string;
dataset: TimelineDatasetSuccess;
dataset: TimelineDatasetSuccess | VizDatasetSUCCESS;
order?: number;
selectedDay: Date;
}

export function Layer(props: LayerProps) {
const { id: layerId, dataset, order, selectedDay } = props;

let isVisible: boolean | undefined;
let opacity: number | undefined;

const datasetAtom = useTimelineDatasetAtom(dataset.data.id);

try {
// @TECH-DEBT: Wrapping this logic with a try/catch because jotai errors because it is unable to find
// 'settings' on undefined value even when dataset has 'settings' key. This is a workaround for now but
// should be revisited. Ideally type should be fine with 'Partial<TimelineDataset>'
const [getSettings] = useTimelineDatasetSettings(datasetAtom);

isVisible = getSettings('isVisible');
opacity = getSettings('opacity');
} catch {
isVisible = true;
opacity = undefined;
}
const { isVisible, opacity } = dataset.settings;

// The date needs to match the dataset's time density.
const relevantDate = useMemo(
() => getTimeDensityStartDate(selectedDay, dataset.data?.timeDensity),
[selectedDay, dataset.data?.timeDensity]
() => {
return getTimeDensityStartDate(selectedDay, dataset.data.timeDensity);
}, [selectedDay, dataset.data.timeDensity]
);

// Resolve config functions.
Expand Down
Loading

0 comments on commit 899459c

Please sign in to comment.