mappa_logout.html 18 KB

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