mappa_logout.html 19 KB

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