2
0
mirror of https://github.com/frappe/books.git synced 2025-01-03 15:17:30 +00:00
books/reports/ProfitAndLoss/ProfitAndLoss.ts

191 lines
4.5 KiB
TypeScript
Raw Permalink Normal View History

2022-05-15 12:13:31 +00:00
import { t } from 'fyo';
import {
AccountRootType,
2022-05-16 10:10:35 +00:00
AccountRootTypeEnum,
2022-05-15 12:13:31 +00:00
} from 'models/baseModels/Account/types';
import {
2022-05-16 10:10:35 +00:00
AccountReport,
convertAccountRootNodesToAccountList,
2022-05-16 10:10:35 +00:00
} from 'reports/AccountReport';
2022-05-16 13:01:58 +00:00
import {
AccountListNode,
AccountTreeNode,
ReportData,
ValueMap,
} from 'reports/types';
2022-05-16 10:10:35 +00:00
export class ProfitAndLoss extends AccountReport {
static title = t`Profit And Loss`;
static reportName = 'profit-and-loss';
loading = false;
2022-05-16 10:10:35 +00:00
get rootTypes(): AccountRootType[] {
return [AccountRootTypeEnum.Income, AccountRootTypeEnum.Expense];
2022-05-16 07:49:01 +00:00
}
async setReportData(filter?: string, force?: boolean) {
2022-05-18 14:58:35 +00:00
this.loading = true;
if (force || filter !== 'hideGroupAmounts') {
2022-05-16 10:10:35 +00:00
await this._setRawData();
}
2022-05-16 07:49:01 +00:00
const map = this._getGroupedMap(true, 'account');
const rangeGroupedMap = await this._getGroupedByDateRanges(map);
2022-05-15 12:13:31 +00:00
const accountTree = await this._getAccountTree(rangeGroupedMap);
for (const name of Object.keys(accountTree)) {
const { rootType } = accountTree[name];
2022-05-16 10:10:35 +00:00
if (this.rootTypes.includes(rootType)) {
2022-05-15 12:13:31 +00:00
continue;
}
delete accountTree[name];
}
2022-05-16 10:10:35 +00:00
/**
* Income Rows
*/
const incomeRoots = this.getRootNodes(
2022-05-16 10:10:35 +00:00
AccountRootTypeEnum.Income,
accountTree
)!;
const incomeList = convertAccountRootNodesToAccountList(incomeRoots);
2022-05-16 10:10:35 +00:00
const incomeRows = this.getReportRowsFromAccountList(incomeList);
/**
* Expense Rows
*/
const expenseRoots = this.getRootNodes(
2022-05-16 10:10:35 +00:00
AccountRootTypeEnum.Expense,
accountTree
)!;
const expenseList = convertAccountRootNodesToAccountList(expenseRoots);
2022-05-16 10:10:35 +00:00
const expenseRows = this.getReportRowsFromAccountList(expenseList);
this.reportData = this.getReportDataFromRows(
incomeRows,
expenseRows,
incomeRoots,
expenseRoots
);
2022-05-18 14:58:35 +00:00
this.loading = false;
}
getReportDataFromRows(
incomeRows: ReportData,
expenseRows: ReportData,
incomeRoots: AccountTreeNode[] | undefined,
expenseRoots: AccountTreeNode[] | undefined
): ReportData {
if (
incomeRoots &&
incomeRoots.length &&
!expenseRoots &&
!expenseRoots.length
) {
return this.getIncomeOrExpenseRows(
incomeRoots,
incomeRows,
t`Total Income (Credit)`
);
}
if (
expenseRoots &&
expenseRoots.length &&
(!incomeRoots || !incomeRoots.length)
) {
return this.getIncomeOrExpenseRows(
expenseRoots,
expenseRows,
t`Total Income (Credit)`
);
}
if (
!incomeRoots ||
!incomeRoots.length ||
!expenseRoots ||
!expenseRoots.length
) {
return [];
}
return this.getIncomeAndExpenseRows(
incomeRows,
expenseRows,
incomeRoots,
expenseRoots
);
}
getIncomeOrExpenseRows(
roots: AccountTreeNode[],
rows: ReportData,
totalRowName: string
): ReportData {
const total = this.getTotalNode(roots, totalRowName);
const totalRow = this.getRowFromAccountListNode(total);
return [rows, totalRow].flat();
}
getIncomeAndExpenseRows(
incomeRows: ReportData,
expenseRows: ReportData,
incomeRoots: AccountTreeNode[],
expenseRoots: AccountTreeNode[]
) {
const totalIncome = this.getTotalNode(
incomeRoots,
t`Total Income (Credit)`
);
const totalIncomeRow = this.getRowFromAccountListNode(totalIncome);
const totalExpense = this.getTotalNode(
expenseRoots,
t`Total Expense (Debit)`
);
const totalExpenseRow = this.getRowFromAccountListNode(totalExpense);
2022-05-16 13:01:58 +00:00
const totalValueMap: ValueMap = new Map();
for (const key of totalIncome.valueMap!.keys()) {
2022-05-16 13:01:58 +00:00
const income = totalIncome.valueMap!.get(key)?.balance ?? 0;
const expense = totalExpense.valueMap!.get(key)?.balance ?? 0;
totalValueMap.set(key, { balance: income - expense });
}
const totalProfit = {
name: t`Total Profit`,
valueMap: totalValueMap,
level: 0,
} as AccountListNode;
2022-05-16 07:49:01 +00:00
const totalProfitRow = this.getRowFromAccountListNode(totalProfit);
totalProfitRow.cells.forEach((c) => {
c.bold = true;
if (typeof c.rawValue !== 'number') {
return;
}
if (c.rawValue > 0) {
c.color = 'green';
} else if (c.rawValue < 0) {
c.color = 'red';
}
});
2022-05-16 10:10:35 +00:00
const emptyRow = this.getEmptyRow();
return [
incomeRows,
totalIncomeRow,
emptyRow,
expenseRows,
totalExpenseRow,
emptyRow,
totalProfitRow,
].flat() as ReportData;
}
}