mappa_home.html 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. <!DOCTYPE html>
  2. <html lang="it">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <title>Mappa Interattiva con Layer Colorati</title>
  7. <!-- Fogli di stile -->
  8. <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-italia@2.13.4/dist/css/bootstrap-italia.min.css">
  9. <link rel="stylesheet" href="https://unpkg.com/leaflet@1.7.1/dist/leaflet.css">
  10. <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.css"/>
  11. <link href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0-beta3/css/all.min.css" rel="stylesheet">
  12. <style>
  13. .top-bar { background-color: #004080; color: white; padding: 0.5rem 0; }
  14. .header { background-color: #0066CC; color: white; padding: 1rem 0; }
  15. /* Stile per il logo "PW" dentro un cerchio */
  16. .logo {
  17. font-family: 'Georgia', serif;
  18. font-size: 1.5rem;
  19. width: 50px; height: 50px;
  20. background-color: white;
  21. color: #0066CC;
  22. border-radius: 50%;
  23. margin-right: 10px;
  24. display: inline-flex;
  25. align-items: center;
  26. justify-content: center;
  27. }
  28. .header-title { color: white; font-size: 1.5rem; font-weight: bold; }
  29. .main-content { background-color: #f8f9fa; padding: 2rem 0; }
  30. #map { height: 500px; width: 100%; border-radius: 8px; }
  31. .search-container { margin: 20px 0; }
  32. .search-box { display: flex; gap: 10px; max-width: 600px; margin: 0 auto; }
  33. .coordinate-box { background: white; padding: 15px; border-radius: 8px; box-shadow: 0 2px 4px rgba(0,0,0,0.1); margin-top: 20px; }
  34. </style>
  35. </head>
  36. <body>
  37. <!-- Header -->
  38. <div class="top-bar">
  39. <div class="container">Project Work</div>
  40. </div>
  41. <!-- Header -->
  42. <header class="header">
  43. <div class="container">
  44. <div class="row align-items-center">
  45. <div class="col d-flex">
  46. <!-- Logo "PW" dentro un cerchio, ora cliccabile -->
  47. <a href="./login.html" class="logo-link">
  48. <div class="logo">PW</div>
  49. </a>
  50. <h1 class="header-title">Mappa</h1> <!-- Titolo "Mappa" con la stessa grandezza -->
  51. </div>
  52. <div class="col-auto">
  53. <a href="./login.html" class="btn btn-light">Login</a>
  54. </div>
  55. </div>
  56. </div>
  57. </header>
  58. <main class="main-content">
  59. <div class="container">
  60. <!-- Barra di ricerca -->
  61. <div class="search-container">
  62. <div class="search-box">
  63. <input type="text"
  64. id="address"
  65. class="form-control"
  66. placeholder="Inserisci l'indirizzo"
  67. style="flex-grow: 1;">
  68. <button id="searchBtn" class="btn btn-primary">
  69. <i class="fas fa-search"></i> Cerca
  70. </button>
  71. </div>
  72. </div>
  73. <!-- Mappa -->
  74. <div id="map"></div>
  75. <!-- Dati geografici -->
  76. <div class="coordinate-box">
  77. <div class="row">
  78. <div class="col-md-4 mb-3">
  79. <label class="form-label">Latitudine:</label>
  80. <input type="number" id="latInput" class="form-control" readonly>
  81. </div>
  82. <div class="col-md-4 mb-3">
  83. <label class="form-label">Longitudine:</label>
  84. <input type="number" id="lngInput" class="form-control" readonly>
  85. </div>
  86. <div class="col-md-4 mb-3">
  87. <label class="form-label">Indirizzo:</label>
  88. <input type="text" id="addressInput" class="form-control" readonly>
  89. </div>
  90. </div>
  91. </div>
  92. </div>
  93. </main>
  94. <!-- Footer -->
  95. <footer id="footer" class="it-footer bg-black" role="contentinfo">
  96. <!-- Contenuto footer mantenuto uguale -->
  97. </footer>
  98. <!-- Script -->
  99. <script src="https://unpkg.com/leaflet@1.7.1/dist/leaflet.js"></script>
  100. <script src="https://cdnjs.cloudflare.com/ajax/libs/leaflet.draw/1.0.4/leaflet.draw.js"></script>
  101. <script>
  102. // Inizializzazione Mappa
  103. const map = L.map('map').setView([41.9028, 12.4964], 13);
  104. L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png').addTo(map);
  105. // Layer edifici colorati
  106. const edificiLayer = L.layerGroup().addTo(map);
  107. let currentMarker = null;
  108. // Funzione principale per caricare gli edifici
  109. async function caricaEdifici() {
  110. if (map.getZoom() >= 16) {
  111. const bounds = map.getBounds();
  112. const bbox = `${bounds.getSouthWest().lat},${bounds.getSouthWest().lng},${bounds.getNorthEast().lat},${bounds.getNorthEast().lng}`;
  113. const query = `[out:json];
  114. (
  115. way["building"](${bbox});
  116. relation["building"](${bbox});
  117. );
  118. out geom;
  119. out tags;`;
  120. try {
  121. const response = await fetch(`https://overpass-api.de/api/interpreter?data=${encodeURIComponent(query)}`);
  122. const data = await response.json();
  123. edificiLayer.clearLayers();
  124. data.elements.forEach(element => {
  125. if (element.type === 'way' && element.geometry) {
  126. const latlngs = element.geometry.map(coord => [coord.lat, coord.lon]);
  127. var statoEdificio = false;
  128. let colorePoligono ;
  129. if (statoEdificio) colorePoligono = 'yellow';
  130. else colorePoligono = 'green';
  131. // const colori = ['#FF0000', '#00FF00'];
  132. const polygon = L.polygon(latlngs, {
  133. fillColor: colorePoligono,
  134. color: colorePoligono,
  135. fillOpacity: 0.5
  136. }).addTo(edificiLayer);
  137. polygon.on('click', async function() {
  138. var buildingType = element.tags && element.tags.building ? element.tags.building : 'Sconosciuto';
  139. try {
  140. await getAddress(latlngs[0][0], latlngs[0][1]);
  141. var popupContent = `
  142. <b>Coordinate:</b> ${latlngs[0][0]}, ${latlngs[0][1]}<br>
  143. <b>Tipo edificio:</b> ${buildingType}<br>
  144. <b>Indirizzo:</b> ${document.getElementById('addressInput').value}<br>
  145. <b>Stato:</b> ${statoEdificio}<br>
  146. <b>Codice Catastale:</b> nessuno
  147. `;
  148. polygon.bindPopup(popupContent).openPopup();
  149. } catch (error) {
  150. console.error("Errore nel recupero dell'indirizzo:", error);
  151. var popupContent = `
  152. <b>Coordinate:</b> ${latlngs[0][0]}, ${latlngs[0][1]}<br>
  153. <b>Tipo edificio:</b> ${buildingType}<br>
  154. <b>Indirizzo:</b> Indirizzo non disponibile<br>
  155. <b>Stato:</b> ${statoEdificio}<br>
  156. <b>Codice Catastale:</b> nessuno
  157. `;
  158. polygon.bindPopup(popupContent).openPopup();
  159. }
  160. });
  161. }
  162. });
  163. } catch (error) {
  164. console.error("Errore nel caricamento edifici:", error);
  165. }
  166. } else {
  167. edificiLayer.clearLayers();
  168. }
  169. }
  170. async function getAddress(lat, lng) {
  171. console.log(typeof(lat));
  172. console.log(typeof(lng));
  173. try {
  174. const response = await fetch(`https://nominatim.openstreetmap.org/reverse?format=jsonv2&lat=${lat}&lon=${lng}`);
  175. const data = await response.json();
  176. if (data.address) {
  177. document.getElementById('addressInput').value = data.display_name;
  178. } else {
  179. document.getElementById('addressInput').value = "Indirizzo non trovato";
  180. }
  181. } catch (error) {
  182. console.error("Errore nel geocoding inverso:", error);
  183. document.getElementById('addressInput').value = "Errore nel geocoding inverso";
  184. }
  185. }
  186. // Funzioni di gestione posizione
  187. async function updatePosition(lat, lng) {
  188. try {
  189. const response = await fetch(`https://nominatim.openstreetmap.org/reverse?format=json&lat=${lat}&lon=${lng}`);
  190. const data = await response.json();
  191. document.getElementById('latInput').value = lat;
  192. document.getElementById('lngInput').value = lng;
  193. document.getElementById('addressInput').value = data.display_name || "Indirizzo non disponibile";
  194. if (currentMarker) map.removeLayer(currentMarker);
  195. currentMarker = L.marker([lat, lng]).addTo(map);
  196. } catch (error) {
  197. console.error("Errore di geocoding inverso:", error);
  198. }
  199. }
  200. // Ricerca indirizzo
  201. async function searchAddress() {
  202. const address = document.getElementById('address').value;
  203. if (!address) return;
  204. try {
  205. const response = await fetch(`https://nominatim.openstreetmap.org/search?format=json&q=${encodeURIComponent(address)}`);
  206. const data = await response.json();
  207. if (data.length === 0) {
  208. alert("Indirizzo non trovato");
  209. return;
  210. }
  211. const { lat, lon: lng } = data[0];
  212. map.setView([lat, lng], 18);
  213. updatePosition(lat, lng);
  214. } catch (error) {
  215. console.error("Errore di ricerca:", error);
  216. }
  217. }
  218. // Event listeners
  219. document.getElementById('searchBtn').addEventListener('click', searchAddress);
  220. document.getElementById('address').addEventListener('keypress', (e) => {
  221. if (e.key === 'Enter') searchAddress();
  222. });
  223. map.on('click', (e) => {
  224. const { lat, lng } = e.latlng;
  225. updatePosition(lat, lng);
  226. });
  227. map.on('moveend', caricaEdifici);
  228. map.on('zoomend', caricaEdifici);
  229. caricaEdifici();
  230. </script>
  231. <script src="https://cdn.jsdelivr.net/npm/bootstrap-italia@2.8.1/dist/js/bootstrap-italia.bundle.min.js"></script>
  232. <!-- Includi lo script di ElevenLabs -->
  233. <script src="https://elevenlabs.io/convai-widget/index.js" async type="text/javascript"></script>
  234. <!-- Widget ElevenLabs ConvAI -->
  235. <div id="ai-widget-container" style="position: fixed; bottom: 20px; right: 20px;">
  236. <elevenlabs-convai agent-id="2MJWbkNuIGTHNI71hKfL"></elevenlabs-convai>
  237. </div>
  238. <footer id="footer" class="it-footer bg-black" role="contentinfo">
  239. <div class="it-footer-main py-3">
  240. <div class="container">
  241. <section class="py-4">
  242. <div class="row">
  243. <div class="col-lg-3 col-md-3 col-sm-6 pb-2">
  244. <div class="link-list-wrapper">
  245. <h2 class="h5">Esplora SINFI</h2>
  246. <ul id="footer-menu-col-1" class="link-list">
  247. <li id="menu-item-sinfi-1" class="menu-item">
  248. <a class="list-item" href="https://sinfi.it/portal/sinfi-menu/che-cose/">Cos'è SINFI</a>
  249. </li>
  250. <li id="menu-item-sinfi-2" class="menu-item">
  251. <a class="list-item" href="https://sinfi.it/realms/master/protocol/openid-connect/auth?client_id=sinfi_user_service&response_type=code&redirect_uri=https://sinfi.it/sinfi_gateway/labs_keycloak/post_login/&scope=email&state=">Infrastrutture Registrate</a>
  252. </li>
  253. <li id="menu-item-sinfi-3" class="menu-item">
  254. <a class="list-item" href="https://sinfi.it/portal/sinfi-menu/riferimenti-normativi/">Normative di Riferimento</a>
  255. </li>
  256. <li id="menu-item-sinfi-4" class="menu-item">
  257. <a class="list-item" href="https://sinfi.it/realms/master/protocol/openid-connect/auth?client_id=sinfi_user_service&response_type=code&redirect_uri=https://sinfi.it/sinfi_gateway/labs_keycloak/post_login/&scope=email&state=">Open Data SINFI</a>
  258. </li>
  259. </ul>
  260. </div>
  261. </div>
  262. <div class="col-lg-3 col-md-3 col-sm-6 pb-2">
  263. <div class="link-list-wrapper">
  264. <h2 class="h5">Aiuto e Supporto</h2>
  265. <ul id="footer-menu-col-2" class="link-list">
  266. <li id="menu-item-aiuto-1" class="menu-item">
  267. <a class="list-item" href="./login.html">Istruzioni</a>
  268. </li>
  269. <li id="menu-item-aiuto-2" class="menu-item">
  270. <a class="list-item" href="./login.html">FAQ - Domande Frequenti</a>
  271. </li>
  272. <li id="menu-item-aiuto-3" class="menu-item">
  273. <a class="list-item"href="./login.html">Segnalazioni e Supporto Tecnico</a>
  274. </li>
  275. </ul>
  276. </div>
  277. </div>
  278. <!-- Colonna "Community" con i loghi dei social -->
  279. <div class="col-lg-3 col-md-3 col-sm-6 pb-2">
  280. <div class="link-list-wrapper">
  281. <h2 class="h5">Community</h2>
  282. <div class="social-icons">
  283. <a href="https://www.linkedin.com" target="_blank" class="me-3"><i class="fab fa-linkedin"></i></a>
  284. <a href="https://www.youtube.com" target="_blank" class="me-3"><i class="fab fa-youtube"></i></a>
  285. <a href="https://www.facebook.com" target="_blank" class="me-3"><i class="fab fa-facebook"></i></a>
  286. <a href="https://www.instagram.com" target="_blank" class="me-3"><i class="fab fa-instagram"></i></a>
  287. </div>
  288. </div>
  289. </div>
  290. <div class="col-lg-3 col-md-3 col-sm-6 pb-2">
  291. <div class="link-list-wrapper">
  292. <h2 class="h5">SINFI</h2>
  293. <ul id="footer-menu-col-4" class="link-list">
  294. <li id="menu-item-377" class="menu-item">
  295. <a target="_blank" href="https://sinfi.it/portal/">Vai al sito ufficiale SINFI</a>
  296. </li>
  297. </ul>
  298. </div>
  299. </div>
  300. </div>
  301. </section>
  302. </div>
  303. </div>
  304. <div class="it-footer-small-prints clearfix">
  305. <div class="container">
  306. <nav class="menu-footer-menu-ita-container" aria-label="link utili">
  307. <ul id="menu-footer-menu-ita" class="it-footer-small-prints-list list-inline mb-0 d-flex flex-column flex-md-row">
  308. <li id="menu-item-413" class="menu-item list-inline-item"><a href="/media-policy/">Media Policy</a></li>
  309. <li id="menu-item-453" class="menu-item list-inline-item"><a href="/note-legali/">Privacy Policy &amp; Note Legali</a></li>
  310. <li id="menu-item-1310" class="menu-item list-inline-item"><a target="_blank" href="https://form.agid.gov.it/view/9df3de50-7a42-11ef-8989-9dcab5eaa914">Dichiarazione di accessibilità</a></li>
  311. <li id="menu-item-411" class="menu-item list-inline-item"><a href="/mappa-del-sito/">Mappa del sito</a></li>
  312. <li id="menu-item-1277" class="menu-item list-inline-item"><a href="/open-data-spid/">Open Data SPID</a></li>
  313. </ul>
  314. </nav>
  315. </div>
  316. </div>
  317. </footer>
  318. </body>
  319. </html>