You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

468 lines
13 KiB

import React, { useCallback, useEffect, useMemo, useRef, useState } from 'react'
import Highcharts from 'highcharts'
import HighchartsReact from 'highcharts-react-official'
import { Col, Row } from 'reactstrap'
import { Badge } from 'primereact/badge'
import { Button as ButtonP } from 'primereact/button'
import { format_angka } from '../util'
import collect from 'collect.js'
import $ from 'jquery'
import 'primereact/resources/themes/bootstrap4-light-blue/theme.css'
import 'primeflex/primeflex.css'
import { Sidebar } from 'primereact/sidebar'
import { MantineReactTable, useMantineReactTable } from 'mantine-react-table'
import { Text } from '@mantine/core'
import { QueryClient, QueryClientProvider, useInfiniteQuery } from '@tanstack/react-query'
import dayjs from 'dayjs'
var relativeTime = require('dayjs/plugin/relativeTime')
const fetchSize = 101
const PenugasanKpdl = ({ dataSend }) => {
const base_url = '<?=base_url()?>'
const refChart = useRef(null)
const [data, setData] = useState({
kpdl: [],
akum: [],
categories: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
})
const [selectedBulan, setSelectedBulan] = useState('')
const [selectedBulanText, setSelectedBulanText] = useState('semua')
const [bulan, setBulan] = useState([])
const [visibleSidebar, setVisibleSidebar] = useState(false)
const [query, setQuery] = useState(null)
useEffect(() => {
if (selectedBulan != '') {
$.get({
url: base_url + 'kewilayahan/kytp/identifikasiLapangan',
dataType: 'json',
type: 'POST',
data: {
...dataSend,
bulan: selectedBulan
},
success: (data) => {
setData(data)
}
})
}
}, [dataSend, selectedBulan])
useEffect(() => {
$.get({
url: base_url + 'kewilayahan/kytp/getBulan',
dataType: 'json',
type: 'GET',
success: (data) => {
setBulan(data)
setSelectedBulan(data[0].value)
}
})
}, [])
const optionsChart1 = () => {
return {
chart: {
zoomType: 'xy',
height: '320pt'
},
title: {
text: '',
align: 'left'
},
subtitle: {
align: 'left'
},
xAxis: [
{
categories: data.categories,
crosshair: true
}
],
yAxis: [
{
labels: {
style: {
color: Highcharts.getOptions().colors[2]
}
},
title: {
text: 'Lokasi KPDL',
style: {
color: Highcharts.getOptions().colors[2]
}
},
opposite: true
},
{
title: {
text: 'Lokasi KPDL s.d.',
style: {
color: Highcharts.getOptions().colors[0]
}
},
labels: {
style: {
color: Highcharts.getOptions().colors[0]
}
},
opposite: true
}
],
tooltip: {
shared: true
},
legend: {
layout: 'horizontal',
align: 'center',
verticalAlign: 'top',
backgroundColor:
Highcharts.defaultOptions.legend.backgroundColor || // theme
'rgba(255,255,255,0.25)'
},
plotOptions: {
series: {
cursor: 'pointer',
point: {
events: {
click: function () {
// console.log(this)
if (this.series.userOptions.seriesType === 'kpdl_tunggal') {
setQuery(this.index + 1)
setVisibleSidebar(true)
} else {
return
}
}
}
}
}
},
series: [
{
name: 'Lokasi KPDL',
seriesType: 'kpdl_tunggal',
type: 'column',
yAxis: 0,
color: Highcharts.getOptions().colors[2],
data: data.kpdl,
marker: {
enabled: true
},
tooltip: {
valueSuffix: ' Kpdl'
}
},
{
name: 'Lokasi KPDL akumulasi',
seriesType: 'kpdl_akumulasi',
type: 'spline',
yAxis: 1,
data: data.akum,
marker: {
enabled: true
},
tooltip: {
valueSuffix: ' data'
},
visible: false
}
]
}
}
const refBulanOnClick = (e) => {
const kodeBulan = e.target.dataset.value
const labelBulan = e.target.dataset.label
setSelectedBulan(kodeBulan)
setSelectedBulanText(labelBulan)
}
const TableDetailGraph = ({ dataSend, query }) => {
const tableContainerRef = useRef(null)
const rowVirtualizerInstanceRef = useRef(null)
const [columnFilters, setColumnFilters] = useState([])
const [globalFilter, setGlobalFilter] = useState()
const [sorting, setSorting] = useState([])
const base_url = '<?=base_url()?>'
const { data, fetchNextPage, isError, isFetching, isLoading } = useInfiniteQuery({
queryKey: ['table-data', columnFilters, globalFilter, sorting],
queryFn: async ({ pageParam = 0 }) => {
const url = new URL(base_url + 'kewilayahan/identaktifitashasil/identifikasilapangan/detail')
url.searchParams.set('start', `${pageParam * fetchSize}`)
url.searchParams.set('size', `${fetchSize}`)
url.searchParams.set('filters', JSON.stringify(columnFilters ?? []))
url.searchParams.set('globalFilter', globalFilter ?? '')
url.searchParams.set('sorting', JSON.stringify(sorting ?? []))
const response = await fetch(url.href, {
method: 'POST',
headers: {
Accept: 'application/json',
'Content-Type': 'application/json'
},
body: JSON.stringify({
query,
selectedBulan,
...dataSend
})
})
const json = await response.json()
return json
},
getNextPageParam: (_lastGroup, groups) => groups.length,
keepPreviousData: true,
refetchOnWindowFocus: false
})
const flatData = useMemo(() => data?.pages.flatMap((page) => page.data) ?? [], [data])
const totalDBRowCount = data?.pages?.[0]?.meta?.totalRowCount ?? 0
const totalFetched = flatData.length
const fetchMoreOnBottomReached = useCallback(
(containerRefElement) => {
if (containerRefElement) {
const { scrollHeight, scrollTop, clientHeight } = containerRefElement
//once the user has scrolled within 400px of the bottom of the table, fetch more data if we can
if (scrollHeight - scrollTop - clientHeight < 400 && !isFetching && totalFetched < totalDBRowCount) {
fetchNextPage()
}
}
},
[fetchNextPage, isFetching, totalFetched, totalDBRowCount]
)
const columns = [
{
accessorKey: 'NAMA',
header: 'Nama'
},
{
accessorKey: 'MERK_USAHA',
header: 'Merk Usaha'
},
{
accessorKey: 'NO_IDENTITAS',
header: 'No Identitas'
},
{
accessorKey: 'NPWP',
header: 'NPWP',
enableClickToCopy: true,
size: 150
},
{
accessorKey: 'ALAMAT',
header: 'Alamat'
},
{
accessorKey: 'KELURAHAN',
header: 'Wil. Adm.',
Cell: (data) => {
const dataRow = data.row.original
return `${dataRow.KELURAHAN} ${dataRow.KECAMATAN} ${dataRow.KABUPATEN} ${dataRow.PROVINSI}`
}
},
{
accessorKey: 'STATUS_WP_MFWP',
header: 'Status WP'
},
{
accessorKey: 'JNS_WP_MFWP',
header: 'Jenis WP'
},
{
accessorKey: 'NM_KANTOR_PENGAMPU',
header: 'KPP Terdaftar'
},
{
accessorKey: 'NM_AR_PENGAMPU',
header: 'AR Pengampu'
},
{
accessorKey: 'JUMLAH_PEMBAYARAN_THN_TERAKHIR',
header: 'Rp',
Cell: ({ cell }) => parseFloat(cell.getValue()).toLocaleString('id-ID'),
mantineTableHeadCellProps: {
align: 'right'
},
mantineTableBodyCellProps: {
align: 'right'
},
size: 100
},
{
accessorKey: 'KETERANGAN',
header: 'SPT'
},
{
accessorKey: 'SUM_NILAI',
header: 'NILAI DATA',
Cell: ({ cell }) => parseFloat(cell.getValue()).toLocaleString('id-ID'),
mantineTableHeadCellProps: {
align: 'right'
},
mantineTableBodyCellProps: {
align: 'right'
}
},
{
accessorKey: 'NM_KPP_ZONA',
header: 'KPP Lokasi'
},
{
accessorKey: 'NM_AR_ZONA',
header: 'AR Wilayah',
filter: false
},
{
accessorKey: 'NM_PEREKAM',
header: 'Perekam'
},
{
accessorKey: 'CREATION_DATE',
header: 'Tgl Rekam',
Cell: ({ cell }) => {
return dayjs(cell.getValue()).format('YYYY-MM-DD HH:mm:ss')
}
}
]
useEffect(() => {
if (rowVirtualizerInstanceRef.current) {
try {
rowVirtualizerInstanceRef.current.scrollToIndex(0)
} catch (e) {
console.error(e)
}
}
}, [sorting, columnFilters, globalFilter])
//a check on mount to see if the table is already scrolled to the bottom and immediately needs to fetch more data
useEffect(() => {
fetchMoreOnBottomReached(tableContainerRef.current)
}, [fetchMoreOnBottomReached])
const table1 = useMantineReactTable({
columns,
data: flatData,
enablePagination: false,
enableRowNumbers: true,
enableRowVirtualization: true, //optional, but recommended if it is likely going to be more than 100 rows
manualFiltering: true,
manualSorting: true,
mantineTableContainerProps: {
ref: tableContainerRef, //get access to the table container element
sx: { maxHeight: '600px' }, //give the table a max height
onScroll: (
event //add an event listener to the table container element
) => fetchMoreOnBottomReached(event.target)
},
mantineToolbarAlertBannerProps: {
color: 'red',
children: 'Error loading data'
},
onColumnFiltersChange: setColumnFilters,
onGlobalFilterChange: setGlobalFilter,
onSortingChange: setSorting,
renderBottomToolbarCustomActions: () => (
<Text className="text-sm">
Fetched {totalFetched} of {totalDBRowCount} total rows.
</Text>
),
state: {
columnFilters,
globalFilter,
isLoading,
showAlertBanner: isError,
showProgressBars: isFetching,
sorting
},
rowVirtualizerInstanceRef, //get access to the virtualizer instance
rowVirtualizerProps: { overscan: 10 },
mantineTableBodyCellProps: { className: 'p-1 text-xs' }
})
return <MantineReactTable table={table1} />
}
const queryClient = new QueryClient()
return (
<>
<Row>
<Col md="12">
<div className="d-flex justify-content-between border-bottom-1 pb-2">
<div>
<span className="mr-2">Bulan :</span>
{bulan.map((val, idx) => {
return (
<Badge
key={idx}
id={idx}
data-value={val.value}
data-label={val.label}
severity="warning"
value={val.label}
className="ref_bulan_a cursor-pointer mr-10"
onClick={(e) => refBulanOnClick(e)}
></Badge>
)
})}
</div>
<div>
<span>Bulan terpilih : </span>
<span>{selectedBulanText}</span>
</div>
</div>
</Col>
</Row>
<Row>
<Col>
<HighchartsReact ref={refChart} highcharts={Highcharts} options={optionsChart1()} />
</Col>
</Row>
<Row>
<Col>
<Sidebar
header={
<>
<h4>Detail Data</h4>
</>
}
visible={visibleSidebar}
position="bottom"
onHide={() => setVisibleSidebar(false)}
style={{ height: 'calc(100vh - 100px)' }}
blockScroll
pt={{ header: { className: 'p-1' }, closeButton: { style: { width: '2rem', height: '1rem' } } }}
>
<Row>
<Col>
<QueryClientProvider client={queryClient}>
<TableDetailGraph dataSend={dataSend} query={query} />
</QueryClientProvider>
</Col>
</Row>
</Sidebar>
</Col>
</Row>
</>
)
}
export default PenugasanKpdl