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.
252 lines
7.9 KiB
252 lines
7.9 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 { format_angka } from '../util' |
|
// import collect from 'collect.js' |
|
import jquery from 'jquery' |
|
|
|
import { Sidebar } from 'primereact/sidebar' |
|
import '/node_modules/primeflex/primeflex.css' |
|
import { MantineReactTable, useMantineReactTable } from 'mantine-react-table' |
|
import { Text } from '@mantine/core' |
|
|
|
import { QueryClient, QueryClientProvider, useInfiniteQuery } from '@tanstack/react-query' |
|
import dayjs from 'dayjs' |
|
// import ViewICale from './ViewICale' |
|
// import { Dialog } from '@mui/material' |
|
var relativeTime = require('dayjs/plugin/relativeTime') |
|
var customParseFormat = require('dayjs/plugin/customParseFormat') |
|
const fetchSize = 101 |
|
|
|
const Pembayaran = ({ dataSend, visibleSidebar, setVisibleSidebar }) => { |
|
const base_url = '/engineN/' |
|
|
|
// const refChart = useRef(null) |
|
// const refChart2 = useRef(null) |
|
// const [dataJenis, setDataJenis] = useState(null) |
|
// const [dataStatus, setDataStatus] = useState(null) |
|
|
|
// const [query, setQuery] = useState(null) |
|
// const [jenisStatus, setJenisStatus] = useState('') |
|
// const [visibleICale, setVisibleIcale] = useState(false) |
|
|
|
const TableDetailGraph = ({ dataSend }) => { |
|
const tableContainerRef = useRef(null) |
|
const rowVirtualizerInstanceRef = useRef(null) |
|
const [columnFilters, setColumnFilters] = useState([]) |
|
const [globalFilter, setGlobalFilter] = useState() |
|
const [sorting, setSorting] = useState([]) |
|
const base_url = location.protocol + '//' + location.hostname + '/engineN/' |
|
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/perubahanperilaku/pembayaran/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({ |
|
...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: 'NPWP', |
|
header: 'NPWP', |
|
enableClickToCopy: true, |
|
size: 150 |
|
}, |
|
{ |
|
accessorKey: 'NAMA_WP', |
|
header: 'Nama' |
|
}, |
|
{ |
|
accessorKey: 'ALAMAT_MFWP', |
|
header: 'Alamat' |
|
}, |
|
{ |
|
accessorKey: 'KELURAHAN_MFWP', |
|
header: 'Wil. Adm.', |
|
Cell: (data) => { |
|
const dataRow = data.row.original |
|
return `${dataRow.KELURAHAN_MFWP ?? ''} ${dataRow.KECAMATAN_MFWP ?? ''} ${dataRow.KOTA_MFWP ?? ''} ${dataRow.PROPINSI_MFWP ?? ''}` |
|
} |
|
}, |
|
{ |
|
accessorKey: 'STATUS_WP_MFWP', |
|
header: 'Status WP' |
|
}, |
|
{ |
|
accessorKey: 'JNS_WP_MFWP', |
|
header: 'Jenis WP' |
|
}, |
|
{ |
|
accessorKey: 'NM_KANTOR', |
|
header: 'KPP Terdaftar' |
|
}, |
|
{ |
|
accessorKey: 'NAMA_AR_MFWP', |
|
header: 'AR' |
|
}, |
|
{ |
|
accessorKey: 'FLAG_WPS_WPK', |
|
header: 'WPS/WPK', |
|
size: 100, |
|
mantineTableBodyCellProps: { |
|
align: 'center' |
|
} |
|
}, |
|
{ |
|
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: 'TGL_DAFTAR', |
|
header: 'Tgl Daftar', |
|
Cell: ({ cell }) => { |
|
return dayjs(cell.getValue(), 'DD-MMM-YY').format('YYYY-MM-DD') |
|
} |
|
} |
|
] |
|
|
|
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) => 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' }, |
|
mantineTableBodyProps: { className: 'mb-3' } |
|
}) |
|
|
|
return <MantineReactTable table={table1} /> |
|
} |
|
|
|
const queryClient = new QueryClient() |
|
|
|
return ( |
|
<> |
|
<Row> |
|
<Col sm="12"> |
|
<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 sm="12"> |
|
<QueryClientProvider client={queryClient}> |
|
<TableDetailGraph dataSend={dataSend} /> |
|
</QueryClientProvider> |
|
</Col> |
|
</Row> |
|
</Sidebar> |
|
</Col> |
|
</Row> |
|
</> |
|
) |
|
} |
|
|
|
export default Pembayaran
|
|
|