Merge pull request #5164 from betagouv/feat/5136

Usager : amélioration de l'interface d'import de données cartographiques
This commit is contained in:
Pierre de La Morinerie 2020-05-20 14:28:31 +02:00 committed by GitHub
commit 8f283d1383
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -6,7 +6,7 @@ import DrawControl from 'react-mapbox-gl-draw';
import SwitchMapStyle from './SwitchMapStyle'; import SwitchMapStyle from './SwitchMapStyle';
import SearchInput from './SearchInput'; import SearchInput from './SearchInput';
import { getJSON, ajax } from '@utils'; import { getJSON, ajax } from '@utils';
import { gpx } from '@tmcw/togeojson/dist/togeojson.es.js'; import { gpx, kml } from '@tmcw/togeojson/dist/togeojson.es.js';
import ortho from './styles/ortho.json'; import ortho from './styles/ortho.json';
import vector from './styles/vector.json'; import vector from './styles/vector.json';
import { polygonCadastresFill, polygonCadastresLine } from './utils'; import { polygonCadastresFill, polygonCadastresLine } from './utils';
@ -32,6 +32,7 @@ function MapEditor({ featureCollection, url, preview }) {
const [zoom, setZoom] = useState([5]); const [zoom, setZoom] = useState([5]);
const [currentMap, setCurrentMap] = useState({}); const [currentMap, setCurrentMap] = useState({});
const [bbox, setBbox] = useState(featureCollection.bbox); const [bbox, setBbox] = useState(featureCollection.bbox);
const [importInputs, setImportInputs] = useState([]);
const mapStyle = style === 'ortho' ? ortho : vector; const mapStyle = style === 'ortho' ? ortho : vector;
const cadastresFeatureCollection = filterFeatureCollection( const cadastresFeatureCollection = filterFeatureCollection(
featureCollection, featureCollection,
@ -50,6 +51,13 @@ function MapEditor({ featureCollection, url, preview }) {
draw.setFeatureProperty(lid, 'id', feature.properties.id); draw.setFeatureProperty(lid, 'id', feature.properties.id);
} }
const generateId = () => Math.random().toString(20).substr(2, 6);
const updateImportInputs = (inputs, inputId) => {
const updatedInputs = inputs.filter((input) => input.id !== inputId);
setImportInputs(updatedInputs);
};
async function onDrawCreate({ features }) { async function onDrawCreate({ features }) {
for (const feature of features) { for (const feature of features) {
const data = await getJSON(url, { feature }, 'post'); const data = await getJSON(url, { feature }, 'post');
@ -95,29 +103,98 @@ function MapEditor({ featureCollection, url, preview }) {
} }
}; };
const onGpxImport = (e) => { const onFileImport = (e, inputId) => {
const isGpxFile = e.target.files[0].name.includes('.gpx');
let reader = new FileReader(); let reader = new FileReader();
reader.readAsText(e.target.files[0], 'UTF-8'); reader.readAsText(e.target.files[0], 'UTF-8');
reader.onload = async (event) => { reader.onload = async (event) => {
const featureCollection = gpx( let featureCollection;
new DOMParser().parseFromString(event.target.result, 'text/xml') isGpxFile
); ? (featureCollection = gpx(
new DOMParser().parseFromString(event.target.result, 'text/xml')
))
: (featureCollection = kml(
new DOMParser().parseFromString(event.target.result, 'text/xml')
));
const resultFeatureCollection = await getJSON( const resultFeatureCollection = await getJSON(
`${url}/import`, `${url}/import`,
featureCollection, featureCollection,
'post' 'post'
); );
let inputs = [...importInputs];
const setInputs = inputs.map((input) => {
if (input.id === inputId) {
input.disabled = true;
input.hasValue = true;
resultFeatureCollection.features.forEach((feature) => {
if (
JSON.stringify(feature.geometry) ===
JSON.stringify(featureCollection.features[0].geometry)
) {
input.featureId = feature.properties.id;
}
});
}
return input;
});
drawControl.current.draw.set( drawControl.current.draw.set(
filterFeatureCollection( filterFeatureCollection(
resultFeatureCollection, resultFeatureCollection,
'selection_utilisateur' 'selection_utilisateur'
) )
); );
updateFeaturesList(resultFeatureCollection.features); updateFeaturesList(resultFeatureCollection.features);
setImportInputs(setInputs);
setBbox(resultFeatureCollection.bbox); setBbox(resultFeatureCollection.bbox);
}; };
}; };
const addInputFile = (e) => {
e.preventDefault();
let inputs = [...importInputs];
inputs.push({
id: generateId(),
disabled: false,
featureId: null,
hasValue: false
});
setImportInputs(inputs);
};
const removeInputFile = async (e, inputId) => {
e.preventDefault();
const draw = drawControl.current.draw;
const featureCollection = draw.getAll();
let inputs = [...importInputs];
let drawFeatureIdToRemove;
const inputToRemove = inputs.find((input) => input.id === inputId);
for (const feature of featureCollection.features) {
if (inputToRemove.featureId === feature.properties.id) {
drawFeatureIdToRemove = feature.id;
}
}
if (inputToRemove.featureId) {
try {
await getJSON(`${url}/${inputToRemove.featureId}`, null, 'delete');
draw.delete(drawFeatureIdToRemove).getAll();
} catch (e) {
throw new Error(
`La feature ${inputToRemove.featureId} a déjà été supprimée manuellement`,
e
);
} finally {
updateImportInputs(inputs, inputId);
}
}
updateImportInputs(inputs, inputId);
};
useEffect(() => { useEffect(() => {
addEventListener('cadastres:update', onCadastresUpdate); addEventListener('cadastres:update', onCadastresUpdate);
return () => removeEventListener('cadastres:update', onCadastresUpdate); return () => removeEventListener('cadastres:update', onCadastresUpdate);
@ -136,13 +213,33 @@ function MapEditor({ featureCollection, url, preview }) {
return ( return (
<> <>
<div className="file-import" style={{ marginBottom: '20px' }}> <div className="file-import" style={{ marginBottom: '20px' }}>
<button className="button send primary" onClick={addInputFile}>
Ajouter un fichier GPX ou KML
</button>
<div> <div>
<p style={{ fontWeight: 'bolder', marginBottom: '10px' }}> {importInputs.map((input) => (
Importer un fichier GPX <div key={input.id}>
</p> <input
</div> title="Choisir un fichier gpx ou kml"
<div> style={{ marginTop: '15px' }}
<input type="file" accept=".gpx" onChange={onGpxImport} /> id={input.id}
type="file"
accept=".gpx, .kml"
disabled={input.disabled}
onChange={(e) => onFileImport(e, input.id)}
/>
{input.hasValue && (
<span
title="Supprimer le fichier"
className="icon refuse"
style={{
cursor: 'pointer'
}}
onClick={(e) => removeInputFile(e, input.id)}
></span>
)}
</div>
))}
</div> </div>
</div> </div>
<div <div