feat: wip journal and general ledger dyanmic columns

This commit is contained in:
Ahmed Bouhuolia
2024-01-06 20:16:22 +02:00
parent c71836ec27
commit 79f3f1b63d
15 changed files with 615 additions and 214 deletions

View File

@@ -1,7 +1,8 @@
import * as R from 'ramda'; import * as R from 'ramda';
import { first } from 'lodash';
import { import {
IColumnMapperMeta,
IJournalEntry, IJournalEntry,
IJournalReport,
IJournalReportEntriesGroup, IJournalReportEntriesGroup,
IJournalReportQuery, IJournalReportQuery,
IJournalTableData, IJournalTableData,
@@ -13,7 +14,7 @@ import { tableRowMapper } from '@/utils';
import { FinancialTable } from '../FinancialTable'; import { FinancialTable } from '../FinancialTable';
import { FinancialSheetStructure } from '../FinancialSheetStructure'; import { FinancialSheetStructure } from '../FinancialSheetStructure';
import FinancialSheet from '../FinancialSheet'; import FinancialSheet from '../FinancialSheet';
import { first } from 'lodash'; import { ROW_TYPE } from './types';
export class JournalSheetTable extends R.compose( export class JournalSheetTable extends R.compose(
FinancialTable, FinancialTable,
@@ -70,6 +71,10 @@ export class JournalSheetTable extends R.compose(
]; ];
}; };
/**
* Retrieves the total entry column accessors.
* @returns {ITableColumnAccessor[]}
*/
private totalEntryColumnAccessors = (): ITableColumnAccessor[] => { private totalEntryColumnAccessors = (): ITableColumnAccessor[] => {
return [ return [
{ key: 'date', accessor: '_empty_' }, { key: 'date', accessor: '_empty_' },
@@ -83,6 +88,23 @@ export class JournalSheetTable extends R.compose(
]; ];
}; };
/**
* Retrieves the total entry column accessors.
* @returns {IColumnMapperMeta[]}
*/
private blankEnrtyColumnAccessors = (): IColumnMapperMeta[] => {
return [
{ key: 'date', value: '' },
{ key: 'transaction_type', value: '' },
{ key: 'transaction_number', value: '' },
{ key: 'description', value: '' },
{ key: 'account_code', value: '' },
{ key: 'account_name', value: '' },
{ key: 'credit', value: '' },
{ key: 'debit', value: '' },
];
};
/** /**
* Retrieves the common columns. * Retrieves the common columns.
* @returns {ITableColumn[]} * @returns {ITableColumn[]}
@@ -113,6 +135,7 @@ export class JournalSheetTable extends R.compose(
}; };
const computedGroup = { ...group, entry: first(group.entries) }; const computedGroup = { ...group, entry: first(group.entries) };
const columns = this.groupColumnsAccessors(); const columns = this.groupColumnsAccessors();
return tableRowMapper(computedGroup, columns, meta); return tableRowMapper(computedGroup, columns, meta);
}; };
@@ -153,6 +176,16 @@ export class JournalSheetTable extends R.compose(
return tableRowMapper(group, total, meta); return tableRowMapper(group, total, meta);
}; };
/**
* Retrieves the blank entry row.
* @returns {ITableRow}
*/
private blankEntryMapper = (): ITableRow => {
const columns = this.blankEnrtyColumnAccessors();
const meta = {};
return tableRowMapper({} as IJournalEntry, columns, meta);
};
/** /**
* Maps the entry group to table rows. * Maps the entry group to table rows.
* @param {IJournalReportEntriesGroup} group - * @param {IJournalReportEntriesGroup} group -
@@ -162,8 +195,9 @@ export class JournalSheetTable extends R.compose(
const firstRow = this.firstEntryGroupMapper(group); const firstRow = this.firstEntryGroupMapper(group);
const lastRows = this.entriesMapper(group); const lastRows = this.entriesMapper(group);
const totalRow = this.totalEntryMapper(group); const totalRow = this.totalEntryMapper(group);
const blankRow = this.blankEntryMapper();
return [firstRow, ...lastRows, totalRow]; return [firstRow, ...lastRows, totalRow, blankRow];
}; };
/** /**

View File

@@ -1,6 +1,5 @@
export enum ROW_TYPE {
enum ROW_TYPE {
ENTRY = 'ENTRY', ENTRY = 'ENTRY',
TOTAL = 'TOTAL' TOTAL = 'TOTAL'
}; };

View File

@@ -12,6 +12,7 @@ import {
import classNames from 'classnames'; import classNames from 'classnames';
import { DashboardActionsBar, FormattedMessage as T, Icon } from '@/components'; import { DashboardActionsBar, FormattedMessage as T, Icon } from '@/components';
import { GeneralLedgerSheetExportMenu } from './components';
import { useGeneralLedgerContext } from './GeneralLedgerProvider'; import { useGeneralLedgerContext } from './GeneralLedgerProvider';
import { compose } from '@/utils'; import { compose } from '@/utils';
@@ -84,11 +85,18 @@ function GeneralLedgerActionsBar({
icon={<Icon icon="print-16" iconSize={16} />} icon={<Icon icon="print-16" iconSize={16} />}
text={<T id={'print'} />} text={<T id={'print'} />}
/> />
<Button <Popover
className={Classes.MINIMAL} content={<GeneralLedgerSheetExportMenu />}
icon={<Icon icon="file-export-16" iconSize={16} />} interactionKind={PopoverInteractionKind.CLICK}
text={<T id={'export'} />} placement="bottom-start"
/> minimal
>
<Button
className={Classes.MINIMAL}
icon={<Icon icon="file-export-16" iconSize={16} />}
text={<T id={'export'} />}
/>
</Popover>
</NavbarGroup> </NavbarGroup>
</DashboardActionsBar> </DashboardActionsBar>
); );

View File

@@ -10,7 +10,7 @@ const GeneralLedgerHeaderDimensionsPanelContext = React.createContext();
/** /**
* General Ledger Header Dimensions Panel provider. * General Ledger Header Dimensions Panel provider.
* @returns * @returns {JSX.Element}
*/ */
function GeneralLedgerHeaderDimensionsPanelProvider({ query, ...props }) { function GeneralLedgerHeaderDimensionsPanelProvider({ query, ...props }) {
// Features guard. // Features guard.

View File

@@ -29,6 +29,7 @@ function GeneralLedgerProvider({ query, ...props }) {
sheetRefresh: refetch, sheetRefresh: refetch,
isFetching, isFetching,
isLoading, isLoading,
httpRequest: requestQuery
}; };
return ( return (
<FinancialReportPage name={'general-ledger-sheet'}> <FinancialReportPage name={'general-ledger-sheet'}>

View File

@@ -13,7 +13,7 @@ import {
} from '@/components'; } from '@/components';
import { useGeneralLedgerContext } from './GeneralLedgerProvider'; import { useGeneralLedgerContext } from './GeneralLedgerProvider';
import { useGeneralLedgerTableColumns } from './components'; import { useGeneralLedgerTableColumns } from './dynamicColumns';
/** /**
* General ledger table. * General ledger table.
@@ -21,7 +21,7 @@ import { useGeneralLedgerTableColumns } from './components';
export default function GeneralLedgerTable({ companyName }) { export default function GeneralLedgerTable({ companyName }) {
// General ledger context. // General ledger context.
const { const {
generalLedger: { tableRows, query }, generalLedger: { query, table },
isLoading, isLoading,
} = useGeneralLedgerContext(); } = useGeneralLedgerContext();
@@ -30,8 +30,8 @@ export default function GeneralLedgerTable({ companyName }) {
// Default expanded rows of general ledger table. // Default expanded rows of general ledger table.
const expandedRows = useMemo( const expandedRows = useMemo(
() => defaultExpanderReducer(tableRows, 1), () => defaultExpanderReducer(table.rows, 1),
[tableRows], [table.rows],
); );
return ( return (
@@ -48,7 +48,7 @@ export default function GeneralLedgerTable({ companyName }) {
'this_report_does_not_contain_any_data_between_date_period', 'this_report_does_not_contain_any_data_between_date_period',
)} )}
columns={columns} columns={columns}
data={tableRows} data={table.rows}
rowClassNames={tableRowTypesToClassnames} rowClassNames={tableRowTypesToClassnames}
expanded={expandedRows} expanded={expandedRows}
virtualizedRows={true} virtualizedRows={true}

View File

@@ -1,107 +1,31 @@
// @ts-nocheck // @ts-nocheck
import React from 'react'; import React, { useRef } from 'react';
import intl from 'react-intl-universal'; import classNames from 'classnames';
import { Button } from '@blueprintjs/core'; import {
import { FormattedMessage as T, Icon, If } from '@/components'; Button,
Classes,
Intent,
Menu,
MenuItem,
ProgressBar,
Text,
} from '@blueprintjs/core';
import {
FormattedMessage as T,
Icon,
If,
Stack,
AppToaster,
} from '@/components';
import { getColumnWidth } from '@/utils';
import { useGeneralLedgerContext } from './GeneralLedgerProvider'; import { useGeneralLedgerContext } from './GeneralLedgerProvider';
import FinancialLoadingBar from '../FinancialLoadingBar'; import FinancialLoadingBar from '../FinancialLoadingBar';
import { FinancialComputeAlert } from '../FinancialReportPage'; import { FinancialComputeAlert } from '../FinancialReportPage';
import { Align } from '@/constants'; import {
useGeneralLedgerSheetCsvExport,
/** useGeneralLedgerSheetXlsxExport,
* Retrieve the general ledger table columns. } from '@/hooks/query';
*/
export function useGeneralLedgerTableColumns() {
// General ledger context.
const {
generalLedger: { tableRows },
} = useGeneralLedgerContext();
return React.useMemo(
() => [
{
Header: intl.get('date'),
accessor: 'date',
className: 'date',
width: 120,
},
{
Header: intl.get('account_name'),
accessor: 'name',
className: 'name',
textOverview: true,
},
{
Header: intl.get('transaction_type'),
accessor: 'reference_type_formatted',
className: 'transaction_type',
width: 125,
textOverview: true,
},
{
Header: intl.get('transaction_number'),
accessor: 'reference_id',
className: 'transaction_number',
width: 100,
textOverview: true,
},
{
Header: intl.get('description'),
accessor: 'note',
className: 'description',
textOverview: true,
},
{
Header: intl.get('credit'),
accessor: 'formatted_credit',
className: 'credit',
width: getColumnWidth(tableRows, 'formatted_credit', {
minWidth: 100,
magicSpacing: 10,
}),
textOverview: true,
align: Align.Right,
},
{
Header: intl.get('debit'),
accessor: 'formatted_debit',
className: 'debit',
width: getColumnWidth(tableRows, 'formatted_debit', {
minWidth: 100,
magicSpacing: 10,
}),
textOverview: true,
align: Align.Right,
},
{
Header: intl.get('amount'),
accessor: 'formatted_amount',
className: 'amount',
width: getColumnWidth(tableRows, 'formatted_amount', {
minWidth: 100,
magicSpacing: 10,
}),
textOverview: true,
align: Align.Right,
},
{
Header: intl.get('running_balance'),
accessor: 'formatted_running_balance',
className: 'running_balance',
width: getColumnWidth(tableRows, 'formatted_running_balance', {
minWidth: 100,
magicSpacing: 10,
}),
textOverview: true,
align: Align.Right,
},
],
[tableRows],
);
}
/** /**
* General ledger sheet alerts. * General ledger sheet alerts.
@@ -144,3 +68,93 @@ export function GeneralLedgerSheetLoadingBar() {
</If> </If>
); );
} }
/**
* Renders the G/L sheet export menu.
* @returns {JSX.Element}
*/
export const GeneralLedgerSheetExportMenu = () => {
const toastKey = useRef(null);
const commonToastConfig = {
isCloseButtonShown: true,
timeout: 2000,
};
const { httpRequest } = useGeneralLedgerContext();
const openProgressToast = (amount: number) => {
return (
<Stack spacing={8}>
<Text>The report has been exported successfully.</Text>
<ProgressBar
className={classNames('toast-progress', {
[Classes.PROGRESS_NO_STRIPES]: amount >= 100,
})}
intent={amount < 100 ? Intent.PRIMARY : Intent.SUCCESS}
value={amount / 100}
/>
</Stack>
);
};
// Export the report to xlsx.
const { mutateAsync: xlsxExport } = useGeneralLedgerSheetXlsxExport(
httpRequest,
{
onDownloadProgress: (xlsxExportProgress: number) => {
if (!toastKey.current) {
toastKey.current = AppToaster.show({
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
});
} else {
AppToaster.show(
{
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
},
toastKey.current,
);
}
},
},
);
// Export the report to csv.
const { mutateAsync: csvExport } = useGeneralLedgerSheetCsvExport(
httpRequest,
{
onDownloadProgress: (xlsxExportProgress: number) => {
if (!toastKey.current) {
toastKey.current = AppToaster.show({
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
});
} else {
AppToaster.show(
{
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
},
toastKey.current,
);
}
},
},
);
// Handle csv export button click.
const handleCsvExportBtnClick = () => {
csvExport().then(() => {});
};
// Handle xlsx export button click.
const handleXlsxExportBtnClick = () => {
xlsxExport().then(() => {});
};
return (
<Menu>
<MenuItem
text={'XLSX (Microsoft Excel)'}
onClick={handleXlsxExportBtnClick}
/>
<MenuItem text={'CSV'} onClick={handleCsvExportBtnClick} />
</Menu>
);
};

View File

@@ -0,0 +1,117 @@
// @ts-nocheck
import { getColumnWidth } from '@/utils';
import * as R from 'ramda';
import { useGeneralLedgerContext } from './GeneralLedgerProvider';
import { Align } from '@/constants';
const getTableCellValueAccessor = (index) => `cells[${index}].value`;
const getReportColWidth = (data, accessor, headerText) => {
return getColumnWidth(
data,
accessor,
{ magicSpacing: 10, minWidth: 100 },
headerText,
);
};
/**
* Account name column mapper.
*/
const commonColumnMapper = R.curry((data, column) => {
const accessor = getTableCellValueAccessor(column.cell_index);
return {
key: column.key,
Header: column.label,
accessor,
className: column.key,
textOverview: true,
};
});
/**
* Numeric columns accessor.
*/
const numericColumnAccessor = R.curry((data, column) => {
const accessor = getTableCellValueAccessor(column.cell_index);
const width = getReportColWidth(data, accessor, column.label);
return {
...column,
align: Align.Right,
width,
};
});
/**
* Date column accessor.
*/
const dateColumnAccessor = R.curry((column) => {
return {
...column,
width: 120,
};
});
/**
* Transaction type column accessor.
*/
const transactionTypeColumnAccessor = (column) => {
return {
...column,
width: 125,
};
};
/**
* Transaction number column accessor.
*/
const transactionIdColumnAccessor = (column) => {
return {
...column,
width: 100,
};
};
const dynamiColumnMapper = R.curry((data, column) => {
const _numericColumnAccessor = numericColumnAccessor(data);
return R.compose(
R.when(R.pathEq(['key'], 'date'), dateColumnAccessor),
R.when(
R.pathEq(['key'], 'reference_type'),
transactionTypeColumnAccessor,
),
R.when(
R.pathEq(['key'], 'reference_number'),
transactionIdColumnAccessor,
),
R.when(R.pathEq(['key'], 'credit'), _numericColumnAccessor),
R.when(R.pathEq(['key'], 'debit'), _numericColumnAccessor),
R.when(R.pathEq(['key'], 'amount'), _numericColumnAccessor),
R.when(R.pathEq(['key'], 'running_balance'), _numericColumnAccessor),
commonColumnMapper(data),
)(column);
});
/**
* Composes the dynamic columns that fetched from request to columns to table component.
*/
export const dynamicColumns = R.curry((data, columns) => {
return R.map(dynamiColumnMapper(data), columns);
});
/**
* Retrieves the G/L sheet table columns for table component.
*/
export const useGeneralLedgerTableColumns = () => {
const { generalLedger } = useGeneralLedgerContext();
if (!generalLedger) {
throw new Error('asdfadsf');
}
const { table } = generalLedger;
return dynamicColumns(table.rows, table.columns);
};

View File

@@ -18,6 +18,7 @@ import withJournal from './withJournal';
import { compose } from '@/utils'; import { compose } from '@/utils';
import { useJournalSheetContext } from './JournalProvider'; import { useJournalSheetContext } from './JournalProvider';
import { JournalSheetExportMenu } from './components';
/** /**
* Journal sheeet - Actions bar. * Journal sheeet - Actions bar.
@@ -85,11 +86,18 @@ function JournalActionsBar({
icon={<Icon icon="print-16" iconSize={16} />} icon={<Icon icon="print-16" iconSize={16} />}
text={<T id={'print'} />} text={<T id={'print'} />}
/> />
<Button <Popover
className={Classes.MINIMAL} content={<JournalSheetExportMenu />}
icon={<Icon icon="file-export-16" iconSize={16} />} interactionKind={PopoverInteractionKind.CLICK}
text={<T id={'export'} />} placement="bottom-start"
/> minimal
>
<Button
className={Classes.MINIMAL}
icon={<Icon icon="file-export-16" iconSize={16} />}
text={<T id={'export'} />}
/>
</Popover>
</NavbarGroup> </NavbarGroup>
</DashboardActionsBar> </DashboardActionsBar>
); );

View File

@@ -28,6 +28,7 @@ function JournalSheetProvider({ query, ...props }) {
isLoading, isLoading,
isFetching, isFetching,
refetchSheet: refetch, refetchSheet: refetch,
httpQuery: requestQuery
}; };
return ( return (

View File

@@ -11,10 +11,10 @@ import {
TableVirtualizedListRows, TableVirtualizedListRows,
} from '@/components'; } from '@/components';
import { useJournalTableColumns } from './components';
import { useJournalSheetContext } from './JournalProvider'; import { useJournalSheetContext } from './JournalProvider';
import { defaultExpanderReducer, tableRowTypesToClassnames } from '@/utils'; import { defaultExpanderReducer, tableRowTypesToClassnames } from '@/utils';
import { useJournalSheetColumns } from './dynamicColumns';
/** /**
* Journal sheet table. * Journal sheet table.
@@ -23,12 +23,12 @@ import { defaultExpanderReducer, tableRowTypesToClassnames } from '@/utils';
export function JournalTable({ companyName }) { export function JournalTable({ companyName }) {
// Journal sheet context. // Journal sheet context.
const { const {
journalSheet: { tableRows, query }, journalSheet: { table, query },
isLoading, isLoading,
} = useJournalSheetContext(); } = useJournalSheetContext();
// Retreive the journal table columns. // Retrieves the journal table columns.
const columns = useJournalTableColumns(); const columns = useJournalSheetColumns();
// Default expanded rows of general journal table. // Default expanded rows of general journal table.
const expandedRows = useMemo(() => defaultExpanderReducer([], 1), []); const expandedRows = useMemo(() => defaultExpanderReducer([], 1), []);
@@ -39,13 +39,13 @@ export function JournalTable({ companyName }) {
sheetType={intl.get('journal_sheet')} sheetType={intl.get('journal_sheet')}
fromDate={query.from_date} fromDate={query.from_date}
toDate={query.to_date} toDate={query.to_date}
name="journal"
loading={isLoading} loading={isLoading}
fullWidth={true} fullWidth={true}
name="journal"
> >
<JournalDataTable <JournalDataTable
columns={columns} columns={columns}
data={tableRows} data={table.rows}
rowClassNames={tableRowTypesToClassnames} rowClassNames={tableRowTypesToClassnames}
noResults={intl.get( noResults={intl.get(
'this_report_does_not_contain_any_data_between_date_period', 'this_report_does_not_contain_any_data_between_date_period',

View File

@@ -1,77 +1,31 @@
// @ts-nocheck // @ts-nocheck
import React from 'react'; import React, { useRef } from 'react';
import intl from 'react-intl-universal'; import classNames from 'classnames';
import moment from 'moment'; import {
import { Button } from '@blueprintjs/core'; Button,
Classes,
Menu,
MenuItem,
ProgressBar,
Text,
Intent,
} from '@blueprintjs/core';
import { Icon, If, FormattedMessage as T } from '@/components'; import {
AppToaster,
Icon,
If,
Stack,
FormattedMessage as T,
} from '@/components';
import { useJournalSheetContext } from './JournalProvider'; import { useJournalSheetContext } from './JournalProvider';
import FinancialLoadingBar from '../FinancialLoadingBar'; import FinancialLoadingBar from '../FinancialLoadingBar';
import { FinancialComputeAlert } from '../FinancialReportPage'; import { FinancialComputeAlert } from '../FinancialReportPage';
import { Align } from '@/constants'; import {
useJournalSheetCsvExport,
/** useJournalSheetXlsxExport,
* Retrieve the journal table columns. } from '@/hooks/query';
*/
export const useJournalTableColumns = () => {
return React.useMemo(
() => [
{
Header: intl.get('date'),
accessor: (row) =>
row.date ? moment(row.date).format('YYYY MMM DD') : '',
className: 'date',
width: 100,
textOverview: true,
},
{
Header: intl.get('transaction_type'),
accessor: 'reference_type_formatted',
className: 'reference_type_formatted',
width: 120,
textOverview: true,
},
{
Header: intl.get('num'),
accessor: 'transaction_number',
className: 'reference_id',
width: 70,
textOverview: true,
},
{
Header: intl.get('description'),
accessor: 'note',
className: 'note',
textOverview: true,
},
{
Header: intl.get('acc_code'),
accessor: 'account_code',
width: 95,
className: 'account_code',
textOverview: true,
},
{
Header: intl.get('account'),
accessor: 'account_name',
className: 'account_name',
textOverview: true,
},
{
Header: intl.get('credit'),
accessor: 'formatted_credit',
align: Align.Right,
},
{
Header: intl.get('debit'),
accessor: 'formatted_debit',
align: Align.Right,
},
],
[],
);
};
/** /**
* Journal sheet loading bar. * Journal sheet loading bar.
@@ -115,3 +69,87 @@ export function JournalSheetAlerts() {
</FinancialComputeAlert> </FinancialComputeAlert>
); );
} }
/**
* Retrieves the journal sheet export menu.
* @returns {JSX.Element}
*/
export const JournalSheetExportMenu = () => {
const toastKey = useRef(null);
const commonToastConfig = {
isCloseButtonShown: true,
timeout: 2000,
};
const { httpQuery } = useJournalSheetContext();
const openProgressToast = (amount: number) => {
return (
<Stack spacing={8}>
<Text>The report has been exported successfully.</Text>
<ProgressBar
className={classNames('toast-progress', {
[Classes.PROGRESS_NO_STRIPES]: amount >= 100,
})}
intent={amount < 100 ? Intent.PRIMARY : Intent.SUCCESS}
value={amount / 100}
/>
</Stack>
);
};
// Exports the report to xlsx.
const { mutateAsync: xlsxExport } = useJournalSheetXlsxExport(httpQuery, {
onDownloadProgress: (xlsxExportProgress: number) => {
if (!toastKey.current) {
toastKey.current = AppToaster.show({
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
});
} else {
AppToaster.show(
{
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
},
toastKey.current,
);
}
},
});
// Exports the report to csv.
const { mutateAsync: csvExport } = useJournalSheetCsvExport(httpQuery, {
onDownloadProgress: (xlsxExportProgress: number) => {
if (!toastKey.current) {
toastKey.current = AppToaster.show({
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
});
} else {
AppToaster.show(
{
message: openProgressToast(xlsxExportProgress),
...commonToastConfig,
},
toastKey.current,
);
}
},
});
// Handle csv export button click.
const handleCsvExportBtnClick = () => {
csvExport();
};
// Handle xlsx export button click.
const handleXlsxExportBtnClick = () => {
xlsxExport();
};
return (
<Menu>
<MenuItem
text={'XLSX (Microsoft Excel)'}
onClick={handleXlsxExportBtnClick}
/>
<MenuItem text={'CSV'} onClick={handleCsvExportBtnClick} />
</Menu>
);
};

View File

@@ -0,0 +1,134 @@
// @ts-nocheck
import { Align } from '@/constants';
import { getColumnWidth } from '@/utils';
import * as R from 'ramda';
import { useJournalSheetContext } from './JournalProvider';
const getTableCellValueAccessor = (index) => `cells[${index}].value`;
const getReportColWidth = (data, accessor, headerText) => {
return getColumnWidth(
data,
accessor,
{ magicSpacing: 10, minWidth: 100 },
headerText,
);
};
/**
* Common column mapper.
*/
const commonAccessor = R.curry((data, column) => {
const accessor = getTableCellValueAccessor(column.cell_index);
return {
key: column.key,
Header: column.label,
accessor,
className: column.key,
textOverview: true,
align: Align.Left,
};
});
/**
* Numeric columns accessor.
*/
const numericColumnAccessor = R.curry((data, column) => {
const accessor = getTableCellValueAccessor(column.cell_index);
const width = getReportColWidth(data, accessor, column.label);
return {
...column,
align: Align.Right,
width,
};
});
/**
* Date column accessor.
*/
const dateColumnAccessor = (column) => {
return {
...column,
width: 100,
};
};
/**
* Transaction type column accessor.
*/
const transactionTypeColumnAccessor = (column) => {
return {
...column,
width: 120,
};
};
/**
* Transaction number column accessor.
*/
const transactionNumberColumnAccessor = (column) => {
return {
...column,
width: 70,
};
};
/**
* Account code column accessor.
*/
const accountCodeColumnAccessor = (column) => {
return {
...column,
width: 70,
};
};
/**
* Dynamic column mapper.
* @param {} data -
* @param {} column -
*/
const dynamicColumnMapper = R.curry((data, column) => {
const _commonAccessor = commonAccessor(data);
const _numericColumnAccessor = numericColumnAccessor(data);
return R.compose(
R.when(R.pathEq(['key'], 'date'), dateColumnAccessor),
R.when(
R.pathEq(['key'], 'transaction_type'),
transactionTypeColumnAccessor,
),
R.when(
R.pathEq(['key'], 'transaction_number'),
transactionNumberColumnAccessor,
),
R.when(R.pathEq(['key'], 'account_code'), accountCodeColumnAccessor),
R.when(R.pathEq(['key'], 'credit'), _numericColumnAccessor),
R.when(R.pathEq(['key'], 'debit'), _numericColumnAccessor),
_commonAccessor,
)(column);
});
/**
* Composes the fetched dynamic columns from the server to the columns to pass it
* to the table component.
*/
export const dynamicColumns = (columns, data) => {
return R.map(dynamicColumnMapper(data), columns);
};
/**
* Retrieves the table columns of journal sheet.
*/
export const useJournalSheetColumns = () => {
const { journalSheet } = useJournalSheetContext();
if (!journalSheet) {
throw new Error('The journal sheet is not loaded');
}
const { table } = journalSheet;
return dynamicColumns(table.columns, table.rows);
};

View File

@@ -1,6 +1,5 @@
// @ts-nocheck // @ts-nocheck
import React from 'react'; import React from 'react';
import { castArray } from 'lodash';
import moment from 'moment'; import moment from 'moment';
import { useAppQueryString } from '@/hooks'; import { useAppQueryString } from '@/hooks';

View File

@@ -1,8 +1,6 @@
// @ts-nocheck // @ts-nocheck
import { useRequestQuery } from '../useQueryRequest'; import { useRequestQuery } from '../useQueryRequest';
import { import {
generalLedgerTableRowsReducer,
journalTableRowsReducer,
inventoryValuationReducer, inventoryValuationReducer,
purchasesByItemsReducer, purchasesByItemsReducer,
salesByItemsReducer, salesByItemsReducer,
@@ -167,21 +165,44 @@ export function useGeneralLedgerSheet(query, props) {
method: 'get', method: 'get',
url: '/financial_statements/general_ledger', url: '/financial_statements/general_ledger',
params: query, params: query,
headers: {
Accept: 'application/json+table',
},
}, },
{ {
select: (res) => ({ select: (res) => res.data,
tableRows: generalLedgerTableRowsReducer(res.data.data),
...res.data,
}),
defaultData: {
tableRows: [],
data: {},
query: {},
},
...props, ...props,
}, },
); );
} }
export const useGeneralLedgerSheetXlsxExport = (query, args) => {
return useDownloadFile({
url: '/financial_statements/general_ledger',
config: {
headers: {
accept: 'application/xlsx',
},
params: query,
},
filename: 'general_ledger.xlsx',
...args,
});
};
export const useGeneralLedgerSheetCsvExport = (query, args) => {
return useDownloadFile({
url: '/financial_statements/general_ledger',
config: {
headers: {
accept: 'application/csv',
},
params: query,
},
filename: 'general_ledger.csv',
...args,
});
};
/** /**
* Retrieve journal sheet. * Retrieve journal sheet.
@@ -189,22 +210,49 @@ export function useGeneralLedgerSheet(query, props) {
export function useJournalSheet(query, props) { export function useJournalSheet(query, props) {
return useRequestQuery( return useRequestQuery(
[t.FINANCIAL_REPORT, t.JOURNAL, query], [t.FINANCIAL_REPORT, t.JOURNAL, query],
{ method: 'get', url: '/financial_statements/journal', params: query },
{ {
select: (res) => ({ method: 'get',
tableRows: journalTableRowsReducer(res.data.data), url: '/financial_statements/journal',
...res.data, params: query,
}), headers: {
defaultData: { Accept: 'application/json+table',
data: {},
tableRows: [],
query: {},
}, },
},
{
select: (res) => res.data,
...props, ...props,
}, },
); );
} }
export const useJournalSheetXlsxExport = (query, args) => {
return useDownloadFile({
url: '/financial_statements/journal',
config: {
headers: {
accept: 'application/xlsx',
},
params: query,
},
filename: 'journal.xlsx',
...args,
});
};
export const useJournalSheetCsvExport = (query, args) => {
return useDownloadFile({
url: '/financial_statements/journal',
config: {
headers: {
accept: 'application/csv',
},
params: query,
},
filename: 'journal.csv',
...args,
});
};
/** /**
* Retrieve A/R aging summary report. * Retrieve A/R aging summary report.
*/ */