libre-charge-map/js/lcm_main.js

1319 lines
48 KiB
JavaScript
Raw Normal View History

2024-12-11 17:32:48 +01:00
/**
* rechercher les bornes de recharge,
* afficher des cercles colorés selon la puissance max de la station
* lister les bornes trouvées dans la page
* @type {boolean}
*/
import lcm_config from './lcm_config.js'
import lcm_utils from './lcm_utils.js'
import lcm_color_utils from './lcm_color_utils.js'
2025-04-18 16:50:26 +02:00
import { sendToJOSM, createJOSMEditLink } from './lcm_editor.js'
2025-02-03 13:41:48 +01:00
let geojsondata;
let lastLatLng;
2025-04-18 17:53:07 +02:00
let searchLocationMarker = null;
2024-12-22 23:39:17 +01:00
2024-10-17 15:01:47 +02:00
// serveurs de tuiles: https://wiki.openstreetmap.org/wiki/Tile_servers
// https://stamen-tiles.a.ssl.fastly.net/toner/{z}/{x}/{y}.png
// https://a.tile.openstreetmap.fr/osmfr/{z}/{x}/{y}.png
2024-10-18 00:47:01 +02:00
// https://tile.openstreetmap.org/{z}/{x}/{y}.png
// 'https://cartodb-basemaps-{s}.global.ssl.fastly.net/light_all/{z}/{x}/{y}.png'
2024-12-22 23:39:17 +01:00
2024-10-17 15:01:47 +02:00
// Créer la carte centrée sur Rouen
2024-12-14 17:39:17 +01:00
// Liste des 20 villes les plus peuplées de France avec leurs coordonnées géographiques
2024-12-17 11:53:02 +01:00
2024-12-14 17:39:17 +01:00
// Initialisation de la carte avec la vue centrée sur la ville choisie
2024-12-18 13:40:58 +01:00
let map = L.map('map')
L.control.scale().addTo(map)
2024-12-16 12:54:15 +01:00
/**
* filtres à toggle par des boutons dans la page
* à appliquer à chaque rafraîchissement des points geojson
* TODO: make buttons and filter in refresh circles
*/
2024-12-22 23:39:17 +01:00
let filterStatesAvailable = ['hide', 'show', 'showOnly']
let filters_features = {
2025-02-03 13:41:48 +01:00
display_unknown_max_power_station: filterStatesAvailable[1]
}
let display_type2_sockets = 'show';
let display_type2_combo_sockets = 'show';
let display_unknown_max_power_station = 'show';
2025-02-03 10:49:43 +01:00
let display_alert_cable_missing = 'show';
let display_known_max_power_station = 'show';
let display_type2_combo_sockets_with_cable = 'show';
let display_lower_than_50kw = 'show';
let display_higer_than_50kw = 'show';
let display_lower_than_200kw = 'show';
let display_higer_than_200kw = 'show';
let display_chelou = 'show'; // les stations avec une valeur suspecte, plus de 400kW
2025-04-18 17:53:07 +02:00
// Ajouter cette fonction avant searchLocation
function moveToLocation(place) {
const lat = parseFloat(place.lat);
const lon = parseFloat(place.lon);
if (isNaN(lat) || isNaN(lon)) {
console.error('Coordonnées invalides:', place);
return;
}
// Supprimer l'ancien marqueur s'il existe
if (searchLocationMarker) {
map.removeLayer(searchLocationMarker);
}
// Créer un nouveau marqueur avec une icône personnalisée
searchLocationMarker = L.marker([lat, lon], {
icon: L.divIcon({
className: 'search-location-marker',
html: '📍',
iconSize: [30, 30],
iconAnchor: [15, 30]
})
});
// Ajouter un popup avec le nom du lieu
const popupContent = `
<strong>${place.display_name}</strong>
${place.type ? `<br>Type: ${place.type}` : ''}
${place.context ? `<br>${place.context}` : ''}
`;
searchLocationMarker.bindPopup(popupContent);
// Ajouter le marqueur à la carte
searchLocationMarker.addTo(map);
// Centrer la carte sur le lieu
map.setView([lat, lon], map.getZoom());
// Ouvrir le popup automatiquement
searchLocationMarker.openPopup();
// Faire disparaître le marqueur après 10 secondes
// setTimeout(() => {
// if (searchLocationMarker) {
// map.removeLayer(searchLocationMarker);
// searchLocationMarker = null;
// }
// }, 10000);
}
// Déplacer searchLocationWithAddok avant searchLocation
function searchLocationWithAddok(searchText, mapCenter) {
const baseUrl = 'https://demo.addok.xyz/search';
const params = new URLSearchParams({
q: searchText,
limit: 10,
lat: mapCenter.lat,
lon: mapCenter.lng
});
const url = `${baseUrl}?${params.toString()}`;
return fetch(url)
.then(response => {
if (!response.ok) {
throw new Error('Erreur réseau lors de la recherche');
}
return response.json();
})
.then(data => {
if (!data.features || data.features.length === 0) {
throw new Error('Aucun résultat trouvé');
}
return data.features.map(feature => ({
lat: feature.geometry.coordinates[1],
lon: feature.geometry.coordinates[0],
display_name: feature.properties.label,
importance: feature.properties.score,
context: feature.properties.context,
type: feature.properties.type,
city: feature.properties.city,
distance: feature.properties.distance
}));
});
}
// Modifier la fonction searchLocation
function searchLocation() {
const location = $('#searchLocation').val();
if (!location) {
alert('Veuillez entrer un lieu à rechercher.');
return;
}
const useAddok = $('#useAddok').is(':checked');
const searchPromise = useAddok
? searchLocationWithAddok(location, map.getCenter())
: fetch(`https://nominatim.openstreetmap.org/search?format=json&q=${encodeURIComponent(location)}`)
.then(response => response.json());
searchPromise
.then(data => {
const resultsDropdown = $('#searchResults');
resultsDropdown.empty();
if (!data || data.length === 0) {
alert('Lieu non trouvé. Veuillez essayer un autre terme de recherche.');
resultsDropdown.hide();
return;
}
// Toujours sélectionner le premier résultat
moveToLocation(data[0]);
// Si il y a plus d'un résultat, les afficher quand même dans la liste
if (data.length > 1) {
// Ajouter le bouton de fermeture avant la liste des résultats
const closeButton = $('<button>')
.addClass('close-results-button')
.html('❌')
.attr('title', 'Fermer les résultats de recherche')
.on('click', function () {
$('#searchResults').hide();
$(this).hide();
$('#searchLocation').val('').focus();
});
resultsDropdown.before(closeButton);
data.forEach((place, index) => {
let displayText = place.display_name;
if (useAddok && place.distance) {
displayText += ` (${Math.round(place.distance)}m)`;
}
const option = $('<option></option>')
.val(index)
.text(displayText)
.data('place', place);
resultsDropdown.append(option);
});
resultsDropdown.show();
closeButton.show();
// Sélectionner visuellement le premier résultat dans la liste
resultsDropdown.val(0);
} else {
resultsDropdown.hide();
}
})
.catch(error => {
console.error('Erreur lors de la recherche du lieu :', error);
alert('Erreur lors de la recherche du lieu : ' + error.message);
});
}
function setRandomView() {
let randomCity = lcm_utils.cities[Math.floor(Math.random() * lcm_utils.cities.length)];
map = map.setView(randomCity.coords, lcm_config.initialZoom);
2024-12-16 12:54:15 +01:00
}
2024-12-18 13:40:58 +01:00
function setCoordinatesOfLeafletMapFromQueryParameters() {
2024-12-28 17:59:15 +01:00
const urlParams = new URLSearchParams(window.location.href);
const lat = urlParams.get('lat');
const lng = urlParams.get('lng');
const zoom = urlParams.get('zoom');
if (lat && lng && zoom) {
2024-12-28 17:59:15 +01:00
map = map.setView([lat, lng], zoom);
} else {
2024-12-28 17:59:15 +01:00
console.error('Les paramètres de coordonnées et de zoom doivent être présents dans l\'URL.');
setRandomView();
}
2024-12-16 12:54:15 +01:00
}
function updateURLWithMapCoordinatesAndZoom() {
// Récupère les coordonnées et le niveau de zoom de la carte
const center = map.getCenter()
const zoom = map.getZoom()
2024-12-16 12:54:15 +01:00
// Construit l'URL avec les paramètres de coordonnées et de zoom
const url = `#coords=1&lat=${center.lat}&lng=${center.lng}&zoom=${zoom}`
// Met à jour l'URL de la page
history.replaceState(null, null, url)
2025-02-03 10:49:43 +01:00
updateExternalEditorsLinks()
2024-12-16 12:54:15 +01:00
}
2024-12-28 17:00:21 +01:00
let all_stations_markers = L.layerGroup().addTo(map) // layer group pour tous les marqueurs
// let stations_much_speed_wow = L.layerGroup().addTo(map) // layer group des stations rapides
var osm = L.tileLayer(lcm_config.tileServers.osm, {
attribution: lcm_config.osmMention + '&copy; <a href="https://www.openstreetmap.org/">OpenStreetMap</a> contributors'
2024-12-14 18:00:09 +01:00
})
var cycle = L.tileLayer(lcm_config.tileServers.cycle, {
attribution: lcm_config.osmMention + '&copy; <a href="https://www.opencyclemap.org/">OpenCycleMap</a> contributors'
2024-12-14 18:00:09 +01:00
})
var transport = L.tileLayer(lcm_config.tileServers.transport, {
attribution: lcm_config.osmMention
2024-12-14 18:00:09 +01:00
})
let tileGrey =
L.tileLayer(lcm_config.tileServers.cartodb, {
attribution: lcm_config.osmMention
})
2024-12-14 18:00:09 +01:00
let stamen =
L.tileLayer(lcm_config.tileServers.stamen, {
attribution: lcm_config.osmMention
})
2025-04-18 17:53:07 +02:00
// Ajouter après les autres déclarations de tileLayer
let bdortho = L.tileLayer('https://wxs.ign.fr/ortho/geoportail/wmts?' +
'SERVICE=WMTS&REQUEST=GetTile&VERSION=1.0.0&LAYER=ORTHOIMAGERY.ORTHOPHOTOS' +
'&STYLE=normal&FORMAT=image/jpeg&TILEMATRIXSET=PM&' +
'TILEMATRIX={z}&TILEROW={y}&TILECOL={x}', {
attribution: '&copy; <a href="https://geoservices.ign.fr/bdortho">IGN-F/Géoportail</a>',
maxZoom: 19
});
// Modifier la définition de baseLayers pour inclure la BD ORTHO
2024-12-14 18:00:09 +01:00
var baseLayers = {
'Grey': tileGrey,
'Stamen': stamen,
'OpenStreetMap': osm,
2025-04-18 17:53:07 +02:00
'BD ORTHO IGN': bdortho,
// 'OpenCycleMap': cycle,
'Transport': transport
2024-12-14 18:00:09 +01:00
}
let overlays = {
stations: all_stations_markers
2024-12-28 17:00:21 +01:00
}
2024-10-17 15:01:47 +02:00
2025-04-18 16:50:26 +02:00
const layerControl = L.control.layers(baseLayers, overlays, { collapsed: true }).addTo(map)
2024-12-14 18:08:00 +01:00
tileGrey.addTo(map)
2024-12-11 23:13:15 +01:00
2024-12-28 17:00:21 +01:00
function buildOverpassApiUrl(map, overpassQuery) {
2024-12-28 17:59:15 +01:00
let baseUrl = 'https://overpass-api.de/api/interpreter';
const kilometersMarginForLoading = 2;
const marginInDegrees = kilometersMarginForLoading / 111;
const south = map.getBounds().getSouth() - marginInDegrees;
const west = map.getBounds().getWest() - marginInDegrees;
const north = map.getBounds().getNorth() + marginInDegrees;
const east = map.getBounds().getEast() + marginInDegrees;
let bounds = south + ',' + west + ',' + north + ',' + east;
let resultUrl, query = '';
2024-10-17 15:01:47 +02:00
if (lcm_config.overrideQuery) {
query = `?data=[out:json][timeout:15];(
nwr[amenity=charging_station](${bounds});
2024-12-28 17:59:15 +01:00
);out body geom;`;
} else {
2024-12-28 17:59:15 +01:00
let nodeQuery = 'node[' + overpassQuery + '](' + bounds + ');';
let wayQuery = 'way[' + overpassQuery + '](' + bounds + ');';
let relationQuery = 'relation[' + overpassQuery + '](' + bounds + ');';
query = '?data=[out:json][timeout:15];(' + nodeQuery + wayQuery + relationQuery + ');out body geom;';
}
2024-12-28 17:59:15 +01:00
resultUrl = baseUrl + query;
return resultUrl;
2025-02-03 13:41:48 +01:00
}
2024-12-28 17:59:15 +01:00
2024-10-17 15:01:47 +02:00
function supprimerMarqueurs() {
2024-12-28 17:59:15 +01:00
all_stations_markers.clearLayers();
map.eachLayer((layer) => {
if (layer instanceof L.Marker) {
2024-12-28 17:59:15 +01:00
layer.remove();
}
2024-12-28 17:59:15 +01:00
});
}
2024-12-11 23:13:15 +01:00
let coef_reduction_bars = 0.8
2024-10-17 15:01:47 +02:00
function calculerPourcentage(partie, total, reduc) {
if (total === 0) {
return 'Division par zéro impossible'
}
let coef_reduction = 1
if (reduc) {
coef_reduction = coef_reduction_bars
}
return ((partie / total) * 100 * coef_reduction).toFixed(1)
2024-12-11 23:13:15 +01:00
}
function displayStatsFromGeoJson(resultAsGeojson) {
2024-12-28 17:59:15 +01:00
let count = resultAsGeojson.features.length;
let count_station_output = 0;
let count_ref_eu = 0;
let output_more_than_300 = 0;
let output_more_than_200 = 0;
let output_more_than_100 = 0;
let output_more_than_50 = 0;
let count_station_outputoutput_between_1_and_50 = 0;
let count_output_unknown = 0;
let count_estimated_type2combo = 0;
let count_found_type2combo = 0;
let count_found_type2 = 0;
2025-02-03 13:41:48 +01:00
$('#count_features_fond').html('⚡' + count + ' stations');
resultAsGeojson.features.map(feature => {
2024-12-28 17:59:15 +01:00
let found_type2_combo = false;
let found_type2 = false;
let keys_of_object = Object.keys(feature.properties.tags);
keys_of_object.map(tagKey => {
if (tagKey.indexOf('type2_combo') !== -1) {
2024-12-28 17:59:15 +01:00
found_type2_combo = true;
}
if (tagKey.indexOf('type2') !== -1) {
2024-12-28 17:59:15 +01:00
found_type2 = true;
}
2024-12-28 17:59:15 +01:00
});
let outputPower = lcm_utils.guessOutputPowerFromFeature(feature);
if (found_type2_combo) {
2024-12-28 17:59:15 +01:00
count_found_type2combo++;
}
if (found_type2) {
2024-12-28 17:59:15 +01:00
count_found_type2++;
}
if (outputPower == 0) {
2024-12-28 17:59:15 +01:00
count_output_unknown++;
}
// filtres
// filtrer les valeurs inconnues
if (outputPower >= 200 && !found_type2_combo) {
2024-12-28 17:59:15 +01:00
count_estimated_type2combo++;
}
if (outputPower > 0 && outputPower < 50) {
2024-12-28 17:59:15 +01:00
count_station_outputoutput_between_1_and_50++;
}
if (outputPower >= 50 && outputPower < 100) {
2024-12-28 17:59:15 +01:00
output_more_than_50++;
} else if (outputPower >= 100 && outputPower < 200) {
2024-12-28 17:59:15 +01:00
output_more_than_100++;
} else if (outputPower >= 200 && outputPower < 300) {
2024-12-28 17:59:15 +01:00
output_more_than_200++;
} else if (outputPower >= 300) {
2024-12-28 17:59:15 +01:00
feature.properties.puissance_haute = true;
output_more_than_300++;
}
if (feature.properties.tags['charging_station:output']) {
2024-12-28 17:59:15 +01:00
count_station_output++;
}
if (feature.properties.tags['ref:EU:EVSE']) {
2024-12-28 17:59:15 +01:00
count_ref_eu++;
}
2024-12-28 17:59:15 +01:00
});
let bar_powers = `<div class="bars-container">
2024-12-11 23:13:15 +01:00
<div class="bar color-unknown" style="width: ${calculerPourcentage(count_output_unknown, count, true)}%">${count_output_unknown}</div>
2024-12-22 19:07:09 +01:00
<div class="bar color-power-lesser-than-50" style="width: ${calculerPourcentage(count_station_outputoutput_between_1_and_50, count, true)}%">${count_station_outputoutput_between_1_and_50 ? count_station_outputoutput_between_1_and_50 : ''}</div>
<div class="bar color-power-lesser-than-100" style="width: ${calculerPourcentage(output_more_than_50, count, true)}%">${output_more_than_50 ? output_more_than_50 : ''}</div>
<div class="bar color-power-lesser-than-200" style="width: ${calculerPourcentage(output_more_than_100, count, true)}%">${output_more_than_100 ? output_more_than_100 : ''}</div>
2024-12-22 23:39:17 +01:00
<div class="bar color-power-lesser-than-300" style="width: ${calculerPourcentage(output_more_than_200, count, true)}%">${output_more_than_200 ? output_more_than_200 : '' | ''}</div>
2024-12-22 19:07:09 +01:00
<div class="bar color-power-lesser-than-max" style="width: ${calculerPourcentage(output_more_than_300, count, true)}%">${output_more_than_300 ? output_more_than_300 : ''}</div>
2024-12-28 17:59:15 +01:00
</div>`;
2024-12-11 23:13:15 +01:00
2024-12-28 17:00:21 +01:00
let stats_content = `<div class="stats-table">
<table>
<tr>
<th>Type</th>
<th>Nombre</th>
<th>Pourcentage</th>
</tr>
<tr>
<td>Puissance inconnue</td>
<td>${count_output_unknown}</td>
<td>${calculerPourcentage(count_output_unknown, count)}%</td>
</tr>
<tr>
<td>1-50 kW</td>
<td>${count_station_outputoutput_between_1_and_50}</td>
<td>${calculerPourcentage(count_station_outputoutput_between_1_and_50, count)}%</td>
</tr>
<tr>
<td>50-100 kW</td>
<td>${output_more_than_50}</td>
<td>${calculerPourcentage(output_more_than_50, count)}%</td>
</tr>
<tr>
<td>100-200 kW</td>
<td>${output_more_than_100}</td>
<td>${calculerPourcentage(output_more_than_100, count)}%</td>
</tr>
<tr>
<td>200-300 kW</td>
<td>${output_more_than_200}</td>
<td>${calculerPourcentage(output_more_than_200, count)}%</td>
</tr>
<tr>
<td>300+ kW</td>
<td>${output_more_than_300}</td>
<td>${calculerPourcentage(output_more_than_300, count)}%</td>
</tr>
</table>
2024-12-28 17:59:15 +01:00
</div>`;
2024-12-28 17:00:21 +01:00
2024-12-28 17:59:15 +01:00
$('#found_charging_stations').html(stats_content);
$('#bars_power').html(bar_powers);
2024-12-11 23:13:15 +01:00
}
2024-12-11 17:32:48 +01:00
2025-04-18 16:50:26 +02:00
// Modifier bindEventsOnJosmRemote pour cibler les boutons dans un contexte (la popup)
function bindEventsOnJosmRemote(popupElement) {
// Cible tous les liens JOSM à l'intérieur de l'élément popup fourni
$(popupElement).find('.edit-button.josm').each(function () {
// Utiliser .off().on() pour éviter les liaisons multiples si la popup est rouverte
$(this).off('click').on('click', (event) => {
event.preventDefault(); // Empêcher le comportement par défaut du lien
let josm_link = $(this).attr('data-href');
console.log('Sending command to JOSM:', josm_link);
// Utiliser fetch pour la requête GET vers JOSM (plus moderne que $.get)
fetch(josm_link)
.then(response => {
if (!response.ok) {
// Gérer les erreurs de communication avec JOSM
console.error('JOSM remote control error:', response.status, response.statusText);
alert('Erreur de communication avec JOSM. Assurez-vous que JOSM est lancé et que le contrôle à distance est activé.');
} else {
console.log('JOSM command sent successfully.');
// Optionnel: Afficher une notification de succès
}
})
.catch(error => {
console.error('Failed to send command to JOSM:', error);
alert('Impossible d\'envoyer la commande à JOSM. Est-il lancé et le contrôle à distance activé ?');
});
});
});
2024-12-12 00:27:27 +01:00
}
2024-12-14 18:00:09 +01:00
function displayPointsFromApi(points, convert_to_osm_json) {
if (points && convert_to_osm_json) {
2024-12-28 17:59:15 +01:00
geojsondata = osmtogeojson(points);
2024-12-23 00:02:46 +01:00
}
2024-12-28 17:59:15 +01:00
displayStatsFromGeoJson(geojsondata);
let resultLayer = L.geoJson(geojsondata, {
style: function (feature) {
2025-04-18 16:50:26 +02:00
return { color: '#f00' };
},
filter: function (feature, layer) {
2024-12-28 17:59:15 +01:00
let isPolygon = (feature.geometry) && (feature.geometry.type !== undefined) && (feature.geometry.type === 'Polygon');
if (isPolygon) {
2024-12-28 17:59:15 +01:00
feature.geometry.type = 'Point';
let polygonCenter = L.latLngBounds(feature.geometry.coordinates[0]).getCenter();
feature.geometry.coordinates = [polygonCenter.lat, polygonCenter.lng];
}
2024-12-28 17:59:15 +01:00
return true;
},
onmoveend: function (event) {
2024-12-28 17:59:15 +01:00
// console.log('déplacement terminé');
},
onzoomend: function (event) {
2024-12-28 17:59:15 +01:00
supprimerMarqueurs();
displayPointsFromApi();
},
2024-12-22 23:39:17 +01:00
onEachFeature: eachFeature,
2024-12-28 17:59:15 +01:00
});
2024-12-22 23:39:17 +01:00
}
function makePopupOfFeature(feature) {
let popupContent = ''
popupContent += '<div class="sockets-list" >'
let type2 = feature.properties.tags['socket:type2']
let type2_combo = feature.properties.tags['socket:type2_combo']
if (type2) {
popupContent += ' <img class="icon-img" src="img/Type2_socket.svg" alt="prise de type 2">'
if (type2 !== 'yes') {
popupContent += '<span class="socket-counter">x ' + type2 + '</span>'
}
}
if (feature.properties.tags['socket:type2_combo']) {
popupContent += ' <img class="icon-img" src="img/type2_combo.svg" alt="prise de type 2 combo CCS">'
if (type2_combo !== 'yes') {
popupContent += '<span class="socket-counter">x ' + type2_combo + '</span>'
}
}
popupContent += '</div>'
popupContent += '<div class="key-values" >'
// ne montrer que certains champs dans la popup
lcm_config.tags_to_display_in_popup.forEach(function (key) {
if (lcm_config.tags_to_display_in_popup.indexOf(key)) {
2024-12-22 23:39:17 +01:00
let value = feature.properties.tags[key]
if (value) {
if (value.indexOf('http') !== -1) {
value = '<a href="' + value + '">' + value + '</a>'
}
2024-12-22 23:39:17 +01:00
popupContent = popupContent + '<br/><strong class="popup-key">' + key + ' :</strong><span class="popup-value">' + value + '</span>'
}
2024-12-22 23:39:17 +01:00
}
})
popupContent += '</div>'
return popupContent;
}
2025-02-03 10:49:43 +01:00
/**
* application des filtres dans la sélection des bornes à afficher
* @param feature
* @param layer
*/
2024-12-22 23:39:17 +01:00
function eachFeature(feature, layer) {
2024-12-28 17:59:15 +01:00
let link_josm = createJOSMEditLink(feature);
2024-12-22 23:39:17 +01:00
2024-12-28 17:59:15 +01:00
let popupContent = makePopupOfFeature(feature);
layer.bindPopup(popupContent);
2024-12-22 23:39:17 +01:00
let outPowerGuessed = lcm_utils.guessOutputPowerFromFeature(feature);
let color = lcm_color_utils.getColor(feature);
2024-12-28 17:59:15 +01:00
let displayOutPowerGuessed = '? kW';
2024-12-22 23:39:17 +01:00
if (outPowerGuessed) {
2024-12-28 17:59:15 +01:00
displayOutPowerGuessed = outPowerGuessed + ' kW max';
2025-02-03 13:41:48 +01:00
if (display_unknown_max_power_station === 'show_only') {
return;
}
2025-02-03 13:41:48 +01:00
} else {
// on cache cette station si on ne veut pas voir les stations à la puissance inconnue
2025-02-03 13:41:48 +01:00
if (display_unknown_max_power_station === 'hide') {
return;
}
2024-12-22 23:39:17 +01:00
}
2025-02-03 10:49:43 +01:00
/**
* bornes sans informations, suggérer d'ajouter des tags dans OSM
*/
2024-12-22 23:39:17 +01:00
if (!popupContent) {
popupContent = `<span class="no-data"> Aucune information renseignée,
2024-12-28 17:59:15 +01:00
<a class="edit-button" href="https://www.openstreetmap.org/edit?editor=remote&node=${feature.properties.id}">ajoutez la dans OpenStreetMap!</a></span>`;
2024-12-22 23:39:17 +01:00
}
2024-12-14 18:00:09 +01:00
// Ajout du lien vers Panoramax
const panoramaxLink = `https://api.panoramax.xyz/#focus=map&map=16.7/${feature.geometry.coordinates[1]}/${feature.geometry.coordinates[0]}&speed=250`;
let html = ` <a href="https://www.openstreetmap.org/directions?from=&to=${feature.geometry.coordinates[1]},${feature.geometry.coordinates[0]}&engine=fossgis_osrm_car#map=14/${feature.geometry.coordinates[1]}/${feature.geometry.coordinates[0]}" class="navigation-link by-car" title="itinéraire en voiture vers cette station"> 🚗</a>
<a href="https://www.openstreetmap.org/directions?from=&to=${feature.geometry.coordinates[1]},${feature.geometry.coordinates[0]}&engine=fossgis_osrm_bike#map=14/${feature.geometry.coordinates[1]}/${feature.geometry.coordinates[0]}" class="navigation-link by-car" title="itinéraire en vélo vers cette station">🚴</a>
<a href="https://www.openstreetmap.org/directions?from=&to=${feature.geometry.coordinates[1]},${feature.geometry.coordinates[0]}&engine=fossgis_osrm_foot#map=14/${feature.geometry.coordinates[1]}/${feature.geometry.coordinates[0]}" class="navigation-link by-car" title="itinéraire à pied vers cette station">👠</a>
<a class="edit-button" href="https://www.openstreetmap.org/edit?editor=id&node=${feature.properties.id}"></a>
<a class="edit-button josm" data-href="${link_josm}" href="#">JOSM</a>
<a href="${panoramaxLink}" target="_blank" class="panoramax-link" title="Voir sur Panoramax">
<img src="styles/images/panoramax.ico" alt="icone"></a>
<span class="color-indication" style="background-color: ${color};">${displayOutPowerGuessed}</span>
2025-02-03 13:41:48 +01:00
<button class="edit-button" id="fullDetails" >détails</button>
<span class="popup-content">${popupContent}</span>`;
2024-11-26 21:31:43 +01:00
2024-12-28 17:59:15 +01:00
let zoom = map.getZoom();
let radius = 20;
let opacity = 0.5;
let ratio_circle = 10;
2024-12-22 23:39:17 +01:00
if (zoom < 13) {
2024-12-28 17:59:15 +01:00
ratio_circle = 5;
2024-12-22 23:39:17 +01:00
} else if (zoom < 15) {
2024-12-28 17:59:15 +01:00
ratio_circle = 1;
opacity = 0.25;
2024-12-22 23:39:17 +01:00
} else if (zoom <= 16) {
2024-12-28 17:59:15 +01:00
ratio_circle = 0.5;
2024-12-22 23:39:17 +01:00
} else if (zoom <= 18) {
2024-12-28 17:59:15 +01:00
ratio_circle = 0.25;
2024-12-22 23:39:17 +01:00
}
2024-12-28 17:59:15 +01:00
if (!layer._latlng) {
if (lastLatLng) {
2024-12-28 17:59:15 +01:00
layer._latlng = lastLatLng;
}
} else {
2024-12-28 17:59:15 +01:00
lastLatLng = layer._latlng;
}
2024-12-22 23:39:17 +01:00
if (!outPowerGuessed) {
2024-12-28 17:59:15 +01:00
radius = radius * ratio_circle;
2024-12-22 23:39:17 +01:00
} else {
/**
* limiter la taille du cercle pour les valeurs aberrantes
* les mettre en valeur en les plafonnant à 1 de plus que le maximum attendu en lcm_config
*/
2025-02-03 13:41:48 +01:00
if (outPowerGuessed > lcm_config.max_possible_station_output) {
console.error("valeur suspecte outPowerGuessed", outPowerGuessed, feature)
outPowerGuessed = lcm_config.max_possible_station_output + 1
2024-12-31 22:44:33 +01:00
}
2024-12-28 17:59:15 +01:00
radius = outPowerGuessed * ratio_circle;
2024-12-22 23:39:17 +01:00
}
2025-02-03 10:49:43 +01:00
/**
* gestion des marqueurs d'alertes
*/
// info de câble manquant
2025-02-03 13:41:48 +01:00
if (display_alert_cable_missing) {
2025-02-03 10:49:43 +01:00
let keys = Object.keys(feature.properties)
2025-02-03 11:18:34 +01:00
/**
* on considère l'information de câble manquante uniquement dans le cas une info de socket de type 2 est présente mais pas le tag socket:type2_cable.
*/
2025-02-03 13:41:48 +01:00
if (keys.indexOf('socket:type2') !== -1 && keys.indexOf('socket:type2_cable') === -1) {
2025-02-03 10:49:43 +01:00
let circle_alert = L.circle(layer._latlng, {
color: 'red',
fillColor: 'orange',
fillOpacity: 1,
colorOpacity: 0.5,
radius: 20
})
circle_alert.bindPopup("information de câble manquante");
circle_alert.on({
mouseover: function () {
this.openPopup();
2025-04-18 16:50:26 +02:00
bindEventsOnJosmRemote(this.getPopup().getElement());
2025-02-03 11:18:34 +01:00
bindFullDetails(feature);
2025-02-03 10:49:43 +01:00
},
mouseout: function () {
// setTimeout(() => this.closePopup(), 15000);
},
click: function () {
this.openPopup();
2025-04-18 16:50:26 +02:00
bindEventsOnJosmRemote(this.getPopup().getElement());
2025-02-03 11:18:34 +01:00
bindFullDetails(feature);
2025-02-03 10:49:43 +01:00
},
});
circle_alert.addTo(all_stations_markers);
}
}
/**
* affichage des marqueurs de stations de recharge
*/
2024-12-22 23:39:17 +01:00
let circle = L.circle(layer._latlng, {
color: color,
fillColor: color,
fillOpacity: opacity,
colorOpacity: opacity,
radius: radius
2024-12-28 17:59:15 +01:00
}).addTo(all_stations_markers);
2024-12-22 23:39:17 +01:00
if (zoom > 15) {
let circle_center = L.circle(layer._latlng, {
color: 'black',
fillColor: color,
fillOpacity: 1,
radius: 0.1
}).addTo(all_stations_markers);
2024-12-22 23:39:17 +01:00
}
2024-12-28 17:59:15 +01:00
circle.bindPopup(html);
2024-12-22 23:39:17 +01:00
circle.on({
mouseover: function () {
2024-12-28 17:59:15 +01:00
this.openPopup();
2025-04-18 16:50:26 +02:00
bindEventsOnJosmRemote(this.getPopup().getElement());
2025-02-03 13:41:48 +01:00
bindFullDetails(feature);
2024-12-22 23:39:17 +01:00
},
mouseout: function () {
2024-12-28 17:59:15 +01:00
// setTimeout(() => this.closePopup(), 15000);
2024-12-22 23:39:17 +01:00
},
click: function () {
2024-12-28 17:59:15 +01:00
this.openPopup();
2025-04-18 16:50:26 +02:00
bindEventsOnJosmRemote(this.getPopup().getElement());
2025-02-03 13:41:48 +01:00
bindFullDetails(feature);
},
2024-12-28 17:59:15 +01:00
});
2025-02-03 10:49:43 +01:00
2024-10-17 15:01:47 +02:00
}
2025-02-03 13:41:48 +01:00
function bindFullDetails(feature) {
$('#fullDetails').on('click', () => {
console.log('details', feature)
console.log("$('#current_station_infos')", $('#current_station_infos')[0])
2025-02-03 11:18:34 +01:00
2025-02-03 13:41:48 +01:00
let content = '<table>' +
'<tbody>' + ''
2025-04-18 16:50:26 +02:00
let ii = 0
2025-02-03 13:41:48 +01:00
let keys = Object.keys(feature.properties.tags)
2025-04-18 16:50:26 +02:00
keys.forEach((elem, val) => {
console.log('elem, val', elem, val)
content += '<tr><td>' + elem + '</td><td>' + feature.properties.tags[elem] + '</td></tr>';
ii++;
2025-02-03 13:41:48 +01:00
})
content += '</tbody>' +
'</table>'
console.log('content', content)
$('#current_station_infos')[0].innerHTML = `<h2>Détails</h2>
${content}`
})
2025-02-03 11:18:34 +01:00
}
2025-02-03 13:41:48 +01:00
function makeCssClassFromTags(tags) {
let tagKeys = Object.keys(tags)
if (!tags) {
return ''
}
let listOfClasses = []
2024-10-17 15:01:47 +02:00
tagKeys.forEach((element) => {
listOfClasses.push('tag-' + element + '_' + tags[element].replace(':', '--').replace(' ', '-'))
})
2024-10-17 15:01:47 +02:00
return listOfClasses.join(' ')
2024-10-17 15:01:47 +02:00
}
function getIconFromTags(tags) {
let iconFileName = ''
// let iconFileName = 'icon_restaurant.png';
if (tags['man_made']) {
iconFileName = 'fountain.png'
}
return iconFileName
2024-10-17 15:01:47 +02:00
}
function toggleMinPower(showHighPower) {
2024-12-28 17:59:15 +01:00
showHighPower = !showHighPower;
addFilteredMarkers(showHighPower);
this.textContent = showHighPower ? 'Montrer puissance haute' : 'Montrer puissance normale';
2024-12-11 23:13:15 +01:00
}
function addFilteredMarkers(showHighPower) {
2024-12-28 17:59:15 +01:00
allMarkers.clearLayers();
2024-12-28 17:59:15 +01:00
let counter = 0;
geojsondata.features.forEach(function (feature) {
if (feature.properties.puissance_haute === showHighPower) {
2024-12-28 17:59:15 +01:00
counter++;
let marker = L.marker(feature.geometry.coordinates).bindPopup(feature.properties.puissance_haute ? 'Puissance haute' : 'Puissance normale');
allMarkers.addLayer(marker);
}
2024-12-28 17:59:15 +01:00
});
2024-12-11 17:32:48 +01:00
}
2024-10-17 15:01:47 +02:00
let isLoading = false
function loadOverpassQuery() {
if (!isLoading) {
2024-12-28 17:59:15 +01:00
isLoading = true;
$('#spinning_icon').fadeIn();
let queryTextfieldValue = $('#query-textfield').val();
let overpassApiUrl = buildOverpassApiUrl(map, queryTextfieldValue);
$.get(overpassApiUrl, function (geoDataPointsFromApi) {
2024-12-28 17:59:15 +01:00
geojsondata = geoDataPointsFromApi;
refreshDisplay(true);
2024-12-28 17:59:15 +01:00
$('#spinning_icon').fadeOut();
$('#message-loading').fadeOut();
isLoading = false;
});
}
2024-10-17 15:01:47 +02:00
}
2025-02-03 13:41:48 +01:00
function refreshDisplay(convert_points_to_osm = false) {
2024-12-28 17:59:15 +01:00
supprimerMarqueurs();
2025-02-03 13:41:48 +01:00
displayPointsFromApi(geojsondata, convert_points_to_osm);
2024-12-22 23:39:17 +01:00
}
function onMapMoveEnd() {
let center = map.getCenter()
let zoom = map.getZoom()
let infos = `Lat: ${center.lat}, Lon: ${center.lng}, Zoom : ${zoom}`
2024-12-28 23:14:38 +01:00
2024-12-28 13:29:29 +01:00
if (zoom < 10) {
$('#zoomMessage').show()
} else {
2024-12-28 13:29:29 +01:00
$('#zoomMessage').hide()
loadOverpassQuery()
}
2024-12-28 23:14:38 +01:00
if (map.getZoom() > 14) {
searchFoodPlaces(map);
} else {
food_places_markers.clearLayers();
}
$('#infos_carte').html(infos)
2024-12-28 17:00:21 +01:00
// Stocker les dernières coordonnées connues
if (!window.lastKnownPosition) {
window.lastKnownPosition = center;
updateURLWithMapCoordinatesAndZoom();
2025-02-03 10:49:43 +01:00
2024-12-28 17:00:21 +01:00
} else {
// Calculer la distance en km entre l'ancienne et la nouvelle position
const distanceKm = map.distance(center, window.lastKnownPosition) / 1000;
2025-02-03 13:41:48 +01:00
2025-04-18 17:53:07 +02:00
console.log('déplacement de ', distanceKm, 'km')
2024-12-28 17:00:21 +01:00
// Ne mettre à jour que si on s'est déplacé de plus de 2km
if (distanceKm > 2) {
window.lastKnownPosition = center;
updateURLWithMapCoordinatesAndZoom();
}
}
2024-12-16 12:54:15 +01:00
2025-04-18 16:50:26 +02:00
// Ajout d'un log pour déboguer
console.log("Zoom actuel:", map.getZoom());
2025-04-18 17:53:07 +02:00
if (map.getZoom() >= 12) {
2025-04-18 16:50:26 +02:00
console.log("Recherche des issues Osmose...");
searchOsmoseIssues(map);
} else {
console.log("Zoom trop faible pour les issues Osmose");
osmose_markers.clearLayers();
}
2024-12-12 00:27:27 +01:00
}
2024-12-16 12:54:15 +01:00
setCoordinatesOfLeafletMapFromQueryParameters()
2024-12-14 18:00:09 +01:00
$(document).ready(function () {
2024-12-28 17:59:15 +01:00
bindEventsOnJosmRemote();
onMapMoveEnd();
map.on('moveend', onMapMoveEnd);
$('#spinning_icon').hide();
2025-02-03 10:49:43 +01:00
/**
* boutons de changement de filtres et de rechargement des bornes à l'affichage
*/
$('#removeMarkers').on('click', function () {
2024-12-28 17:59:15 +01:00
supprimerMarqueurs();
});
$('#load').on('click', function () {
2024-12-28 17:59:15 +01:00
loadOverpassQuery();
});
2024-12-28 17:00:21 +01:00
$('#toggleSidePanel').on('click', function () {
2024-12-28 17:59:15 +01:00
$('body').toggleClass('side-panel-open');
});
2024-12-23 00:09:52 +01:00
$('#chercherButton').on('click', function () {
2024-12-28 17:59:15 +01:00
supprimerMarqueurs();
loadOverpassQuery();
});
2024-12-23 00:30:04 +01:00
$('#setRandomView').on('click', function () {
2024-12-28 17:59:15 +01:00
setRandomView();
loadOverpassQuery();
});
$('#filterUnkown').on('click', function () {
2024-12-28 17:59:15 +01:00
display_unknown_max_power_station = cycleVariableState(display_unknown_max_power_station, '#filterUnkown');
showActiveFilter(display_unknown_max_power_station, '#filterUnkown');
refreshDisplay();
});
2025-02-03 10:49:43 +01:00
/**
* toggle des alertes de tags décrivant la présence de cable
*/
$('#cableMissing').on('click', function () {
2025-02-03 13:41:48 +01:00
display_alert_cable_missing = !display_alert_cable_missing;
2025-02-03 10:49:43 +01:00
showActiveFilter(display_alert_cable_missing, '#cableMissing');
refreshDisplay();
});
2024-12-28 17:59:15 +01:00
showActiveFilter(display_unknown_max_power_station, '#filterUnkown');
2025-04-27 21:24:25 +02:00
$('#shareUrl').on('click', copyCurrentUrl);
// Écouteurs pour les nouveaux filtres
Object.keys(lcm_config.filterConfigs).forEach(filterId => {
$(`#${filterId}`).on('change', function() {
lcm_config[lcm_config.filterConfigs[filterId]] = this.checked;
refreshDisplay();
});
});
2024-12-28 17:59:15 +01:00
});
2024-12-16 12:13:19 +01:00
function showActiveFilter(filterVariableName, selectorId) {
2024-12-22 23:39:17 +01:00
$(selectorId).attr('class', 'filter-state-' + filterVariableName)
}
2025-02-03 10:49:43 +01:00
/**
* mettre à jour les liens vers des éditeurs externes
*/
2025-02-03 13:41:48 +01:00
function updateExternalEditorsLinks() {
2025-02-03 10:49:43 +01:00
const center = map.getCenter()
const zoom = map.getZoom()
2025-02-03 13:41:48 +01:00
mapCompleteLink(center.lat, center.lng, zoom)
idLink(center.lat, center.lng, zoom)
2025-02-03 10:49:43 +01:00
}
2025-02-03 13:41:48 +01:00
function mapCompleteLink(lat, lon, zoom) {
$("mapCompleteLink").attr('href', `https://mapcomplete.org/charging_stations?z=${zoom}&lat=${lat}&lon=${lon}`)
2025-02-03 10:49:43 +01:00
}
2025-02-03 13:41:48 +01:00
function idLink(lat, lon, zoom) {
let href = `https://www.openstreetmap.org/edit?editor=id#map=${zoom}/${lat}/${lon}`
2025-02-03 10:49:43 +01:00
console.log('idlink', href)
$("idLink").attr('href', href)
}
2024-12-28 17:00:21 +01:00
function cycleVariableState(filterVariableName, selectorId) {
2024-12-22 23:39:17 +01:00
console.log('filterVariableName', filterVariableName, filterStatesAvailable)
if (filterVariableName) {
if (filterVariableName == filterStatesAvailable[0]) {
filterVariableName = filterStatesAvailable[1]
} else if (filterVariableName == filterStatesAvailable[1]) {
filterVariableName = filterStatesAvailable[2]
} else if (filterVariableName == filterStatesAvailable[2]) {
filterVariableName = filterStatesAvailable[0]
}
} else {
filterVariableName = filterStatesAvailable[0]
}
showActiveFilter(filterVariableName, selectorId)
2024-12-22 23:39:17 +01:00
console.log('filterVariableName after', filterVariableName)
return filterVariableName
}
2024-12-16 12:13:19 +01:00
2024-12-28 13:29:29 +01:00
// toggle des stats
2025-02-03 13:41:48 +01:00
$('#toggle-stats').on('click', function () {
2024-12-28 13:29:29 +01:00
$('#found_charging_stations').slideToggle();
2025-02-03 13:41:48 +01:00
2024-12-28 13:29:29 +01:00
// Change le symbole de la flèche
let text = $(this).text();
2025-02-03 13:41:48 +01:00
if (text.includes('🔽')) {
2024-12-28 13:29:29 +01:00
$(this).text(text.replace('🔽', '🔼'));
} else {
2025-02-03 13:41:48 +01:00
$(this).text(text.replace('🔼', '🔽'));
2024-12-28 13:29:29 +01:00
}
});
2024-12-28 17:00:21 +01:00
// Ajouter ces variables avec les autres déclarations globales
let food_places_markers = L.layerGroup();
const foodIcon = L.divIcon({
className: 'food-marker',
html: '🍽️',
iconSize: [20, 20],
iconAnchor: [10, 10]
});
// Ajouter cette fonction avec les autres fonctions de recherche
function searchFoodPlaces(map) {
const bounds = map.getBounds();
const bbox = bounds.getSouth() + ',' + bounds.getWest() + ',' + bounds.getNorth() + ',' + bounds.getEast();
2025-02-03 13:41:48 +01:00
2024-12-28 17:00:21 +01:00
const query = `
[out:json][timeout:25];
(
2025-04-18 16:50:26 +02:00
nwr["amenity"="restaurant"](${bbox});
nwr["amenity"="cafe"](${bbox});
2024-12-28 17:00:21 +01:00
);
2025-04-18 16:50:26 +02:00
out center;`;
2024-12-28 17:00:21 +01:00
const url = `https://overpass-api.de/api/interpreter?data=${encodeURIComponent(query)}`;
food_places_markers.clearLayers();
fetch(url)
.then(response => response.json())
.then(data => {
2025-04-18 16:50:26 +02:00
data.elements.forEach(element => {
// Utiliser les coordonnées du centre pour les ways et relations
const lat = element.lat || element.center.lat;
const lon = element.lon || element.center.lon;
const name = element.tags.name || 'Sans nom';
const type = element.tags.amenity;
const marker = L.marker([lat, lon], {
2024-12-28 17:00:21 +01:00
icon: foodIcon
});
2025-02-03 13:41:48 +01:00
2024-12-28 17:00:21 +01:00
marker.bindPopup(`
<strong>${name}</strong><br>
Type: ${type}<br>
2025-04-18 16:50:26 +02:00
${element.tags.cuisine ? 'Cuisine: ' + element.tags.cuisine : ''}
2024-12-28 17:00:21 +01:00
`);
2025-02-03 13:41:48 +01:00
2024-12-28 17:00:21 +01:00
food_places_markers.addLayer(marker);
});
})
.catch(error => console.error('Erreur lors de la recherche des restaurants:', error));
}
2025-04-18 16:50:26 +02:00
// Ajouter après la déclaration des autres variables globales
let osmose_markers = L.layerGroup();
const osmoseIcon = L.divIcon({
className: 'osmose-marker-drop',
html: '<div class="osmose-marker-inner">⚡</div>',
iconSize: [30, 40],
iconAnchor: [15, 40]
});
2025-04-18 17:53:07 +02:00
// Ajouter cette fonction utilitaire pour calculer la distance entre deux points
function calculateDistance(lat1, lon1, lat2, lon2) {
return L.latLng(lat1, lon1).distanceTo([lat2, lon2]);
}
// Ajouter cette fonction pour vérifier si le calque des stations est actif
function isChargingStationLayerActive() {
return map.hasLayer(all_stations_markers);
}
// Modifier la fonction searchOsmoseIssues
2025-04-18 16:50:26 +02:00
function searchOsmoseIssues(map) {
const bounds = map.getBounds();
const zoom = map.getZoom();
const bbox = `${bounds.getWest()}%2C${bounds.getSouth()}%2C${bounds.getEast()}%2C${bounds.getNorth()}`;
const url = `https://osmose.openstreetmap.fr/api/0.3/issues?zoom=${zoom}&item=8410%2C8411&level=1%2C2%2C3&limit=500&bbox=${bbox}`;
console.log("Recherche des issues Osmose (liste) avec l'URL :", url);
osmose_markers.clearLayers();
2025-04-18 17:53:07 +02:00
// Modifier la vérification des stations existantes
const existingStations = [];
if (lcm_config.hide_osmose_markers_if_close_to_existing_charging_stations &&
isChargingStationLayerActive() && // Nouvelle condition
geojsondata &&
geojsondata.features) {
geojsondata.features.forEach(feature => {
if (feature.geometry && feature.geometry.coordinates) {
existingStations.push({
lat: feature.geometry.coordinates[1],
lon: feature.geometry.coordinates[0]
});
}
});
}
2025-04-18 16:50:26 +02:00
fetch(url)
.then(response => {
if (!response.ok) {
return response.text().then(text => { throw new Error(`Erreur HTTP ${response.status}: ${response.statusText}. Réponse : ${text}`); });
}
return response.json();
})
.then(data => {
if (!data || !Array.isArray(data.issues)) {
console.warn("Réponse Osmose (liste) inattendue ou pas d'issues:", data);
return;
}
const issuesList = data.issues;
console.log(`Nombre d'issues Osmose (liste) trouvées: ${issuesList.length}`);
issuesList.forEach(issueInfo => {
2025-04-18 17:53:07 +02:00
if (issueInfo.lat == null || issueInfo.lon == null || !issueInfo.id) {
return;
2025-04-18 16:50:26 +02:00
}
const lat = parseFloat(issueInfo.lat);
const lon = parseFloat(issueInfo.lon);
2025-04-18 17:53:07 +02:00
// Vérifier la distance avec les stations existantes
if (lcm_config.hide_osmose_markers_if_close_to_existing_charging_stations) {
const tooClose = existingStations.some(station => {
const distance = calculateDistance(lat, lon, station.lat, station.lon);
return distance <= lcm_config.hide_osmose_markers_if_close_to_existing_charging_stations_distance;
});
if (tooClose) {
console.log(`Marqueur Osmose ignoré car trop proche d'une station existante: ${lat},${lon}`);
return;
}
2025-04-18 16:50:26 +02:00
}
2025-04-18 17:53:07 +02:00
// Créer le marqueur Osmose si il n'est pas trop proche d'une station existante
2025-04-18 16:50:26 +02:00
const osmoseMarker = L.circle([lat, lon], {
color: "purple",
fillColor: "purple",
fillOpacity: 0.8,
radius: 10,
className: 'leaflet-osmose-layer',
pane: 'markerPane',
2025-04-18 17:53:07 +02:00
issueId: issueInfo.id
2025-04-18 16:50:26 +02:00
});
// Préparer une popup de chargement simple
osmoseMarker.bindPopup("Chargement des détails...");
osmoseMarker.on('click', function (e) {
const clickedMarker = e.target;
const storedIssueId = clickedMarker.options.issueId;
2025-04-18 17:53:07 +02:00
handleMarkerClick(clickedMarker, map); // Nouvelle gestion du clic
2025-04-18 16:50:26 +02:00
const detailUrl = `https://osmose.openstreetmap.fr/api/0.3/issue/${storedIssueId}?langs=auto`;
console.log("Récupération des détails pour l'issue:", detailUrl);
fetch(detailUrl)
.then(response => {
if (!response.ok) {
return response.text().then(text => { throw new Error(`Erreur HTTP ${response.status}: ${response.statusText}. Réponse : ${text}`); });
}
return response.json();
})
.then(issueDetails => {
// Construire le contenu de la popup avec les détails
let popupContent = `<strong>${issueDetails.title?.auto || 'Titre non disponible'}</strong><br>`;
if (issueDetails.subtitle?.auto) {
popupContent += `<p>${issueDetails.subtitle.auto}</p>`;
}
let proposedTags = '';
if (issueDetails.new_elems && issueDetails.new_elems[0] && issueDetails.new_elems[0].add) {
proposedTags = '<table class="proposed-tags">';
issueDetails.new_elems[0].add.forEach(tag => {
proposedTags += `<tr><td>${tag.k}</td><td>${tag.v}</td></tr>`;
});
proposedTags += '</table>';
popupContent += `<div class="proposed-tags-container"><h4>Tags proposés :</h4>${proposedTags}</div>`;
}
// Construire le lien JOSM /import
const josmFixUrl = `http://localhost:8111/import?url=https://osmose.openstreetmap.fr/api/0.3/issue/${storedIssueId}/fix/0`;
popupContent += `
<div class="action-buttons">
<a class="edit-button josm" data-href="${josmFixUrl}" href="#" title="Ouvre JOSM et charge la correction proposée"> Réparer dans JOSM</a>
<a href="https://osmose.openstreetmap.fr/fr/issue/${storedIssueId}" target="_blank" title="Voir les détails de l'alerte sur le site Osmose">Voir sur Osmose</a>
</div>`;
// Mettre à jour le contenu de la popup et s'assurer qu'elle est ouverte
clickedMarker.setPopupContent(popupContent);
clickedMarker.openPopup(); // Rouvrir pour ajuster la taille si nécessaire
// Lier l'événement au bouton JOSM DANS la popup
bindEventsOnJosmRemote(clickedMarker.getPopup().getElement());
})
.catch(error => {
console.error("Erreur lors de la récupération des détails de l'issue Osmose:", error);
clickedMarker.setPopupContent(`Erreur lors du chargement des détails.<br><a href="https://osmose.openstreetmap.fr/fr/issue/${storedIssueId}" target="_blank">Voir sur Osmose</a>`);
clickedMarker.openPopup();
});
});
osmose_markers.addLayer(osmoseMarker);
});
})
.catch(error => {
console.error('Erreur détaillée lors de la recherche de la liste des issues Osmose:', error);
});
}
2025-04-18 17:53:07 +02:00
// Modifier la gestion du clic sur les marqueurs Osmose
function handleMarkerClick(marker, map) {
const popup = marker.getPopup();
const markerLatLng = marker.getLatLng();
// Calculer la position relative du marqueur dans la vue
const markerPoint = map.latLngToContainerPoint(markerLatLng);
const mapHeight = map.getContainer().clientHeight;
// Si le marqueur est dans la moitié supérieure de l'écran
if (markerPoint.y < mapHeight / 2) {
// Calculer le décalage nécessaire pour centrer la popup
const targetLatLng = map.containerPointToLatLng([
markerPoint.x,
mapHeight / 2
]);
// Déplacer la carte avec une animation
map.once('moveend', () => {
marker.openPopup();
});
map.panTo(targetLatLng, {
animate: true,
duration: 0.5
});
} else {
marker.openPopup();
}
}
2024-12-28 17:00:21 +01:00
2025-04-18 17:53:07 +02:00
// Ajouter un écouteur d'événements pour le changement de visibilité des calques
function init() {
2024-12-28 17:00:21 +01:00
food_places_markers.addTo(map);
$('#found_charging_stations').hide();
2025-04-18 16:50:26 +02:00
// Mettre à jour le contrôle des calques
2024-12-28 17:00:21 +01:00
const overlayMaps = {
"Stations de recharge": all_stations_markers,
2025-04-18 16:50:26 +02:00
"Restaurants et cafés": food_places_markers,
"Bornes potentielles (Osmose)": osmose_markers
2024-12-28 17:00:21 +01:00
};
2025-02-03 13:41:48 +01:00
2025-04-18 17:53:07 +02:00
// Créer deux contrôles de couches séparés
const baseLayerControl = L.control.layers(baseLayers, null, {
collapsed: true,
className: 'leaflet-control-layers base-layers',
id: 'base-layers-control'
}).addTo(map);
2024-12-28 17:00:21 +01:00
2025-04-18 17:53:07 +02:00
const overlayControl = L.control.layers(null, overlayMaps, {
collapsed: true,
className: 'leaflet-control-layers overlay-layers',
id: 'overlay-layers-control'
}).addTo(map);
2024-12-28 17:00:21 +01:00
2024-12-28 23:14:38 +01:00
$('#sendToJOSM').on('click', () => {
sendToJOSM(map, geojsondata)
.then(() => {
console.log('Données envoyées à JOSM avec succès !');
})
.catch(() => {
alert('Erreur : JOSM doit être ouvert avec l\'option "Contrôle à distance" activée');
});
2024-12-28 17:00:21 +01:00
});
2025-02-03 10:49:43 +01:00
$('#josmLink').on('click', () => {
sendToJOSM(map, geojsondata)
.then(() => {
console.log('Données envoyées à JOSM avec succès !');
})
.catch(() => {
2025-04-18 16:50:26 +02:00
alert('Erreur : JOSM doit être ouvert avec l\'option de télécommande "Contrôle à distance" activée dans ses paramètres (accédez-y avec F12)');
2025-02-03 10:49:43 +01:00
});
});
2024-12-28 17:59:15 +01:00
$('#searchButton').on('click', searchLocation);
$('#shareUrl').on('click', copyCurrentUrl);
2025-04-18 16:50:26 +02:00
$('#searchResults').on('change', function () {
2025-02-10 12:22:49 +01:00
const selectedIndex = $(this).val();
if (selectedIndex !== null) {
const selectedPlace = $(this).find('option:selected').data('place');
moveToLocation(selectedPlace);
}
});
2025-04-18 16:50:26 +02:00
osmose_markers.addTo(map);
2025-04-18 17:53:07 +02:00
// Ajouter le contrôle de source de recherche
const searchControl = `
<div class="search-source-control">
<label>
<input type="checkbox" id="useAddok" checked>
Utiliser Addok (France)
</label>
</div>
`;
$('#searchLocation').after(searchControl);
// Mettre à jour les marqueurs Osmose quand la visibilité des stations change
map.on('overlayremove overlayadd', function (e) {
if (e.name === "Stations de recharge" && map.getZoom() > 12) {
searchOsmoseIssues(map);
}
});
2024-12-28 17:59:15 +01:00
}
2024-12-28 17:59:15 +01:00
function copyCurrentUrl() {
2025-02-03 13:41:48 +01:00
const url = window.location.href;
2024-12-28 17:59:15 +01:00
var dummy = document.createElement('input'),
2025-02-03 13:41:48 +01:00
text = window.location.href;
2024-12-28 17:59:15 +01:00
document.body.appendChild(dummy);
dummy.value = text;
dummy.select();
document.execCommand('copy');
document.body.removeChild(dummy);
}
2025-04-18 16:50:26 +02:00
init()
// Créer un nouveau pane pour les marqueurs Osmose avec un zIndex plus élevé
map.createPane('osmosePane');
map.getPane('osmosePane').style.zIndex = 1000;