2
0
mirror of https://github.com/frappe/books.git synced 2025-01-22 22:58:28 +00:00
books/models/helpers.ts

258 lines
5.9 KiB
TypeScript
Raw Normal View History

import { Fyo, t } from 'fyo';
import { Doc } from 'fyo/model/doc';
2022-06-14 14:40:46 +05:30
import { Action, ColumnConfig, DocStatus, RenderData } from 'fyo/model/types';
import { NotFoundError } from 'fyo/utils/errors';
import { DateTime } from 'luxon';
2022-05-23 11:00:54 +05:30
import { Money } from 'pesa';
import { Router } from 'vue-router';
import {
AccountRootType,
AccountRootTypeEnum,
} from './baseModels/Account/types';
import { InvoiceStatus, ModelNameEnum } from './types';
export function getInvoiceActions(
schemaName: ModelNameEnum.PurchaseInvoice | ModelNameEnum.SalesInvoice,
fyo: Fyo
): Action[] {
2022-04-14 13:31:33 +05:30
return [
{
label: fyo.t`Make Payment`,
2022-04-14 13:31:33 +05:30
condition: (doc: Doc) =>
doc.isSubmitted && !(doc.outstandingAmount as Money).isZero(),
2022-04-14 13:31:33 +05:30
action: async function makePayment(doc: Doc) {
const payment = fyo.doc.getNewDoc('Payment');
payment.once('afterSync', async () => {
2022-04-14 13:31:33 +05:30
await payment.submit();
});
2022-04-14 13:31:33 +05:30
const isSales = schemaName === 'SalesInvoice';
const party = doc.party as string;
2022-04-14 13:31:33 +05:30
const paymentType = isSales ? 'Receive' : 'Pay';
const hideAccountField = isSales ? 'account' : 'paymentAccount';
2022-04-20 12:08:47 +05:30
const { openQuickEdit } = await import('src/utils/ui');
2022-04-14 13:31:33 +05:30
await openQuickEdit({
schemaName: 'Payment',
name: payment.name as string,
hideFields: ['party', 'date', hideAccountField, 'paymentType', 'for'],
defaults: {
party,
[hideAccountField]: doc.account,
date: new Date().toISOString().slice(0, 10),
paymentType,
for: [
{
2022-05-01 10:15:47 +05:30
referenceType: doc.schemaName,
2022-04-14 13:31:33 +05:30
referenceName: doc.name,
amount: doc.outstandingAmount,
},
],
},
});
},
},
getLedgerLinkAction(fyo),
2022-04-14 13:31:33 +05:30
];
}
export function getLedgerLinkAction(fyo: Fyo): Action {
return {
label: fyo.t`Ledger Entries`,
condition: (doc: Doc) => doc.isSubmitted,
action: async (doc: Doc, router: Router) => {
router.push({
name: 'Report',
params: {
2022-05-27 12:45:07 +05:30
reportClassName: 'GeneralLedger',
defaultFilters: JSON.stringify({
referenceType: doc.schemaName,
referenceName: doc.name,
2022-05-27 12:45:07 +05:30
}),
},
});
},
};
}
export function getTransactionStatusColumn(): ColumnConfig {
2022-06-14 14:40:46 +05:30
const statusMap = getStatusMap();
2022-04-14 13:31:33 +05:30
return {
label: t`Status`,
2022-04-14 13:31:33 +05:30
fieldname: 'status',
fieldtype: 'Select',
2022-06-14 14:40:46 +05:30
render(doc) {
const status = getDocStatus(doc) as InvoiceStatus;
2022-04-14 13:31:33 +05:30
const color = statusColor[status];
const label = statusMap[status];
2022-06-14 14:40:46 +05:30
2022-04-14 13:31:33 +05:30
return {
template: `<Badge class="text-xs" color="${color}">${label}</Badge>`,
};
},
};
}
2022-06-14 14:40:46 +05:30
export const statusColor: Record<
DocStatus | InvoiceStatus,
string | undefined
> = {
'': 'gray',
2022-04-14 13:31:33 +05:30
Draft: 'gray',
Unpaid: 'orange',
Paid: 'green',
2022-06-14 14:40:46 +05:30
Saved: 'gray',
NotSaved: 'gray',
Submitted: 'green',
2022-04-14 13:31:33 +05:30
Cancelled: 'red',
};
2022-06-14 14:40:46 +05:30
export function getStatusMap(): Record<DocStatus | InvoiceStatus, string> {
return {
'': '',
Draft: t`Draft`,
Unpaid: t`Unpaid`,
Paid: t`Paid`,
Saved: t`Saved`,
NotSaved: t`Not Saved`,
Submitted: t`Submitted`,
Cancelled: t`Cancelled`,
};
}
export function getDocStatus(
doc?: RenderData | Doc
): DocStatus | InvoiceStatus {
if (!doc) {
return '';
}
if (doc.notInserted) {
return 'Draft';
2022-04-14 13:31:33 +05:30
}
2022-06-14 14:40:46 +05:30
if (doc.dirty) {
return 'NotSaved';
}
if (!doc.schema?.isSubmittable) {
return 'Saved';
}
return getSubmittableDocStatus(doc);
}
function getSubmittableDocStatus(doc: RenderData | Doc) {
if (
[ModelNameEnum.SalesInvoice, ModelNameEnum.PurchaseInvoice].includes(
doc.schema.name as ModelNameEnum
)
) {
return getInvoiceStatus(doc);
}
if (!!doc.submitted && !doc.cancelled) {
return 'Submitted';
}
if (!!doc.submitted && !!doc.cancelled) {
return 'Cancelled';
}
return 'Saved';
}
export function getInvoiceStatus(doc: RenderData | Doc): InvoiceStatus {
if (
doc.submitted &&
!doc.cancelled &&
(doc.outstandingAmount as Money).isZero()
) {
return 'Paid';
}
if (
doc.submitted &&
!doc.cancelled &&
(doc.outstandingAmount as Money).isPositive()
) {
return 'Unpaid';
2022-04-14 13:31:33 +05:30
}
if (doc.cancelled) {
2022-06-14 14:40:46 +05:30
return 'Cancelled';
2022-04-14 13:31:33 +05:30
}
2022-06-14 14:40:46 +05:30
return 'Saved';
2022-04-14 13:31:33 +05:30
}
export async function getExchangeRate({
fromCurrency,
toCurrency,
date,
}: {
fromCurrency: string;
toCurrency: string;
date?: string;
}) {
if (!date) {
date = DateTime.local().toISODate();
}
if (!fromCurrency || !toCurrency) {
throw new NotFoundError(
'Please provide `fromCurrency` and `toCurrency` to get exchange rate.'
);
}
const cacheKey = `currencyExchangeRate:${date}:${fromCurrency}:${toCurrency}`;
let exchangeRate = 0;
if (localStorage) {
exchangeRate = parseFloat(
localStorage.getItem(cacheKey as string) as string
);
}
if (!exchangeRate && fetch) {
try {
const res = await fetch(
` https://api.vatcomply.com/rates?date=${date}&base=${fromCurrency}&symbols=${toCurrency}`
);
const data = await res.json();
exchangeRate = data.rates[toCurrency];
if (localStorage) {
localStorage.setItem(cacheKey, String(exchangeRate));
}
} catch (error) {
console.error(error);
throw new Error(
`Could not fetch exchange rate for ${fromCurrency} -> ${toCurrency}`
);
}
} else {
exchangeRate = 1;
}
return exchangeRate;
}
export function isCredit(rootType: AccountRootType) {
switch (rootType) {
case AccountRootTypeEnum.Asset:
return false;
case AccountRootTypeEnum.Liability:
return true;
case AccountRootTypeEnum.Equity:
return true;
case AccountRootTypeEnum.Expense:
return false;
case AccountRootTypeEnum.Income:
return true;
default:
return true;
}
}