2022-05-16 13:01:58 +00:00
|
|
|
import { t } from 'fyo';
|
|
|
|
import { ValueError } from 'fyo/utils/errors';
|
|
|
|
import { DateTime } from 'luxon';
|
|
|
|
import {
|
|
|
|
AccountRootType,
|
|
|
|
AccountRootTypeEnum,
|
|
|
|
} from 'models/baseModels/Account/types';
|
|
|
|
import {
|
|
|
|
AccountReport,
|
|
|
|
ACC_BAL_WIDTH,
|
|
|
|
ACC_NAME_WIDTH,
|
|
|
|
convertAccountRootNodeToAccountList,
|
|
|
|
getFiscalEndpoints,
|
|
|
|
} from 'reports/AccountReport';
|
|
|
|
import {
|
|
|
|
Account,
|
|
|
|
AccountListNode,
|
|
|
|
AccountNameValueMapMap,
|
|
|
|
ColumnField,
|
|
|
|
DateRange,
|
|
|
|
GroupedMap,
|
|
|
|
LedgerEntry,
|
|
|
|
ReportCell,
|
|
|
|
ReportData,
|
|
|
|
ReportRow,
|
|
|
|
RootTypeRow,
|
|
|
|
ValueMap,
|
|
|
|
} from 'reports/types';
|
|
|
|
import { Field } from 'schemas/types';
|
|
|
|
import { QueryFilter } from 'utils/db/types';
|
|
|
|
|
|
|
|
export class TrialBalance extends AccountReport {
|
|
|
|
static title = t`Trial Balance`;
|
|
|
|
static reportName = 'trial-balance';
|
|
|
|
|
|
|
|
fromDate?: string;
|
|
|
|
toDate?: string;
|
2023-06-22 08:52:54 +00:00
|
|
|
hideGroupAmounts = false;
|
|
|
|
loading = false;
|
2022-05-16 13:01:58 +00:00
|
|
|
|
|
|
|
_rawData: LedgerEntry[] = [];
|
|
|
|
_dateRanges?: DateRange[];
|
|
|
|
|
|
|
|
accountMap?: Record<string, Account>;
|
|
|
|
|
|
|
|
get rootTypes(): AccountRootType[] {
|
|
|
|
return [
|
|
|
|
AccountRootTypeEnum.Asset,
|
|
|
|
AccountRootTypeEnum.Liability,
|
|
|
|
AccountRootTypeEnum.Income,
|
|
|
|
AccountRootTypeEnum.Expense,
|
|
|
|
AccountRootTypeEnum.Equity,
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2022-05-30 11:34:25 +00:00
|
|
|
async setReportData(filter?: string, force?: boolean) {
|
2022-05-18 14:58:35 +00:00
|
|
|
this.loading = true;
|
2022-05-30 11:34:25 +00:00
|
|
|
if (force || filter !== 'hideGroupAmounts') {
|
2022-05-16 13:01:58 +00:00
|
|
|
await this._setRawData();
|
|
|
|
}
|
|
|
|
|
|
|
|
const map = this._getGroupedMap(true, 'account');
|
|
|
|
const rangeGroupedMap = await this._getGroupedByDateRanges(map);
|
|
|
|
const accountTree = await this._getAccountTree(rangeGroupedMap);
|
|
|
|
|
|
|
|
const rootTypeRows: RootTypeRow[] = this.rootTypes
|
|
|
|
.map((rootType) => {
|
|
|
|
const rootNode = this.getRootNode(rootType, accountTree)!;
|
|
|
|
const rootList = convertAccountRootNodeToAccountList(rootNode);
|
|
|
|
return {
|
|
|
|
rootType,
|
|
|
|
rootNode,
|
|
|
|
rows: this.getReportRowsFromAccountList(rootList),
|
|
|
|
};
|
|
|
|
})
|
|
|
|
.filter((row) => !!row.rootNode);
|
|
|
|
|
|
|
|
this.reportData = await this.getReportDataFromRows(rootTypeRows);
|
2022-05-18 14:58:35 +00:00
|
|
|
this.loading = false;
|
2022-05-16 13:01:58 +00:00
|
|
|
}
|
|
|
|
|
2023-06-22 08:52:54 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/require-await
|
2022-05-16 13:01:58 +00:00
|
|
|
async getReportDataFromRows(
|
|
|
|
rootTypeRows: RootTypeRow[]
|
|
|
|
): Promise<ReportData> {
|
|
|
|
const reportData = rootTypeRows.reduce((reportData, r) => {
|
|
|
|
reportData.push(...r.rows);
|
|
|
|
reportData.push(this.getEmptyRow());
|
|
|
|
return reportData;
|
|
|
|
}, [] as ReportData);
|
|
|
|
|
|
|
|
reportData.pop();
|
|
|
|
|
|
|
|
return reportData;
|
|
|
|
}
|
|
|
|
|
2023-06-22 08:52:54 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/require-await
|
2022-05-16 13:01:58 +00:00
|
|
|
async _getGroupedByDateRanges(
|
|
|
|
map: GroupedMap
|
|
|
|
): Promise<AccountNameValueMapMap> {
|
|
|
|
const accountValueMap: AccountNameValueMapMap = new Map();
|
|
|
|
|
|
|
|
for (const account of map.keys()) {
|
|
|
|
const valueMap: ValueMap = new Map();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set Balance for every DateRange key
|
|
|
|
*/
|
|
|
|
for (const entry of map.get(account)!) {
|
|
|
|
const key = this._getRangeMapKey(entry);
|
|
|
|
if (key === null) {
|
|
|
|
throw new ValueError(
|
2023-06-22 08:52:54 +00:00
|
|
|
`invalid entry in trial balance ${entry.date?.toISOString() ?? ''}`
|
2022-05-16 13:01:58 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-06-22 08:52:54 +00:00
|
|
|
const map = valueMap.get(key);
|
2022-05-16 13:01:58 +00:00
|
|
|
const totalCredit = map?.credit ?? 0;
|
|
|
|
const totalDebit = map?.debit ?? 0;
|
|
|
|
|
|
|
|
valueMap.set(key, {
|
|
|
|
credit: totalCredit + (entry.credit ?? 0),
|
|
|
|
debit: totalDebit + (entry.debit ?? 0),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
accountValueMap.set(account, valueMap);
|
|
|
|
}
|
|
|
|
|
|
|
|
return accountValueMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
async _getDateRanges(): Promise<DateRange[]> {
|
|
|
|
if (!this.toDate || !this.fromDate) {
|
|
|
|
await this.setDefaultFilters();
|
|
|
|
}
|
|
|
|
|
|
|
|
const toDate = DateTime.fromISO(this.toDate!);
|
|
|
|
const fromDate = DateTime.fromISO(this.fromDate!);
|
|
|
|
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
fromDate: DateTime.fromISO('0001-01-01'),
|
|
|
|
toDate: fromDate,
|
|
|
|
},
|
|
|
|
{ fromDate, toDate },
|
|
|
|
{
|
|
|
|
fromDate: toDate,
|
|
|
|
toDate: DateTime.fromISO('9999-12-31'),
|
|
|
|
},
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
getRowFromAccountListNode(al: AccountListNode) {
|
|
|
|
const nameCell = {
|
|
|
|
value: al.name,
|
|
|
|
rawValue: al.name,
|
|
|
|
align: 'left',
|
|
|
|
width: ACC_NAME_WIDTH,
|
|
|
|
bold: !al.level,
|
|
|
|
indent: al.level ?? 0,
|
|
|
|
} as ReportCell;
|
|
|
|
|
|
|
|
const balanceCells = this._dateRanges!.map((k) => {
|
|
|
|
const map = al.valueMap?.get(k);
|
|
|
|
const hide = this.hideGroupAmounts && al.isGroup;
|
|
|
|
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
rawValue: map?.debit ?? 0,
|
|
|
|
value: hide ? '' : this.fyo.format(map?.debit ?? 0, 'Currency'),
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
rawValue: map?.credit ?? 0,
|
|
|
|
value: hide ? '' : this.fyo.format(map?.credit ?? 0, 'Currency'),
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
} as ReportCell,
|
|
|
|
];
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
cells: [nameCell, balanceCells].flat(2),
|
|
|
|
level: al.level,
|
|
|
|
isGroup: !!al.isGroup,
|
|
|
|
folded: false,
|
|
|
|
foldedBelow: false,
|
|
|
|
} as ReportRow;
|
|
|
|
}
|
|
|
|
|
2023-06-22 08:52:54 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/require-await
|
2022-05-16 13:01:58 +00:00
|
|
|
async _getQueryFilters(): Promise<QueryFilter> {
|
|
|
|
const filters: QueryFilter = {};
|
|
|
|
filters.reverted = false;
|
|
|
|
return filters;
|
|
|
|
}
|
|
|
|
|
|
|
|
async setDefaultFilters(): Promise<void> {
|
|
|
|
if (!this.toDate || !this.fromDate) {
|
|
|
|
const { year } = DateTime.now();
|
2022-12-23 07:47:02 +00:00
|
|
|
const endpoints = await getFiscalEndpoints(year + 1, year, this.fyo);
|
2022-05-16 13:01:58 +00:00
|
|
|
|
|
|
|
this.fromDate = endpoints.fromDate;
|
|
|
|
this.toDate = DateTime.fromISO(endpoints.toDate)
|
|
|
|
.minus({ days: 1 })
|
|
|
|
.toISODate();
|
|
|
|
}
|
|
|
|
|
|
|
|
await this._setDateRanges();
|
|
|
|
}
|
|
|
|
|
|
|
|
getFilters(): Field[] {
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
fieldtype: 'Date',
|
|
|
|
fieldname: 'fromDate',
|
|
|
|
placeholder: t`From Date`,
|
|
|
|
label: t`From Date`,
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldtype: 'Date',
|
|
|
|
fieldname: 'toDate',
|
|
|
|
placeholder: t`To Date`,
|
|
|
|
label: t`To Date`,
|
|
|
|
required: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fieldtype: 'Check',
|
|
|
|
label: t`Hide Group Amounts`,
|
|
|
|
fieldname: 'hideGroupAmounts',
|
|
|
|
} as Field,
|
|
|
|
] as Field[];
|
|
|
|
}
|
|
|
|
|
|
|
|
getColumns(): ColumnField[] {
|
|
|
|
return [
|
|
|
|
{
|
|
|
|
label: t`Account`,
|
|
|
|
fieldtype: 'Link',
|
|
|
|
fieldname: 'account',
|
|
|
|
align: 'left',
|
|
|
|
width: ACC_NAME_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: t`Opening (Dr)`,
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'openingDebit',
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: t`Opening (Cr)`,
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'openingCredit',
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: t`Debit`,
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'debit',
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: t`Credit`,
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'credit',
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: t`Closing (Dr)`,
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'closingDebit',
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
label: t`Closing (Cr)`,
|
|
|
|
fieldtype: 'Data',
|
|
|
|
fieldname: 'closingCredit',
|
|
|
|
align: 'right',
|
|
|
|
width: ACC_BAL_WIDTH,
|
|
|
|
},
|
|
|
|
] as ColumnField[];
|
|
|
|
}
|
|
|
|
}
|