2022-04-28 18:07:07 +05:30
|
|
|
import { Fyo, t } from 'fyo';
|
2022-04-24 12:18:44 +05:30
|
|
|
import { Doc } from 'fyo/model/doc';
|
2022-06-14 14:40:46 +05:30
|
|
|
import { Action, ColumnConfig, DocStatus, RenderData } from 'fyo/model/types';
|
2022-04-19 11:35:39 +05:30
|
|
|
import { NotFoundError } from 'fyo/utils/errors';
|
|
|
|
import { DateTime } from 'luxon';
|
2022-05-23 11:00:54 +05:30
|
|
|
import { Money } from 'pesa';
|
2022-04-14 10:54:11 +05:30
|
|
|
import { Router } from 'vue-router';
|
2022-05-14 14:16:05 +05:30
|
|
|
import {
|
|
|
|
AccountRootType,
|
|
|
|
AccountRootTypeEnum,
|
|
|
|
} from './baseModels/Account/types';
|
2022-05-02 20:47:14 +05:30
|
|
|
import { InvoiceStatus, ModelNameEnum } from './types';
|
2022-04-14 10:54:11 +05:30
|
|
|
|
2022-05-02 20:47:14 +05:30
|
|
|
export function getInvoiceActions(
|
|
|
|
schemaName: ModelNameEnum.PurchaseInvoice | ModelNameEnum.SalesInvoice,
|
|
|
|
fyo: Fyo
|
|
|
|
): Action[] {
|
2022-04-14 13:31:33 +05:30
|
|
|
return [
|
|
|
|
{
|
2022-04-19 11:29:36 +05:30
|
|
|
label: fyo.t`Make Payment`,
|
2022-04-14 13:31:33 +05:30
|
|
|
condition: (doc: Doc) =>
|
2022-05-03 18:43:47 +05:30
|
|
|
doc.isSubmitted && !(doc.outstandingAmount as Money).isZero(),
|
2022-04-14 13:31:33 +05:30
|
|
|
action: async function makePayment(doc: Doc) {
|
2022-05-11 16:14:31 +05:30
|
|
|
const payment = fyo.doc.getNewDoc('Payment');
|
2022-04-26 15:42:33 +05:30
|
|
|
payment.once('afterSync', async () => {
|
2022-04-14 13:31:33 +05:30
|
|
|
await payment.submit();
|
|
|
|
});
|
2022-05-03 18:43:47 +05:30
|
|
|
|
2022-04-14 13:31:33 +05:30
|
|
|
const isSales = schemaName === 'SalesInvoice';
|
2022-05-03 18:43:47 +05:30
|
|
|
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,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
},
|
2022-04-19 11:29:36 +05:30
|
|
|
getLedgerLinkAction(fyo),
|
2022-04-14 13:31:33 +05:30
|
|
|
];
|
|
|
|
}
|
|
|
|
|
2022-05-03 18:43:47 +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({
|
2022-05-03 18:43:47 +05:30
|
|
|
referenceType: doc.schemaName,
|
|
|
|
referenceName: doc.name,
|
2022-05-27 12:45:07 +05:30
|
|
|
}),
|
2022-05-03 18:43:47 +05:30
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-04-28 18:07: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 {
|
2022-04-28 18:07:07 +05:30
|
|
|
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
|
|
|
}
|
2022-04-19 11:35:39 +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;
|
|
|
|
}
|
2022-05-14 14:16:05 +05:30
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|