proyectos.html 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. {% extends "base.html" %}
  2. {% block content %}
  3. <div class="row">
  4. <div class="col-12">
  5. <div class="d-flex justify-content-between align-items-center mb-4">
  6. <h2>Simulaciones Guardadas</h2>
  7. <a href="{{ url_for('main.index') }}" class="btn btn-secondary">Volver al Inicio</a>
  8. </div>
  9. <!-- Filter Form -->
  10. <div class="card mb-4">
  11. <div class="card-body">
  12. <form action="{{ url_for('main.proyectos') }}" method="GET" class="row g-3 align-items-end">
  13. <div class="col-md-4">
  14. <label for="id_casa" class="form-label">Filtrar por Casa</label>
  15. <select name="id_casa" class="form-select" onchange="this.form.submit()">
  16. <option value="">Todas las casas</option>
  17. {% for casa in casas %}
  18. <option value="{{ casa.id_casa }}" {% if selected_casa == casa.id_casa %}selected{% endif %}>{{ casa.nombre }}</option>
  19. {% endfor %}
  20. </select>
  21. </div>
  22. <div class="col-md-2">
  23. <a href="{{ url_for('main.proyectos') }}" class="btn btn-outline-secondary">Limpiar Filtro</a>
  24. </div>
  25. </form>
  26. </div>
  27. </div>
  28. {% if proyectos %}
  29. <div class="card mb-4">
  30. <div class="card-header">
  31. Comparativa de Costos Mensuales: Sin Solar vs Con Solar
  32. </div>
  33. <div class="card-body">
  34. <canvas id="savingsChart" width="400" height="100"></canvas>
  35. </div>
  36. </div>
  37. <div class="table-responsive">
  38. <table class="table table-striped table-hover">
  39. <thead>
  40. <tr>
  41. <th>Fecha</th>
  42. <th>Cliente</th>
  43. <th>Ciudad</th>
  44. <th>Casa</th>
  45. <th>Panel</th>
  46. <th>Cantidad</th>
  47. <th>Energía (kWh/mes)</th>
  48. <th>Ahorro ($/mes)</th>
  49. <th>Acciones</th>
  50. </tr>
  51. </thead>
  52. <tbody>
  53. {% for proyecto in proyectos %}
  54. <tr>
  55. <td>{{ proyecto.fecha_referencia.strftime('%Y-%m-%d') }}</td>
  56. <td>{{ proyecto.nombre_cliente }}</td>
  57. <td>{{ proyecto.ciudad.nombre_ciudad }}</td>
  58. <td>{{ proyecto.casa.nombre if proyecto.casa else 'N/A' }}</td>
  59. <td>{{ proyecto.panel.marca }} {{ proyecto.panel.modelo }}</td>
  60. <td>{{ proyecto.cantidad_paneles }}</td>
  61. <td>{{ "%.2f"|format(proyecto.energia_estimada_mensual) }}</td>
  62. <td>{{ "%.2f"|format(proyecto.ahorro_estimado) if proyecto.ahorro_estimado else '-' }}</td>
  63. <td>
  64. <button type="button" class="btn btn-info btn-sm text-white" data-bs-toggle="modal" data-bs-target="#detallesModal"
  65. data-cliente="{{ proyecto.nombre_cliente }}"
  66. data-fecha="{{ proyecto.fecha_referencia.strftime('%Y-%m-%d') }}"
  67. data-casa="{{ proyecto.casa.nombre if proyecto.casa else 'N/A' }}"
  68. data-ciudad="{{ proyecto.ciudad.nombre_ciudad }}"
  69. data-provincia="{{ proyecto.ciudad.provincia }}"
  70. data-panel="{{ proyecto.panel.marca }} {{ proyecto.panel.modelo }}"
  71. data-eficiencia="{{ '%.1f'|format(proyecto.panel.eficiencia_r * 100) }}"
  72. data-cantidad="{{ proyecto.cantidad_paneles }}"
  73. data-energia="{{ '%.2f'|format(proyecto.energia_estimada_mensual) }}"
  74. data-ahorro="{{ '%.2f'|format(proyecto.ahorro_estimado) if proyecto.ahorro_estimado else '0' }}"
  75. data-consumo="{{ proyecto.consumo_actual_kwh if proyecto.consumo_actual_kwh else '0' }}"
  76. data-costo="{{ proyecto.costo_actual_mensual if proyecto.costo_actual_mensual else '0' }}"
  77. >
  78. Ver Detalles
  79. </button>
  80. </td>
  81. </tr>
  82. {% endfor %}
  83. </tbody>
  84. </table>
  85. </div>
  86. {% else %}
  87. <div class="alert alert-info">
  88. No hay simulaciones guardadas todavía.
  89. </div>
  90. {% endif %}
  91. </div>
  92. </div>
  93. <!-- Modal Detalles -->
  94. <div class="modal fade" id="detallesModal" tabindex="-1" aria-labelledby="detallesModalLabel" aria-hidden="true">
  95. <div class="modal-dialog modal-lg">
  96. <div class="modal-content">
  97. <div class="modal-header bg-primary text-white">
  98. <h5 class="modal-title" id="detallesModalLabel">Detalles del Proyecto</h5>
  99. <button type="button" class="btn-close btn-close-white" data-bs-dismiss="modal" aria-label="Close"></button>
  100. </div>
  101. <div class="modal-body">
  102. <div class="row mb-4">
  103. <div class="col-md-6">
  104. <h6 class="text-muted">Información General</h6>
  105. <p><strong>Cliente:</strong> <span id="modalCliente"></span></p>
  106. <p><strong>Fecha Simulación:</strong> <span id="modalFecha"></span></p>
  107. <p><strong>Casa:</strong> <span id="modalCasa"></span></p>
  108. <p><strong>Ubicación:</strong> <span id="modalUbicacion"></span></p>
  109. </div>
  110. <div class="col-md-6">
  111. <h6 class="text-muted">Sistema Solar</h6>
  112. <p><strong>Panel:</strong> <span id="modalPanel"></span></p>
  113. <p><strong>Eficiencia:</strong> <span id="modalEficiencia"></span>%</p>
  114. <p><strong>Cantidad:</strong> <span id="modalCantidad"></span> paneles</p>
  115. </div>
  116. </div>
  117. <hr>
  118. <h6 class="text-muted mb-3">Análisis Financiero y Energético</h6>
  119. <div class="row text-center">
  120. <div class="col-md-3">
  121. <div class="p-2 border rounded bg-light">
  122. <small class="d-block text-muted">Consumo Actual</small>
  123. <strong class="h5"><span id="modalConsumo"></span> kWh</strong>
  124. </div>
  125. </div>
  126. <div class="col-md-3">
  127. <div class="p-2 border rounded bg-light">
  128. <small class="d-block text-muted">Factura Actual</small>
  129. <strong class="h5 text-danger">$<span id="modalCosto"></span></strong>
  130. </div>
  131. </div>
  132. <div class="col-md-3">
  133. <div class="p-2 border rounded bg-light">
  134. <small class="d-block text-muted">Generación Solar</small>
  135. <strong class="h5 text-success"><span id="modalEnergia"></span> kWh</strong>
  136. </div>
  137. </div>
  138. <div class="col-md-3">
  139. <div class="p-2 border rounded bg-light">
  140. <small class="d-block text-muted">Ahorro Estimado</small>
  141. <strong class="h5 text-primary">$<span id="modalAhorro"></span></strong>
  142. </div>
  143. </div>
  144. </div>
  145. <div class="alert alert-info mt-4 text-center">
  146. <strong>Nuevo Costo Estimado:</strong> $<span id="modalNuevoCosto"></span> / mes
  147. </div>
  148. </div>
  149. <div class="modal-footer">
  150. <button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cerrar</button>
  151. </div>
  152. </div>
  153. </div>
  154. </div>
  155. {% endblock %}
  156. {% block scripts %}
  157. <script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
  158. <script>
  159. // Modal Logic
  160. const detallesModal = document.getElementById('detallesModal');
  161. detallesModal.addEventListener('show.bs.modal', event => {
  162. const button = event.relatedTarget;
  163. // Extract info from data-* attributes
  164. const cliente = button.getAttribute('data-cliente');
  165. const fecha = button.getAttribute('data-fecha');
  166. const casa = button.getAttribute('data-casa');
  167. const ciudad = button.getAttribute('data-ciudad');
  168. const provincia = button.getAttribute('data-provincia');
  169. const panel = button.getAttribute('data-panel');
  170. const eficiencia = button.getAttribute('data-eficiencia');
  171. const cantidad = button.getAttribute('data-cantidad');
  172. const energia = button.getAttribute('data-energia');
  173. const ahorro = parseFloat(button.getAttribute('data-ahorro'));
  174. const consumo = button.getAttribute('data-consumo');
  175. const costo = parseFloat(button.getAttribute('data-costo'));
  176. // Calculate new cost
  177. const nuevoCosto = Math.max(0, costo - ahorro).toFixed(2);
  178. // Update the modal's content.
  179. detallesModal.querySelector('#modalCliente').textContent = cliente;
  180. detallesModal.querySelector('#modalFecha').textContent = fecha;
  181. detallesModal.querySelector('#modalCasa').textContent = casa;
  182. detallesModal.querySelector('#modalUbicacion').textContent = `${ciudad}, ${provincia}`;
  183. detallesModal.querySelector('#modalPanel').textContent = panel;
  184. detallesModal.querySelector('#modalEficiencia').textContent = eficiencia;
  185. detallesModal.querySelector('#modalCantidad').textContent = cantidad;
  186. detallesModal.querySelector('#modalEnergia').textContent = energia;
  187. detallesModal.querySelector('#modalAhorro').textContent = ahorro.toFixed(2);
  188. detallesModal.querySelector('#modalConsumo').textContent = consumo;
  189. detallesModal.querySelector('#modalCosto').textContent = costo.toFixed(2);
  190. detallesModal.querySelector('#modalNuevoCosto').textContent = nuevoCosto;
  191. });
  192. const ctx = document.getElementById('savingsChart').getContext('2d');
  193. const savingsChart = new Chart(ctx, {
  194. type: 'bar',
  195. data: {
  196. labels: {{ labels | tojson }},
  197. datasets: [
  198. {
  199. label: 'Costo Sin Solar ($)',
  200. data: {{ costo_sin_solar | tojson }},
  201. backgroundColor: 'rgba(220, 53, 69, 0.6)',
  202. borderColor: 'rgba(220, 53, 69, 1)',
  203. borderWidth: 1
  204. },
  205. {
  206. label: 'Costo Con Solar ($)',
  207. data: {{ costo_con_solar | tojson }},
  208. backgroundColor: 'rgba(25, 135, 84, 0.6)',
  209. borderColor: 'rgba(25, 135, 84, 1)',
  210. borderWidth: 1
  211. }
  212. ]
  213. },
  214. options: {
  215. scales: {
  216. y: {
  217. beginAtZero: true,
  218. title: {
  219. display: true,
  220. text: 'Costo Mensual ($)'
  221. }
  222. }
  223. },
  224. plugins: {
  225. tooltip: {
  226. callbacks: {
  227. footer: function(tooltipItems) {
  228. let sinSolar = 0;
  229. let conSolar = 0;
  230. tooltipItems.forEach(function(tooltipItem) {
  231. if (tooltipItem.datasetIndex === 0) {
  232. sinSolar = tooltipItem.raw;
  233. } else if (tooltipItem.datasetIndex === 1) {
  234. conSolar = tooltipItem.raw;
  235. }
  236. });
  237. // Nota: Como el tooltip muestra un item a la vez por defecto en modo 'nearest',
  238. // esto funciona mejor si el modo es 'index'.
  239. return '';
  240. }
  241. }
  242. }
  243. },
  244. interaction: {
  245. mode: 'index',
  246. intersect: false,
  247. }
  248. }
  249. });
  250. </script>
  251. {% endblock %}