demarches-normaliennes/app/javascript/components/MapEditor/components/DrawLayer.tsx
2024-05-22 08:53:51 +02:00

230 lines
6.2 KiB
TypeScript

import { useCallback, useRef, useEffect } from 'react';
import type { LngLatBoundsLike, LngLatLike } from 'maplibre-gl';
import DrawControl from '@mapbox/mapbox-gl-draw';
import type { FeatureCollection, Feature, Point } from 'geojson';
import { useMapLibre } from '../../shared/maplibre/MapLibre';
import {
useFitBounds,
useFitBoundsNoFly,
useEvent,
useMapEvent,
useFlyTo
} from '../../shared/maplibre/hooks';
import {
filterFeatureCollection,
findFeature,
getBounds
} from '../../shared/maplibre/utils';
import {
SOURCE_SELECTION_UTILISATEUR,
CreateFeatures,
UpdateFatures,
DeleteFeatures
} from '../hooks';
export function DrawLayer({
featureCollection,
createFeatures,
updateFeatures,
deleteFeatures,
enabled
}: {
featureCollection: FeatureCollection;
createFeatures: CreateFeatures;
updateFeatures: UpdateFatures;
deleteFeatures: DeleteFeatures;
enabled: boolean;
}) {
const map = useMapLibre();
const drawRef = useRef<DrawControl | null>();
useEffect(() => {
if (!drawRef.current && enabled) {
const draw = new DrawControl({
displayControlsDefault: false,
controls: {
point: true,
line_string: true,
polygon: true,
trash: true
}
});
// We use mapbox-draw plugin with maplibre. They are compatible but types are not.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
map.addControl(draw as any, 'top-left');
draw.set(
filterFeatureCollection(featureCollection, SOURCE_SELECTION_UTILISATEUR)
);
drawRef.current = draw;
for (const [selector, translation] of translations) {
const element = document.querySelector(selector);
if (element) {
element.setAttribute('title', translation);
}
}
}
return () => {
if (drawRef.current) {
// We use mapbox-draw plugin with maplibre. They are compatible but types are not.
// eslint-disable-next-line @typescript-eslint/no-explicit-any
map.removeControl(drawRef.current as any);
drawRef.current = null;
}
};
// We only want to rerender draw layer on component mount or when the layer is toggled.
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [map, enabled]);
const onSetId = useCallback(
({ detail }: CustomEvent<{ lid: string; id: string }>) => {
drawRef.current?.setFeatureProperty(detail.lid, 'id', detail.id);
},
[]
);
const onAddFeature = useCallback(
({ detail }: CustomEvent<{ feature: Feature }>) => {
drawRef.current?.add(detail.feature);
},
[]
);
const onDeleteFature = useCallback(
({ detail }: CustomEvent<{ id: string }>) => {
drawRef.current?.delete(detail.id);
},
[]
);
useMapEvent('draw.create', createFeatures);
useMapEvent('draw.update', updateFeatures);
useMapEvent('draw.delete', deleteFeatures);
useEvent('map:internal:draw:setId', onSetId);
useEvent('map:internal:draw:add', onAddFeature);
useEvent('map:internal:draw:delete', onDeleteFature);
useExternalEvents(featureCollection, {
createFeatures,
updateFeatures,
deleteFeatures
});
return null;
}
function useExternalEvents(
featureCollection: FeatureCollection,
{
createFeatures,
updateFeatures,
deleteFeatures
}: {
createFeatures: CreateFeatures;
updateFeatures: UpdateFatures;
deleteFeatures: DeleteFeatures;
}
) {
const fitBounds = useFitBounds();
const fitBoundsNoFly = useFitBoundsNoFly();
const flyTo = useFlyTo();
const onFeatureFocus = useCallback(
({ detail }: CustomEvent<{ id: string; bbox: LngLatBoundsLike }>) => {
const { id, bbox } = detail;
if (id) {
const feature = findFeature(featureCollection, id);
if (feature) {
fitBounds(getBounds(feature.geometry));
}
} else if (bbox) {
fitBounds(bbox);
}
},
[featureCollection, fitBounds]
);
const onZoomFocus = useCallback(
({
detail
}: CustomEvent<{
feature: Feature<Point>;
featureCollection: FeatureCollection;
}>) => {
if (detail.feature && detail.featureCollection == featureCollection) {
flyTo(17, detail.feature.geometry.coordinates as LngLatLike);
}
},
[flyTo, featureCollection]
);
const onFeatureCreate = useCallback(
({
detail
}: CustomEvent<{
feature: Feature;
featureCollection: FeatureCollection;
}>) => {
const { feature } = detail;
const { geometry, properties } = feature;
if (
feature &&
feature.geometry &&
detail.featureCollection == featureCollection
) {
createFeatures({
features: [{ type: 'Feature', geometry, properties }],
external: true
});
}
},
[createFeatures, featureCollection]
);
const onFeatureUpdate = useCallback(
({
detail
}: CustomEvent<{ id: string; properties: Feature['properties'] }>) => {
const { id, properties } = detail;
const feature = findFeature(featureCollection, id);
if (feature) {
feature.properties = { ...feature.properties, ...properties };
updateFeatures({ features: [feature], external: true });
}
},
[featureCollection, updateFeatures]
);
const onFeatureDelete = useCallback(
({ detail }: CustomEvent<{ id: string }>) => {
const { id } = detail;
const feature = findFeature(featureCollection, id);
if (feature) {
deleteFeatures({ features: [feature], external: true });
}
},
[featureCollection, deleteFeatures]
);
useEffect(() => {
fitBoundsNoFly(featureCollection.bbox as LngLatBoundsLike);
// We only want to zoom on bbox on component mount.
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [fitBoundsNoFly]);
useEvent('map:feature:focus', onFeatureFocus);
useEvent('map:feature:create', onFeatureCreate);
useEvent('map:feature:update', onFeatureUpdate);
useEvent('map:feature:delete', onFeatureDelete);
useEvent('map:zoom', onZoomFocus);
}
const translations = [
['.mapbox-gl-draw_line', 'Tracer une ligne'],
['.mapbox-gl-draw_polygon', 'Dessiner un polygone'],
['.mapbox-gl-draw_point', 'Ajouter un point'],
['.mapbox-gl-draw_trash', 'Supprimer']
];