You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
137 lines
3.4 KiB
137 lines
3.4 KiB
import type React from 'react';
|
|
import { useEffect, useRef } from 'react';
|
|
|
|
import { ScaleControl } from 'mapbox-gl';
|
|
|
|
import { MapboxContext } from '@components/MapBox/mapboxContext';
|
|
import { MapStyles } from '@core/mapStyles';
|
|
import {
|
|
setBearing,
|
|
setLatLng,
|
|
setMapStyle,
|
|
setPitch,
|
|
setZoom,
|
|
} from '@core/slices/mapSlice';
|
|
import { useAppDispatch } from '@hooks/useAppDispatch';
|
|
import { useAppSelector } from '@hooks/useAppSelector';
|
|
import { useCreateMapbox } from '@hooks/useCreateMapbox';
|
|
|
|
export type MapboxProviderProps = {
|
|
children: React.ReactNode;
|
|
};
|
|
|
|
export const MapboxProvider = ({
|
|
children,
|
|
}: MapboxProviderProps): JSX.Element => {
|
|
const darkMode = useAppSelector((state) => state.app.darkMode);
|
|
const mapState = useAppSelector((state) => state.map);
|
|
const dispatch = useAppDispatch();
|
|
const ref = useRef<HTMLDivElement>(null);
|
|
|
|
const map = useCreateMapbox({
|
|
ref,
|
|
accessToken:
|
|
'pk.eyJ1Ijoic2FjaGF3IiwiYSI6ImNrNW9meXozZjBsdW0zbHBjM2FnNnV6cmsifQ.3E4n8eFGD9ZOFo-XDVeZnQ',
|
|
options: {
|
|
center: mapState.latLng,
|
|
zoom: mapState.zoom,
|
|
bearing: mapState.bearing,
|
|
pitch: mapState.pitch,
|
|
style: MapStyles[mapState.style].data,
|
|
},
|
|
});
|
|
|
|
useEffect(() => {
|
|
map?.on('load', () => {
|
|
map.addControl(new ScaleControl());
|
|
});
|
|
map?.on('styledata', () => {
|
|
if (!map.getSource('mapbox-dem')) {
|
|
map.addSource('mapbox-dem', {
|
|
type: 'raster-dem',
|
|
url: 'mapbox://mapbox.mapbox-terrain-dem-v1',
|
|
tileSize: 512,
|
|
maxzoom: 14,
|
|
});
|
|
}
|
|
map.setTerrain({
|
|
source: 'mapbox-dem',
|
|
exaggeration: mapState.exaggeration ? 1.5 : 0,
|
|
});
|
|
});
|
|
map?.on('dragend', (e) => {
|
|
dispatch(setLatLng(e.target.getCenter()));
|
|
});
|
|
map?.on('zoomend', (e) => {
|
|
dispatch(setZoom(e.target.getZoom()));
|
|
});
|
|
map?.on('rotate', (e) => {
|
|
dispatch(setBearing(e.target.getBearing()));
|
|
});
|
|
map?.on('pitch', (e) => {
|
|
dispatch(setPitch(e.target.getPitch()));
|
|
});
|
|
}, [dispatch, map, mapState.exaggeration]);
|
|
|
|
useEffect(() => {
|
|
const center = map?.getCenter();
|
|
if (center !== mapState.latLng) {
|
|
map?.setCenter(mapState.latLng);
|
|
}
|
|
}, [map, mapState.latLng]);
|
|
|
|
useEffect(() => {
|
|
if (['Light', 'Dark'].includes(mapState.style)) {
|
|
dispatch(setMapStyle(darkMode ? 'Dark' : 'Light'));
|
|
}
|
|
}, [dispatch, darkMode, mapState.style]);
|
|
|
|
/**
|
|
* Hill Shading
|
|
*/
|
|
useEffect(() => {
|
|
if (map?.loaded()) {
|
|
if (mapState.hillShade) {
|
|
map.addLayer(
|
|
{
|
|
id: 'hillshading',
|
|
source: 'mapbox-dem',
|
|
type: 'hillshade',
|
|
// insert below waterway-river-canal-shadow;
|
|
// where hillshading sits in the Mapbox Outdoors style
|
|
},
|
|
'waterway-river-canal-shadow',
|
|
);
|
|
} else {
|
|
map.removeLayer('hillshading');
|
|
}
|
|
}
|
|
}, [map, mapState.hillShade]);
|
|
|
|
/**
|
|
* Exaggeration
|
|
*/
|
|
useEffect(() => {
|
|
if (map?.loaded()) {
|
|
map.setTerrain({
|
|
source: 'mapbox-dem',
|
|
exaggeration: mapState.exaggeration ? 1.5 : 0,
|
|
});
|
|
}
|
|
}, [map, mapState.exaggeration]);
|
|
|
|
/**
|
|
* Map Style
|
|
*/
|
|
useEffect(() => {
|
|
if (map?.loaded()) {
|
|
map.setStyle(MapStyles[mapState.style].data);
|
|
}
|
|
}, [map, mapState.style]);
|
|
|
|
return (
|
|
<MapboxContext.Provider value={{ map, ref }}>
|
|
{children}
|
|
</MapboxContext.Provider>
|
|
);
|
|
};
|
|
|