diff --git a/src/components/Map/Map.stories.jsx b/src/components/Map/Map.stories.jsx
index 3d9c8f100..62653ac27 100644
--- a/src/components/Map/Map.stories.jsx
+++ b/src/components/Map/Map.stories.jsx
@@ -16,7 +16,14 @@ export default {
component: LeafletMap,
decorators: [withMapLayout, ConfigDecorator],
args: {
- markerCoordinates: [52.1326332, 5.291266],
+ geoJsonFeature: {
+ type: 'Feature',
+ properties: {},
+ geometry: {
+ type: 'Point',
+ coordinates: [5.291266, 52.1326332],
+ },
+ },
defaultCenter: [52.1326332, 5.291266],
defaultZoomLevel: 12,
disabled: false,
diff --git a/src/components/Map/index.jsx b/src/components/Map/index.jsx
index e2139e7c5..8f59fba72 100644
--- a/src/components/Map/index.jsx
+++ b/src/components/Map/index.jsx
@@ -1,8 +1,12 @@
+import * as Leaflet from 'leaflet';
+import 'leaflet-draw/dist/leaflet.draw.css';
import {GeoSearchControl} from 'leaflet-geosearch';
+import 'leaflet/dist/leaflet.css';
import PropTypes from 'prop-types';
-import React, {useCallback, useContext, useEffect} from 'react';
+import React, {useCallback, useContext, useEffect, useRef} from 'react';
import {defineMessages, useIntl} from 'react-intl';
-import {MapContainer, Marker, TileLayer, useMap, useMapEvent} from 'react-leaflet';
+import {FeatureGroup, MapContainer, Marker, TileLayer, useMap, useMapEvent} from 'react-leaflet';
+import {EditControl} from 'react-leaflet-draw';
import {useGeolocation} from 'react-use';
import {ConfigContext} from 'Context';
@@ -60,19 +64,38 @@ const useDefaultCoordinates = () => {
};
const LeaftletMap = ({
- markerCoordinates,
- onMarkerSet,
+ geoJsonFeature,
+ onGeoJsonFeatureSet,
defaultCenter = DEFAULT_LAT_LNG,
defaultZoomLevel = DEFAULT_ZOOM,
disabled = false,
}) => {
+ const ref = useRef();
const intl = useIntl();
const defaultCoordinates = useDefaultCoordinates();
- const coordinates = markerCoordinates || defaultCoordinates;
+ const coordinates = defaultCoordinates;
const modifiers = disabled ? ['disabled'] : [];
const className = getBEMClassName('leaflet-map', modifiers);
+ const onFeatureCreate = event => {
+ // Remove the old layers and add the new one.
+ // This limits the amount of features to 1
+ const newLayer = event.layer;
+ ref.current?.clearLayers();
+ ref.current?.addLayer(newLayer);
+
+ onGeoJsonFeatureSet(ref.current?.toGeoJSON());
+ };
+
+ useEffect(() => {
+ if (!ref.current) {
+ return;
+ }
+ ref.current?.clearLayers();
+ Leaflet.geoJSON(geoJsonFeature).addTo(ref.current);
+ }, [geoJsonFeature, ref.current]);
+
return (
<>
+
+
+
{coordinates ? (
<>
>
) : null}
console.log('TODO')}
options={{
showMarker: false,
showPopup: false,
@@ -114,16 +155,34 @@ const LeaftletMap = ({
/>
{/*{disabled ? : }*/}
- {markerCoordinates && markerCoordinates.length && (
-
- )}
+ {/*{markerCoordinates && markerCoordinates.length && (*/}
+ {/* */}
+ {/*)}*/}
>
);
};
LeaftletMap.propTypes = {
- markerCoordinates: PropTypes.arrayOf(PropTypes.number),
- onMarkerSet: PropTypes.func,
+ geoJsonFeature: PropTypes.shape({
+ type: PropTypes.oneOf(['Feature']).isRequired,
+ properties: PropTypes.object,
+ geometry: PropTypes.oneOfType([
+ PropTypes.shape({
+ type: PropTypes.oneOf(['Point']).isRequired,
+ coordinates: PropTypes.arrayOf(PropTypes.number).isRequired,
+ }),
+ PropTypes.shape({
+ type: PropTypes.oneOf(['LineString']).isRequired,
+ coordinates: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.number)).isRequired,
+ }),
+ PropTypes.shape({
+ type: PropTypes.oneOf(['Polygon']).isRequired,
+ coordinates: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.number)))
+ .isRequired,
+ }),
+ ]).isRequired,
+ }),
+ onGeoJsonFeatureSet: PropTypes.func,
disabled: PropTypes.bool,
};
diff --git a/src/formio/components/Map.jsx b/src/formio/components/Map.jsx
index f527ac270..c82cc9bcc 100644
--- a/src/formio/components/Map.jsx
+++ b/src/formio/components/Map.jsx
@@ -91,7 +91,7 @@ export default class Map extends Field {
super.destroy();
}
- onMarkerSet(newLatLng) {
+ onGeoJsonSet(newLatLng) {
this.setValue(newLatLng, {modified: true});
}
@@ -100,7 +100,7 @@ export default class Map extends Field {
const {lat = defaultLat, lng = defaultLon} = this.component?.initialCenter || {};
const defaultCenter = [lat, lng];
- const markerCoordinates = this.getValue();
+ const geoJsonFeature = this.getValue();
const container = this.refs.mapContainer;
const zoom = this.component.defaultZoom;
@@ -111,8 +111,8 @@ export default class Map extends Field {