diff --git a/client/src/components/map/provider.tsx b/client/src/components/map/provider.tsx index 4ea811e..b7c9516 100644 --- a/client/src/components/map/provider.tsx +++ b/client/src/components/map/provider.tsx @@ -144,7 +144,6 @@ export const useDeckMapboxOverlay = ({ useEffect(() => { if (!layer) return; return () => { - console.log("removeLayer", i); removeLayer(i); }; }, [i, removeLayer]); // eslint-disable-line react-hooks/exhaustive-deps diff --git a/client/src/containers/datasets/components/temporal/absolute.tsx b/client/src/containers/datasets/components/temporal/absolute.tsx index d02e670..8b2acf7 100644 --- a/client/src/containers/datasets/components/temporal/absolute.tsx +++ b/client/src/containers/datasets/components/temporal/absolute.tsx @@ -24,20 +24,30 @@ const isCorrectTimeSelect = (timeSelect: unknown): timeSelect is [number, number timeSelect.every((t) => typeof t === "number") ); }; + +const isCorrectTimeValues = (timeValues: unknown): timeValues is number[] => { + return Array.isArray(timeValues) && timeValues.every((t) => typeof t === "number"); +}; export const TemporalDatasetItem = ({ layer }: TemporalDatasetItemProps) => { const timeSelect = (layer?.attributes?.params_config as Record[])?.find( (p) => p.key === "time-select", )?.default; + const timeValues = (layer?.attributes?.params_config as Record[])?.find( + (p) => p.key === "time-values", + )?.default; const defaultSelected = (layer?.attributes?.params_config as Record[])?.find( - (p) => p.key === "year", + (p) => p.key === "startYear", )?.default; const [layersSettings, setLayersSettings] = useSyncLayersSettings(); const [layers] = useSyncLayers(); - const options = isCorrectTimeSelect(timeSelect) - ? Array.from({ length: timeSelect[1] - timeSelect[0] + 1 }, (_, i) => timeSelect[0] + i) - : undefined; + const options = + timeValues && isCorrectTimeValues(timeValues) + ? timeValues + : isCorrectTimeSelect(timeSelect) + ? Array.from({ length: timeSelect[1] - timeSelect[0] + 1 }, (_, i) => timeSelect[0] + i) + : undefined; const layerSlug = layer?.attributes?.slug; @@ -48,13 +58,13 @@ export const TemporalDatasetItem = ({ layer }: TemporalDatasetItemProps) => { ...prev, [layerSlug]: { ...(prev ? prev[layerSlug] : {}), - year: parseInt(value), + startYear: parseInt(value), }, }; }); }; - const value = layerSlug && (layersSettings?.[layerSlug]?.year as string | undefined); + const value = layerSlug && (layersSettings?.[layerSlug]?.startYear as string | undefined); const defaultValue = typeof defaultSelected === "number" ? `${defaultSelected}` : undefined; const isDisabled = !layerSlug || !layers?.includes(layerSlug); diff --git a/client/src/containers/datasets/components/temporal/changes.tsx b/client/src/containers/datasets/components/temporal/changes.tsx index 37ee5fc..6878acd 100644 --- a/client/src/containers/datasets/components/temporal/changes.tsx +++ b/client/src/containers/datasets/components/temporal/changes.tsx @@ -7,7 +7,7 @@ import { SelectValue, } from "@/components/ui/select"; import { useSyncLayers, useSyncLayersSettings } from "@/store/map"; -import { DefaultLayerComponent, LayerListResponseDataItem } from "@/types/generated/strapi.schemas"; +import { LayerListResponseDataItem } from "@/types/generated/strapi.schemas"; import { CalendarDaysIcon } from "lucide-react"; import { useTranslations } from "next-intl"; import { useEffect, useMemo } from "react"; @@ -20,6 +20,10 @@ const isCorrectTimeSelect = (timeSelect: unknown): timeSelect is [number, number ); }; +const isCorrectTimeValues = (timeValues: unknown): timeValues is number[] => { + return Array.isArray(timeValues) && timeValues.every((t) => typeof t === "number"); +}; + type TemporalDatasetItemSelectProps = { value?: number | undefined; disabled?: boolean; @@ -59,19 +63,27 @@ const TemporalDatasetItemSelect = ({ }; const _getOptions = (params_config: unknown, start?: number, end?: number) => { + const timeValues = (params_config as Record[])?.find( + (p) => p.key === "time-values", + )?.default; const timeSelect = (params_config as Record[])?.find( (p) => p.key === "time-select", )?.default; - return isCorrectTimeSelect(timeSelect) - ? Array.from({ length: timeSelect[1] - timeSelect[0] + 1 }, (_, i) => { - const value = timeSelect[0] + i; - return { - value, - disabled: (!!start && value <= Number(start)) || (!!end && value >= Number(end)), - }; - }) - : []; + return timeValues && isCorrectTimeValues(timeValues) + ? timeValues.map((value) => ({ + value, + disabled: (!!start && value <= Number(start)) || (!!end && value >= Number(end)), + })) + : isCorrectTimeSelect(timeSelect) + ? Array.from({ length: timeSelect[1] - timeSelect[0] + 1 }, (_, i) => { + const value = timeSelect[0] + i; + return { + value, + disabled: (!!start && value <= Number(start)) || (!!end && value >= Number(end)), + }; + }) + : []; }; const selectTypes = ["absolute", "changes"] as const; diff --git a/client/src/containers/datasets/item.tsx b/client/src/containers/datasets/item.tsx index f673406..433115a 100644 --- a/client/src/containers/datasets/item.tsx +++ b/client/src/containers/datasets/item.tsx @@ -46,6 +46,12 @@ const DatasetsItem = ({ attributes, className }: DatasetsItemProps) => { { query: { enabled: !!datasetLayers.length, + select: (data) => ({ + ...data, + data: data.data?.sort( + (a, b) => datasetLayers.indexOf(a.id) - datasetLayers.indexOf(b.id), + ), + }), }, }, ); diff --git a/client/src/containers/map/layer-manager/index.tsx b/client/src/containers/map/layer-manager/index.tsx index c563399..5fb256e 100644 --- a/client/src/containers/map/layer-manager/index.tsx +++ b/client/src/containers/map/layer-manager/index.tsx @@ -84,6 +84,7 @@ const LayerManager = () => { */} {LAYERS.map((l, i) => { const beforeId = i === 0 ? baseLayer : `${LAYERS[i - 1]}-layer`; + return ( ' + '' + +'' + +export const GriddedLivestockTotalStyle = + +'' + + '' + + '' + + '' + + '' + + '' + + '' + + '' + + '' + + '' + + '' + '' \ No newline at end of file diff --git a/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-chicken.ts b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-chicken.ts new file mode 100644 index 0000000..6049f61 --- /dev/null +++ b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-chicken.ts @@ -0,0 +1,31 @@ +import { CategoricalDataset } from './earth-engine-dataset'; +import ee from '@google/earthengine'; +import { EarthEngineUtils } from "../earth-engine-utils"; +import { GriddedLivestockCommonStyles} from "./common-styles-utils"; + +export const GriddedLivestockChicken: CategoricalDataset = { + assetPath: { + default: "projects/gmvad-grass/assets/5_Ch_2015_Da" + + }, + + bandName: 'b1', + + sldStyles: GriddedLivestockCommonStyles, + + + areYearsValid(startYear?: number, endYear?: number): boolean { + // This Asset is static, and year selector is irrelevant + return true; + }, + + getEEAsset() { + return ee.Image(this.assetPath.default); + }, + + async getMapUrl(z, x, y) { + const image = this.getEEAsset().select(this.bandName).sldStyle(this.sldStyles); + const mapId = await EarthEngineUtils.getMapId(image); + return ee.data.getTileUrl(mapId, x, y, z); + }, +}; diff --git a/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-duck.ts b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-duck.ts new file mode 100644 index 0000000..cdabb27 --- /dev/null +++ b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-duck.ts @@ -0,0 +1,31 @@ +import { CategoricalDataset } from './earth-engine-dataset'; +import ee from '@google/earthengine'; +import { EarthEngineUtils } from "../earth-engine-utils"; +import { GriddedLivestockCommonStyles} from "./common-styles-utils"; + +export const GriddedLivestockDuck: CategoricalDataset = { + assetPath: { + default: "projects/gmvad-grass/assets/5_Dk_2015_Da" + + }, + + bandName: 'b1', + + sldStyles: GriddedLivestockCommonStyles, + + + areYearsValid(startYear?: number, endYear?: number): boolean { + // This Asset is static, and year selector is irrelevant + return true; + }, + + getEEAsset() { + return ee.Image(this.assetPath.default); + }, + + async getMapUrl(z, x, y) { + const image = this.getEEAsset().select(this.bandName).sldStyle(this.sldStyles); + const mapId = await EarthEngineUtils.getMapId(image); + return ee.data.getTileUrl(mapId, x, y, z); + }, +}; diff --git a/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-pig.ts b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-pig.ts new file mode 100644 index 0000000..2eb9acf --- /dev/null +++ b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-pig.ts @@ -0,0 +1,31 @@ +import { CategoricalDataset } from './earth-engine-dataset'; +import ee from '@google/earthengine'; +import { EarthEngineUtils } from "../earth-engine-utils"; +import { GriddedLivestockCommonStyles} from "./common-styles-utils"; + +export const GriddedLivestockPig: CategoricalDataset = { + assetPath: { + default: "projects/gmvad-grass/assets/5_Pg_2015_Da" + + }, + + bandName: 'b1', + + sldStyles: GriddedLivestockCommonStyles, + + + areYearsValid(startYear?: number, endYear?: number): boolean { + // This Asset is static, and year selector is irrelevant + return true; + }, + + getEEAsset() { + return ee.Image(this.assetPath.default); + }, + + async getMapUrl(z, x, y) { + const image = this.getEEAsset().select(this.bandName).sldStyle(this.sldStyles); + const mapId = await EarthEngineUtils.getMapId(image); + return ee.data.getTileUrl(mapId, x, y, z); + }, +}; diff --git a/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-total.ts b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-total.ts new file mode 100644 index 0000000..22c4eb5 --- /dev/null +++ b/cloud_functions/earth_engine_tiler/src/geeAssets/gridded-livestock-total.ts @@ -0,0 +1,54 @@ +import { CategoricalDataset } from './earth-engine-dataset'; +import ee from '@google/earthengine'; +import { EarthEngineUtils } from "../earth-engine-utils"; +import { GriddedLivestockBuffalo } from './gridded-livestock-buffalo'; +import { GriddedLivestockCattle } from './gridded-livestock-cattle'; +import { GriddedLivestockChicken } from './gridded-livestock-chicken'; +import { GriddedLivestockDuck } from './gridded-livestock-duck'; +import { GriddedLivestockGoat } from './gridded-livestock-goat'; +import { GriddedLivestockHorse } from './gridded-livestock-horse'; +import { GriddedLivestockPig } from './gridded-livestock-pig'; +import { GriddedLivestockSheep } from './gridded-livestock-sheep'; +import { GriddedLivestockTotalStyle } from "./common-styles-utils"; + +export const GriddedLivestockTotal: CategoricalDataset = { + assetPath: { + }, + bandName: 'b1', + sldStyles: GriddedLivestockTotalStyle, + + areYearsValid(startYear?: number, endYear?: number): boolean { + // This Asset is static, and year selector is irrelevant + return true; + }, + + // Unmask(0) gives 0 to no data pixels so that they can be included in the sum + getEEAsset() { + const buffaloImage = ee.Image(GriddedLivestockBuffalo.assetPath.default).select(GriddedLivestockBuffalo.bandName).unmask(0); + const cattleImage = ee.Image(GriddedLivestockCattle.assetPath.default).select(GriddedLivestockCattle.bandName).unmask(0); + const chickenImage = ee.Image(GriddedLivestockChicken.assetPath.default).select(GriddedLivestockChicken.bandName).unmask(0); + const duckImage = ee.Image(GriddedLivestockDuck.assetPath.default).select(GriddedLivestockDuck.bandName).unmask(0); + const goatImage = ee.Image(GriddedLivestockGoat.assetPath.default).select(GriddedLivestockGoat.bandName).unmask(0); + const horseImage = ee.Image(GriddedLivestockHorse.assetPath.default).select(GriddedLivestockHorse.bandName).unmask(0); + const pigImage = ee.Image(GriddedLivestockPig.assetPath.default).select(GriddedLivestockPig.bandName).unmask(0); + const sheepImage = ee.Image(GriddedLivestockSheep.assetPath.default).select(GriddedLivestockSheep.bandName).unmask(0); + + // Sum all the images + const totalImage = buffaloImage + .add(cattleImage) + .add(chickenImage) + .add(duckImage) + .add(goatImage) + .add(horseImage) + .add(pigImage) + .add(sheepImage); + + return totalImage; + }, + + async getMapUrl(z, x, y) { + const image = this.getEEAsset().sldStyle(this.sldStyles); + const mapId = await EarthEngineUtils.getMapId(image); + return ee.data.getTileUrl(mapId, x, y, z); + }, +}; \ No newline at end of file diff --git a/cloud_functions/earth_engine_tiler/src/index.ts b/cloud_functions/earth_engine_tiler/src/index.ts index 53f4340..e830421 100644 --- a/cloud_functions/earth_engine_tiler/src/index.ts +++ b/cloud_functions/earth_engine_tiler/src/index.ts @@ -8,10 +8,14 @@ import {ModisNetPrimaryProductionChange} from './geeAssets/modis-net-primary-pro import {AnthropogenicBiomes} from './geeAssets/anthropogenic-biomes'; import {LivestockProductionSystems} from './geeAssets/livestock-production-systems'; import {ForestLoss} from './geeAssets/forest-loss'; +import {GriddedLivestockTotal} from './geeAssets/gridded-livestock-total'; +import {GriddedLivestockBuffalo} from './geeAssets/gridded-livestock-buffalo'; import {GriddedLivestockCattle} from './geeAssets/gridded-livestock-cattle'; +import {GriddedLivestockChicken} from './geeAssets/gridded-livestock-chicken'; +import {GriddedLivestockDuck} from './geeAssets/gridded-livestock-duck'; import {GriddedLivestockGoat} from './geeAssets/gridded-livestock-goat'; -import {GriddedLivestockHorse} from './geeAssets/gridded-livestock-horse'; -import {GriddedLivestockBuffalo} from './geeAssets/gridded-livestock-buffalo'; +import {GriddedLivestockHorse} from './geeAssets/gridded-livestock-horse'; +import {GriddedLivestockPig} from './geeAssets/gridded-livestock-pig'; import {GriddedLivestockSheep} from './geeAssets/gridded-livestock-sheep'; import {EarthEngineDataset} from "./geeAssets/earth-engine-dataset"; import {TileRequestDTO, Tilesets} from "./tile-request.dto"; @@ -26,10 +30,14 @@ const assets: Record = { [Tilesets.anthropogenic_biomes]: AnthropogenicBiomes, [Tilesets.livestock_production_systems]: LivestockProductionSystems, [Tilesets.forest_loss]: ForestLoss, + [Tilesets.gridded_livestock_total]: GriddedLivestockTotal, + [Tilesets.gridded_livestock_buffalo]: GriddedLivestockBuffalo, [Tilesets.gridded_livestock_cattle]: GriddedLivestockCattle, + [Tilesets.gridded_livestock_chicken]: GriddedLivestockChicken, + [Tilesets.gridded_livestock_duck]: GriddedLivestockDuck, [Tilesets.gridded_livestock_goat]: GriddedLivestockGoat, [Tilesets.gridded_livestock_horse]: GriddedLivestockHorse, - [Tilesets.gridded_livestock_buffalo]: GriddedLivestockBuffalo, + [Tilesets.gridded_livestock_pig]: GriddedLivestockPig, [Tilesets.gridded_livestock_sheep]: GriddedLivestockSheep, } diff --git a/cloud_functions/earth_engine_tiler/src/tile-request.dto.ts b/cloud_functions/earth_engine_tiler/src/tile-request.dto.ts index fc70818..f272e3a 100644 --- a/cloud_functions/earth_engine_tiler/src/tile-request.dto.ts +++ b/cloud_functions/earth_engine_tiler/src/tile-request.dto.ts @@ -7,10 +7,14 @@ export enum Tilesets { anthropogenic_biomes = "anthropogenic_biomes", livestock_production_systems = "livestock_production_systems", forest_loss = "forest_loss", + gridded_livestock_total = "gridded_livestock_total", + gridded_livestock_buffalo = "gridded_livestock_buffalo", gridded_livestock_cattle = "gridded_livestock_cattle", + gridded_livestock_chicken = "gridded_livestock_chicken", + gridded_livestock_duck = "gridded_livestock_duck", gridded_livestock_goat = "gridded_livestock_goat", gridded_livestock_horse = "gridded_livestock_horse", - gridded_livestock_buffalo = "gridded_livestock_buffalo", + gridded_livestock_pig = "gridded_livestock_pig", gridded_livestock_sheep = "gridded_livestock_sheep" }