2
0
mirror of https://github.com/frappe/books.git synced 2024-11-14 17:34:04 +00:00
books/models/baseModels/Party/Party.ts
18alantom 7c03b98d67 fix: from account not found
- add a bunch of Payment validations
- add inline form error message display
2022-10-08 00:11:50 +05:30

191 lines
5.0 KiB
TypeScript

import { Fyo } from 'fyo';
import { Doc } from 'fyo/model/doc';
import {
Action,
FiltersMap,
FormulaMap,
ListViewSettings,
ValidationMap,
} from 'fyo/model/types';
import {
validateEmail,
validatePhoneNumber,
} from 'fyo/model/validationFunction';
import { Money } from 'pesa';
import { PartyRole } from './types';
export class Party extends Doc {
role?: PartyRole;
defaultAccount?: string;
outstandingAmount?: Money;
async updateOutstandingAmount() {
/**
* If Role === "Both" then outstanding Amount
* will be the amount to be paid to the party.
*/
const role = this.role as PartyRole;
let outstandingAmount = this.fyo.pesa(0);
if (role === 'Customer' || role === 'Both') {
const outstandingReceive = await this._getTotalOutstandingAmount(
'SalesInvoice'
);
outstandingAmount = outstandingAmount.add(outstandingReceive);
}
if (role === 'Supplier') {
const outstandingPay = await this._getTotalOutstandingAmount(
'PurchaseInvoice'
);
outstandingAmount = outstandingAmount.add(outstandingPay);
}
if (role === 'Both') {
const outstandingPay = await this._getTotalOutstandingAmount(
'PurchaseInvoice'
);
outstandingAmount = outstandingAmount.sub(outstandingPay);
}
await this.setAndSync({ outstandingAmount });
}
async _getTotalOutstandingAmount(
schemaName: 'SalesInvoice' | 'PurchaseInvoice'
) {
const outstandingAmounts = await this.fyo.db.getAllRaw(schemaName, {
fields: ['outstandingAmount'],
filters: {
submitted: true,
cancelled: false,
party: this.name as string,
},
});
return outstandingAmounts
.map(({ outstandingAmount }) =>
this.fyo.pesa(outstandingAmount as number)
)
.reduce((a, b) => a.add(b), this.fyo.pesa(0));
}
formulas: FormulaMap = {
defaultAccount: {
formula: async () => {
const role = this.role as PartyRole;
if (role === 'Both') {
return '';
}
let accountName = 'Debtors';
if (role === 'Supplier') {
accountName = 'Creditors';
}
const accountExists = await this.fyo.db.exists('Account', accountName);
return accountExists ? accountName : '';
},
dependsOn: ['role'],
},
currency: {
formula: async () => {
if (!this.currency) {
return this.fyo.singles.SystemSettings!.currency as string;
}
},
},
};
validations: ValidationMap = {
email: validateEmail,
phone: validatePhoneNumber,
};
static filters: FiltersMap = {
defaultAccount: (doc: Doc) => {
const role = doc.role as PartyRole;
if (role === 'Both') {
return {
isGroup: false,
accountType: ['in', ['Payable', 'Receivable']],
};
}
return {
isGroup: false,
accountType: role === 'Customer' ? 'Receivable' : 'Payable',
};
},
};
static getListViewSettings(): ListViewSettings {
return {
columns: ['name', 'email', 'phone', 'outstandingAmount'],
};
}
static getActions(fyo: Fyo): Action[] {
return [
{
label: fyo.t`Create Purchase`,
condition: (doc: Doc) =>
!doc.notInserted && (doc.role as PartyRole) !== 'Customer',
action: async (partyDoc, router) => {
const doc = await fyo.doc.getNewDoc('PurchaseInvoice', {
party: partyDoc.name,
account: partyDoc.defaultAccount as string,
});
router.push({
path: `/edit/PurchaseInvoice/${doc.name}`,
query: {
schemaName: 'PurchaseInvoice',
values: {
// @ts-ignore
party: partyDoc.name!,
},
},
});
},
},
{
label: fyo.t`View Purchases`,
condition: (doc: Doc) =>
!doc.notInserted && (doc.role as PartyRole) !== 'Customer',
action: async (partyDoc, router) => {
router.push(`/list/PurchaseInvoice/party/${partyDoc.name}`);
},
},
{
label: fyo.t`Create Sale`,
condition: (doc: Doc) =>
!doc.notInserted && (doc.role as PartyRole) !== 'Supplier',
action: async (partyDoc, router) => {
const doc = await fyo.doc.getNewDoc('SalesInvoice', {
party: partyDoc.name,
account: partyDoc.defaultAccount as string,
});
router.push({
path: `/edit/SalesInvoice/${doc.name}`,
query: {
schemaName: 'SalesInvoice',
values: {
// @ts-ignore
party: partyDoc.name!,
},
},
});
},
},
{
label: fyo.t`View Sales`,
condition: (doc: Doc) =>
!doc.notInserted && (doc.role as PartyRole) !== 'Supplier',
action: async (partyDoc, router) => {
router.push(`/list/SalesInvoice/party/${partyDoc.name}`);
},
},
];
}
}