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