document.addEventListener('DOMContentLoaded', () => { const tableState = { data: [], page: 1, pageSize: 10, search: '', lastPage: 1, total: 0, startDate: '', endDate: '' }; const tbody = document.getElementById('tablePendingFile'); const paginationEl = document.getElementById('paginationControls'); const summaryEl = document.getElementById('tableSummary'); const pageSizeSelect = document.getElementById('tablePageSize'); const startDateInput = document.getElementById('startDate'); const endDateInput = document.getElementById('endDate'); const selectAllCheckbox = document.getElementById('selectAllPending'); const bulkApproveBtn = document.getElementById('bulkApproveBtn'); const clearSelectionBtn = document.getElementById('clearSelectionBtn'); const selectedCountEl = document.getElementById('selectedCount'); const csrfToken = document.querySelector('meta[name="csrf-token"]')?.getAttribute('content'); const selectedIds = new Set(); if (pageSizeSelect) { const initialSize = parseInt(pageSizeSelect.value); if (!isNaN(initialSize)) tableState.pageSize = initialSize; pageSizeSelect.addEventListener('change', (e) => { const val = parseInt(e.target.value); if (!isNaN(val) && val > 0) { tableState.pageSize = val; tableState.page = 1; fetchData(); } }); } window.applyDateFilter = function(){ tableState.startDate = startDateInput.value || ''; tableState.endDate = endDateInput.value || ''; tableState.page = 1; fetchData(); } function formatTanggal(dateString) { if (!dateString) return '-'; const d = new Date(dateString); return d.toLocaleDateString('id-ID', { day: '2-digit', month: 'short', year: 'numeric' }); } function statusBadge(status){ if (status === 'rejected') return 'Rejected'; if (status === 'revised') return 'Revised'; return 'Pending'; } function aksesBadge(akses){ if (akses){ return 'Umum'; } else{ return 'Internal Unit'; } } function safeText(val){ return val ? String(val) : '-'; } function isRejected(item){ return item?.status_action === 'rejected'; } function getSelectableIdsOnPage(){ return (tableState.data || []) .filter((item) => !isRejected(item)) .map((item) => String(item.file_directory_id)); } function updateSelectAllState(){ if (!selectAllCheckbox) return; const selectableIds = getSelectableIdsOnPage(); if (selectableIds.length === 0) { selectAllCheckbox.checked = false; selectAllCheckbox.indeterminate = false; selectAllCheckbox.disabled = true; return; } selectAllCheckbox.disabled = false; const selectedOnPage = selectableIds.filter((id) => selectedIds.has(id)).length; selectAllCheckbox.checked = selectedOnPage > 0 && selectedOnPage === selectableIds.length; selectAllCheckbox.indeterminate = selectedOnPage > 0 && selectedOnPage < selectableIds.length; } function updateSelectionUI(){ const count = selectedIds.size; if (selectedCountEl) selectedCountEl.textContent = String(count); if (bulkApproveBtn) bulkApproveBtn.disabled = count === 0; if (clearSelectionBtn) clearSelectionBtn.disabled = count === 0; updateSelectAllState(); } function buildRow(item){ const tanggal = item.entry_at ? formatTanggal(item.entry_at) : '-'; const tanggalExp = item.tgl_expired ? formatTanggal(item.tgl_expired) : '-'; const tanggalTerbit = item.tanggal_terbit ? formatTanggal(item.tanggal_terbit) : '-'; const id = String(item.file_directory_id); const rejected = isRejected(item); const checked = selectedIds.has(id); const aksi = `
`; return ` ${rejected ? '' : aksi} ${safeText(item.no_dokumen)} ${statusBadge(item?.status_action)} ${aksesBadge(item?.permission_file)} ${item.nama_dokumen}
${safeText(item.folder)}
${safeText(item.part)}
${tanggalTerbit} ${tanggalExp} ${tanggal}
${safeText(item.pegawai_nama_entry)}
`; } function renderPagination(totalPages){ if (!paginationEl) return; if (totalPages <= 1) { paginationEl.innerHTML = ''; return; } const maxButtons = 5; let start = Math.max(1, tableState.page - Math.floor(maxButtons/2)); let end = Math.min(totalPages, start + maxButtons - 1); start = Math.max(1, end - maxButtons + 1); let buttons = ''; buttons += ``; for (let i = start; i <= end; i++) { buttons += ``; } buttons += ``; paginationEl.innerHTML = `
${buttons}
Halaman ${tableState.page} dari ${totalPages}
`; } if (paginationEl) { paginationEl.addEventListener('click', (e) => { const page = e.target.getAttribute('data-page'); if (!page) return; if (page === 'prev' && tableState.page > 1) tableState.page--; else if (page === 'next') { if (tableState.page < tableState.lastPage) tableState.page++; } else { tableState.page = parseInt(page); } fetchData(); }); } function renderTable(){ const pageData = tableState.data || []; if (pageData.length === 0) { tbody.innerHTML = ` Tidak ada data `; } else { tbody.innerHTML = pageData.map(buildRow).join(''); } const from = tableState.total === 0 ? 0 : ((tableState.page - 1) * tableState.pageSize) + 1; const to = Math.min(((tableState.page - 1) * tableState.pageSize) + pageData.length, tableState.total); if (summaryEl) { summaryEl.textContent = tableState.total ? `Menampilkan ${from} - ${to} dari ${tableState.total} data` : 'Tidak ada data'; } renderPagination(tableState.lastPage || 1); updateSelectionUI(); } let searchDebounce; window.debouncedTableSearch = function(value){ clearTimeout(searchDebounce); searchDebounce = setTimeout(() => { tableState.search = value.trim(); tableState.page = 1; fetchData(); }, 250); } window.refreshLog = function(){ tableState.search = ''; tableState.startDate = ''; tableState.endDate = ''; document.getElementById('tableSearch').value = ''; startDateInput.value = ''; endDateInput.value = ''; selectedIds.clear(); tableState.page = 1; fetchData(); } function fetchData(){ if (summaryEl) summaryEl.textContent = 'Memuat data...'; const params = new URLSearchParams({ page: tableState.page, per_page: tableState.pageSize, keyword: tableState.search || '', start_date: tableState.startDate || '', end_date: tableState.endDate || '' }); fetch(`/datatable/pending-file?${params.toString()}`) .then(res => res.json()) .then(data => { tableState.data = data?.data || []; tableState.lastPage = data?.pagination?.last_page || 1; tableState.total = data?.pagination?.total || 0; renderTable(); }) .catch(err => { console.error(err); if (summaryEl) summaryEl.textContent = 'Gagal memuat data'; }); } window.approvePending = function(id, fileName){ Swal.fire({ title: 'Approve dokumen?', text: fileName || 'Dokumen akan disetujui.', icon: 'question', showCancelButton: true, confirmButtonText: 'Approve', cancelButtonText: 'Batal', }).then((result) => { if (!result.isConfirmed) return; fetch(`/pending-file/${id}/approve`, { method: 'POST', headers: { 'X-CSRF-TOKEN': csrfToken, 'Content-Type': 'application/json' }, }).then(async(res) => { const data = await res.json(); if (!res.ok || !data?.status) { throw new Error(data?.message || 'Gagal approve.'); } Swal.fire({ icon: 'success', title: 'Berhasil', text: data.message || 'Dokumen disetujui.', timer: 1500, showConfirmButton: false }); selectedIds.delete(String(id)); countData() fetchData(); }).catch((err) => { Swal.fire({ icon: 'error', title: 'Gagal', text: err.message || 'Terjadi kesalahan.' }); }); }); } window.rejectPending = function(id, fileName){ Swal.fire({ title: 'Reject dokumen?', text: fileName || 'Dokumen akan ditolak.', icon: 'warning', showCancelButton: true, confirmButtonText: 'Reject', cancelButtonText: 'Batal', input: 'textarea', inputLabel: 'Catatan', inputPlaceholder: 'Tulis alasan atau catatan revisi...', inputAttributes: { 'aria-label': 'Catatan' }, preConfirm: (value) => { const revision = (value || '').trim(); if (!revision) { Swal.showValidationMessage('Catatan revisi wajib diisi.'); } return revision; } }).then((result) => { if (!result.isConfirmed) return; fetch(`/pending-file/${id}/reject`, { method: 'POST', headers: { 'X-CSRF-TOKEN': csrfToken, 'Content-Type': 'application/json' }, body: JSON.stringify({ revision: result.value }) }).then(async(res) => { const data = await res.json(); if (!res.ok || !data?.status) { throw new Error(data?.message || 'Gagal reject.'); } Swal.fire({ icon: 'success', title: 'Berhasil', text: data.message || 'Dokumen ditolak.', timer: 1500, showConfirmButton: false }); selectedIds.delete(String(id)); countData() fetchData(); }).catch((err) => { Swal.fire({ icon: 'error', title: 'Gagal', text: err.message || 'Terjadi kesalahan.' }); }); }); } window.infoReject = function(id, fileName){ const item = id; const revision = item?.revision ? String(item.revision) : 'Tidak ada catatan revisi.'; Swal.fire({ title: 'Catatan Revisi', text: revision, icon: 'info', confirmButtonText: 'Tutup' }); } if (tbody) { tbody.addEventListener('change', (e) => { const target = e.target; if (!target.classList.contains('row-select')) return; const id = target.getAttribute('data-id'); if (!id) return; if (target.checked) selectedIds.add(id); else selectedIds.delete(id); const row = target.closest('tr'); if (row) row.classList.toggle('table-active', target.checked); updateSelectionUI(); }); } if (selectAllCheckbox) { selectAllCheckbox.addEventListener('change', () => { const selectableIds = getSelectableIdsOnPage(); if (selectableIds.length === 0) return; if (selectAllCheckbox.checked) { selectableIds.forEach((id) => selectedIds.add(id)); } else { selectableIds.forEach((id) => selectedIds.delete(id)); } renderTable(); }); } if (clearSelectionBtn) { clearSelectionBtn.addEventListener('click', () => { selectedIds.clear(); renderTable(); }); } if (bulkApproveBtn) { bulkApproveBtn.addEventListener('click', () => { if (selectedIds.size === 0) return; const ids = Array.from(selectedIds); Swal.fire({ title: 'Approve dokumen terpilih?', text: `Total ${ids.length} dokumen akan disetujui.`, icon: 'question', showCancelButton: true, confirmButtonText: 'Approve', cancelButtonText: 'Batal', }).then((result) => { if (!result.isConfirmed) return; fetch(`/pending-file/approve-multiple`, { method: 'POST', headers: { 'X-CSRF-TOKEN': csrfToken, 'Content-Type': 'application/json' }, body: JSON.stringify({ ids }) }).then(async(res) => { const data = await res.json(); if (!res.ok || !data?.status) { throw new Error(data?.message || 'Gagal approve dokumen.'); } ids.forEach((id) => selectedIds.delete(String(id))); Swal.fire({ icon: 'success', title: 'Berhasil', text: data.message || 'Dokumen disetujui.', timer: 1500, showConfirmButton: false }); countData() fetchData(); }).catch((err) => { Swal.fire({ icon: 'error', title: 'Gagal', text: err.message || 'Terjadi kesalahan.' }); }); }); }); } fetchData(); }); document.addEventListener('click', function(e){ if(e.target.matches('.file-link')){ e.preventDefault(); let fileUrl = e.target.getAttribute('data-file'); let noDokumen = e.target.getAttribute('data-no_dokumen') let tanggalTerbit = e.target.getAttribute('data-tanggal_terbit') let permissionFile = e.target.getAttribute('data-permission_file') let fileName = e.target.getAttribute('data-fileName') currentFile = fileUrl; idDirectory = e.target.getAttribute('data-id'); const titleEl = document.getElementById('confirm_preview_file'); if (titleEl) titleEl.textContent = fileName; // set footer info const noEl = document.getElementById('confirm-upload-dokumen'); if (noEl) noEl.textContent = noDokumen; const tglEl = document.getElementById('confirm-time-dokumen'); if (tglEl) tglEl.textContent = tanggalTerbit; const permEl = document.getElementById('confirm-permission'); if (permEl) { const publicDoc = isPublic(permissionFile); permEl.textContent = publicDoc ? 'Umum' : 'Internal Unit'; permEl.className = 'badge ' + (publicDoc ? 'bg-success' : 'bg-secondary'); } let previewBox = document.getElementById('file-preview'); previewBox.innerHTML = `
`; openPreview(idDirectory); $("#previewModal").modal('show') } if(e.target.matches('#btn-view-full')){ window.open(`/full-preview/${idDirectory}`, '_blank'); } }) function isPublic(permissionVal){ if(permissionVal === null || permissionVal === undefined) return false; const val = String(permissionVal).toLowerCase(); return val === '1' || val === 'true' || val === 'iya' || val === 'yes'; }