alta_de_horario.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. <?php
  2. require_once 'class/c_login.php';
  3. $user = Login::get_user();
  4. $user->access();
  5. if (in_array($user->acceso, ['r', 'n']))
  6. die(header('Location: main.php?error=1'));
  7. $user->print_to_log('Consultar: Alta de horario');
  8. ?>
  9. <!DOCTYPE html>
  10. <html lang="en">
  11. <head>
  12. <title>Cargar horario desde Excel | <?= $user->facultad['facultad'] ?? 'General' ?></title>
  13. <meta charset="utf-8">
  14. <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
  15. <?php include_once "import/html_css_files.php"; ?>
  16. </head>
  17. <body style="display: block;">
  18. <?php
  19. include('include/constantes.php');
  20. include("import/html_header.php");
  21. html_header("Cargar horario desde Excel", "Sistema de gestión de checador");
  22. ?>
  23. <main class="container content marco content-margin" id="local-app">
  24. <div class="row mb-3">
  25. <div class="col-12 text-right">
  26. <button class="btn btn-outline-secondary" data-toggle="modal" data-target="#modalDescargarPlantilla">
  27. <span class="ing-descarga ing-fw"></span>
  28. Descargar plantilla
  29. </button>
  30. </div>
  31. </div>
  32. <section id="message"></section>
  33. <?php require('import/periodo.php') ?>
  34. <form>
  35. <div class="form-group">
  36. <div class="form-box">
  37. <?php
  38. $periodo = $db->where('id', $user->periodo_id)->getOne('fs_periodo');
  39. $carreras = $db
  40. ->where('nivel', $periodo['nivel_id'])
  41. ->where('facultad', $user->facultad['facultad_id'])
  42. ->orderBy('carrera')
  43. ->get('fs_carrera');
  44. ?>
  45. <div class="form-group row" id="input-carrera">
  46. <label for="filter_carrera" class="col-4 col-form-label">Carrera</label>
  47. <div class="col-6">
  48. <div id="dlcarrera" class="datalist datalist-select mb-1 w-100">
  49. <div class="datalist-input">Seleccionar carrera</div>
  50. <span class="ing-buscar icono"></span>
  51. <ul style="display:none">
  52. <?php
  53. foreach ($carreras as $carrera) {
  54. ?>
  55. <li data-id="<?= $carrera['id'] ?>">
  56. <?= $carrera['carrera'] ?>
  57. </li>
  58. <?php
  59. }
  60. ?>
  61. </ul>
  62. <input type="hidden" id="filter_carrera" name="carrera" value="">
  63. </div>
  64. </div>
  65. </div>
  66. <div class="form-group row" id="input-file">
  67. <label for="excel" class="col-4 col-form-label">Archivo de horarios</label>
  68. <div class="col-8 col-sm-6">
  69. <input class="form-control-file" id="excel" name="archivo" accept=".xlsx, .xls" require>
  70. </div>
  71. </div>
  72. <div class="form-group mt-5 row justify-content-center">
  73. <!-- on click reload -->
  74. <button id="btn-cancelar" type="button" class="btn btn-danger mx-2" onclick="location.reload()">
  75. <span class="ing-cancelar"></span>
  76. Cancelar
  77. </button>
  78. <button id="btn-cargar" type="button" class="btn btn-primary" onclick="submit_files()">
  79. <span class="ing-guardar"></span>
  80. Guardar horario
  81. </button>
  82. </div>
  83. </div>
  84. </div>
  85. </form>
  86. </main>
  87. <div class="modal fade" id="modalDescargarPlantilla" tabindex="-1" aria-labelledby="descargarPlantillaModalLabel" aria-hidden="true">
  88. <div class="modal-dialog modal-xl modal-dialog-centered modal-dialog-scrollable">
  89. <div class="modal-content">
  90. <div class="modal-header">
  91. <h5 class="modal-title" id="descargarPlantillaModalLabel">Instrucciones</h5>
  92. <button type="button" class="close text-white" data-dismiss="modal" aria-label="Close">
  93. <span aria-hidden="true">&times;</span>
  94. </button>
  95. </div>
  96. <div class="modal-body">
  97. <ol>
  98. <li> Los encabezados deberán mantenerse en todo momento, tal cual se encuentran en la plantilla. </li>
  99. <li> Las horas válidas son de 7:00 a 22:00 en bloques de 15 minutos. </li>
  100. <li> La columna DURACIÓN es la duración de una clase en horas. Si la clase es de 7:15–8:45 se pone que la hora de la clase es de 7:15 con una duración de 180. </li>
  101. <!-- <li> Únicamente las columnas de … </li> -->
  102. <li> Si se encuentra un error en la revisión del archivo no se guardará ningún registro. </li>
  103. <li> Una vez revisado el archivo haz clic en el botón [Guardar horario] para que se guarde la información. </li>
  104. </ol>
  105. </div>
  106. <div class="modal-footer">
  107. <button type="button" class="btn btn-danger" data-dismiss="modal">Aceptar</button>
  108. <button type="button" class="btn btn-primary" id="descargarPlantilla"><span class="ing-descarga"></span> Descargar</button>
  109. <button type="button" class="btn btn-primary" id="descargarPlantillaEjemplo"><span class="ing-descarga"></span> Descargar ejemplo</button>
  110. </div>
  111. </div>
  112. </div>
  113. </div>
  114. </body>
  115. <?php
  116. require_once("import/html_footer.php");
  117. require_once("js/messages.php")
  118. ?>
  119. <script src="js/scrollables.js"></script>
  120. <script src="js/jquery.min.js"></script>
  121. <script src="js/bootstrap/bootstrap.min.js"></script>
  122. <script src="js/custominputfile.min-es.js"></script>
  123. <link rel="stylesheet" href="css/custominputfile.min.css">
  124. <script src="js/fetchlib.js"></script>
  125. <script>
  126. var datum = []
  127. $(document).ready(function() {
  128. $('#excel').customFile({
  129. allowed: ['xlsx', 'xls'],
  130. maxFiles: 1,
  131. callbacks: {
  132. onSuccess: async function(item) {
  133. var formData = $.customFile.serialize('archivo');
  134. const {
  135. status,
  136. message,
  137. data
  138. } = await fetch('action/action_revisar_excel.php', {
  139. method: 'POST',
  140. body: formData,
  141. })
  142. .then(response => response.json())
  143. .catch(error => {
  144. return {
  145. status: 'error',
  146. message: 'Error al cargar el archivo',
  147. }
  148. });
  149. if (status == 'error') {
  150. triggerMessage(message, 'Error en el formato del archivo');
  151. item.destroy();
  152. return
  153. }
  154. triggerMessage(message + " Haz clic en el botón <b>Guardar horario</b> para guardar los datos", `Archivo revisado, aún <b>no guardado</b>`, 'primary');
  155. datum = data;
  156. // hide form
  157. document.querySelector('#input-file').classList.add('d-none');
  158. document.querySelector('#input-carrera').classList.add('d-none');
  159. // show button
  160. document.querySelector('#btn-cargar').classList.remove('d-none');
  161. document.querySelector('#btn-cancelar').classList.remove('d-none');
  162. },
  163. }
  164. });
  165. // hide
  166. document.querySelector('#input-file').classList.add('d-none');
  167. document.querySelector('#btn-cancelar').classList.add('d-none');
  168. document.querySelector('#btn-cargar').classList.add('d-none');
  169. // on click in carrera
  170. [... document.querySelectorAll('#dlcarrera ul li')].forEach(
  171. li => li.addEventListener('click', () => {
  172. document.querySelector('#input-file').classList.remove('d-none')
  173. })
  174. )
  175. })
  176. async function submit_files() {
  177. // disable button
  178. const button = document.querySelector('#btn-cargar');
  179. // add class disabled to button
  180. button.classList.add('disabled');
  181. // disable button
  182. button.disabled = true;
  183. // add loading icon
  184. button.innerHTML = '<span class="ing-cargando"></span> Cargando...';
  185. let missing = [];
  186. let carrera = $('#filter_carrera').val();
  187. if (carrera == '') missing.push('Carrera');
  188. if (datum.length == 0) missing.push('Archivo de horarios');
  189. let facultad = <?= $user->facultad['facultad_id'] ?>;
  190. if (missing.length > 0) {
  191. messageMissingInputs(missing);
  192. // remove class disabled to button
  193. button.classList.remove('disabled');
  194. // enable button
  195. button.disabled = false;
  196. // remove loading icon
  197. button.innerHTML = '<span class="ing-guardar"></span> Guardar horario';
  198. return;
  199. }
  200. const formData = new FormData();
  201. formData.append('carrera', carrera);
  202. formData.append('facultad', facultad);
  203. formData.append('periodo', <?= $user->periodo_id ?>);
  204. formData.append('data', JSON.stringify(datum));
  205. const {
  206. status,
  207. message
  208. } = await fetch('action/action_horario_excel.php', {
  209. method: 'POST',
  210. body: formData
  211. })
  212. .then(response => response.json())
  213. .catch(error => {
  214. return {
  215. status: 'error',
  216. message: 'Error al cargar el archivo',
  217. }
  218. });
  219. triggerMessage(message, (status == 'error') ? 'Error al guardar el archivo' : 'Horarios guardados', (status == 'error') ? 'danger' : 'success');
  220. // await 1 second
  221. // await setTimeout(() => {}, 1000);
  222. // remove class disabled to
  223. button.classList.remove('disabled');
  224. // enable button
  225. button.disabled = false;
  226. button.innerHTML = 'Cargar otro horario';
  227. // refresh page
  228. button.onclick = () => location.reload()
  229. // hide button
  230. document.querySelector('#btn-cancelar').classList.add('d-none');
  231. }
  232. document.querySelector('#descargarPlantilla').addEventListener('click', function() {
  233. window.open('template/plantilla.xlsx', '_blank');
  234. });
  235. document.querySelector('#descargarPlantillaEjemplo').addEventListener('click', function() {
  236. window.open('template/ejemplo.xlsx', '_blank');
  237. });
  238. </script>
  239. </html>