reports.js 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. import 'tailwindcss';
  2. import $ from 'jquery';
  3. import 'laravel-datatables-vite';
  4. import JSZip from 'jszip'; // Required for Excel and CSV export
  5. import 'datatables.net';
  6. import 'datatables.net-buttons';
  7. import 'datatables.net-buttons/js/buttons.html5.js'; // For HTML5 export buttons
  8. import 'datatables.net-buttons/js/buttons.print.min.js'; // For the print button
  9. import 'datatables.net-responsive';
  10. import 'pdfmake/build/pdfmake.js'; // Required for PDF export
  11. import 'pdfmake/build/vfs_fonts.js'; // Required for PDF export
  12. import './dataTables.tailwindcss.js';
  13. import 'datatables.net-responsive-dt';
  14. // Ensure jQuery is available globally
  15. window.$ = window.jQuery = $;
  16. window.JSZip = JSZip;
  17. // Initialize DataTable
  18. var dataTableInstance;
  19. var searchTerms ={};
  20. function initializeDataTable() {
  21. var dataTableName = document.getElementById('datatable-config').dataset.datatableName;
  22. var dataTableColumnsConfig = JSON.parse(document.getElementById('datatable-config').dataset.datatableColumnsConfig);
  23. var dataTableHeadersConfig = JSON.parse(document.getElementById('datatable-config').dataset.datatableHeadersConfig);
  24. var dataTablesButtonsConfig = JSON.parse(document.getElementById('datatable-config').dataset.datatableButtonsConfig);
  25. var dataTablesColumnDefsConfig = JSON.parse(document.getElementById('datatable-config').dataset.datatableColumnDefsConfig);
  26. //Strip out HTML from headers when exporting
  27. dataTablesButtonsConfig.forEach(function(buttonConfig){
  28. buttonConfig.exportOptions = buttonConfig.exportOptions || {};
  29. for(let key in buttonConfig)
  30. {
  31. if(key === "exportOptions")
  32. {
  33. buttonConfig.exportOptions.format = {
  34. header: function (data){
  35. const tempDiv = document.createElement('div');
  36. tempDiv.innerHTML = data;
  37. // Remove all <select> elements
  38. tempDiv.querySelectorAll('select').forEach(el => el.remove());
  39. return tempDiv.textContent.trim().replace(/\s+/g, ' ');
  40. }
  41. }
  42. }
  43. }
  44. })
  45. //Set up any action columns
  46. function doActionButton(url, id)
  47. {
  48. window.open(url, "_blank");
  49. }
  50. window.doActionButton=doActionButton;
  51. var renderFunctions = [];
  52. dataTableHeadersConfig.forEach(function (headerColumn, columnIndex){
  53. if(headerColumn.type === "action")
  54. {
  55. let actionButtons = dataTableColumnsConfig[columnIndex].actionButtons;
  56. let renderBody = '';
  57. renderBody = `if (type === "display"){`;
  58. renderBody += `return '`;
  59. actionButtons.forEach(function (actionButton){
  60. let buttonUrl = actionButton.buttonUrl;
  61. let parameters = actionButton.querystring_parameters;
  62. renderBody += `<button class="action-button" onclick="doActionButton(`;
  63. renderBody += `\\'`+buttonUrl;
  64. parameters.forEach(function (parameter, index){
  65. if(index === 0)
  66. {
  67. renderBody += '?';
  68. }
  69. else
  70. {
  71. renderBody += '&';
  72. }
  73. for(let key in parameter)
  74. {
  75. renderBody += key+'=';
  76. if(parameter[key] === 'row.id')
  77. {
  78. renderBody += '\'+'+encodeURIComponent(parameter[key])+'+\'';
  79. }
  80. else
  81. {
  82. renderBody += encodeURIComponent(parameter[key]);
  83. }
  84. }
  85. })
  86. renderBody += `\\'`;
  87. renderBody += ")\">"+actionButton.buttonText+"</button>&nbsp;";
  88. });
  89. renderBody += "'}return data;";
  90. renderFunctions[columnIndex] = new Function('data', 'type', 'row', 'meta', renderBody);
  91. dataTableColumnsConfig[columnIndex].searchable = false;
  92. dataTableColumnsConfig[columnIndex].orderable = false;
  93. dataTableColumnsConfig[columnIndex].render = renderFunctions[columnIndex];
  94. }
  95. })
  96. dataTableInstance = $(`#${dataTableName}-table`).DataTable({
  97. responsive: true,
  98. retrieve: true,
  99. searching: false,
  100. processing: true,
  101. serverSide: true,
  102. orderCellsTop: true,
  103. fixedHeader: true,
  104. ajax: {
  105. url: '/api/report-data/' + dataTableName,
  106. type: 'POST',
  107. dataSrc: function (json) {
  108. //populate any dropdown headers
  109. dataTableHeadersConfig.forEach(function(headerConfig, index){
  110. if(headerConfig.searchtype == "dropdown")
  111. {
  112. var dropdown = $('#'+headerConfig.header.replace(' ', '-')+'FilterDropdown');
  113. dropdown.empty().append($('<option>',{
  114. value: '',
  115. text: '-- Not Filtered --'
  116. }));
  117. json.dropdowndata[index].forEach(function (option) {
  118. var selected = false;
  119. if(json.searchterms && json.searchterms[index]){
  120. Object.keys(searchTerms[index]).forEach(function (key){
  121. var searchValue = searchTerms[index][key]
  122. if(option === searchValue)
  123. {
  124. selected = true;
  125. }})}
  126. dropdown.append($('<option>',{
  127. value: option,
  128. text: option
  129. }).prop('selected', selected))});
  130. }});
  131. return json.data;
  132. },
  133. data: function (d) {
  134. d.searchTerms = searchTerms || {};
  135. }
  136. },
  137. columns: dataTableColumnsConfig,
  138. dom: 'lBfrtip',
  139. buttons: dataTablesButtonsConfig,
  140. layout: {
  141. topStart: 'buttons'
  142. },
  143. columnDefs: dataTablesColumnDefsConfig
  144. });
  145. }
  146. document.addEventListener('livewire:init', function () {
  147. initializeDataTable();
  148. });
  149. let inner = {};
  150. function getFilterInput(searchTerm, index, searchType, extra){
  151. switch (searchType) {
  152. case "text":
  153. case "number":
  154. inner[index] = searchTerm;
  155. break;
  156. case "date":
  157. if($.isEmptyObject(inner[index])) {
  158. inner[index] = {"start": "", "end": ""};
  159. }
  160. if (extra === "start") {
  161. inner[index].start = searchTerm;
  162. }
  163. else if (extra === "end") {
  164. inner[index].end = searchTerm;
  165. }
  166. break;
  167. case "dropdown":
  168. inner[index] = searchTerm;
  169. break;
  170. }
  171. searchTerms[index] = inner;
  172. dataTableInstance.ajax.reload();
  173. }
  174. window.getFilterInput = getFilterInput;
  175. document.addEventListener('DOMContentLoaded', () => {
  176. });