mirror of
https://github.com/bigcapitalhq/bigcapital.git
synced 2026-02-20 14:50:32 +00:00
Compare commits
9 Commits
advanced-p
...
v0.18.10
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4cd0405078 | ||
|
|
783102449f | ||
|
|
ae617b2e1d | ||
|
|
53f37f4f48 | ||
|
|
0a7b522b87 | ||
|
|
9e6500ac79 | ||
|
|
b93cb546f4 | ||
|
|
6d17f9cbeb | ||
|
|
6b6b73b77c |
@@ -126,8 +126,6 @@ export default class BillsPayments extends BaseController {
|
|||||||
|
|
||||||
check('attachments').isArray().optional(),
|
check('attachments').isArray().optional(),
|
||||||
check('attachments.*.key').exists().isString(),
|
check('attachments.*.key').exists().isString(),
|
||||||
|
|
||||||
check('prepard_expenses_account_id').optional().isNumeric().toInt(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -150,9 +150,8 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
check('customer_id').exists().isNumeric().toInt(),
|
check('customer_id').exists().isNumeric().toInt(),
|
||||||
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
check('exchange_rate').optional().isFloat({ gt: 0 }).toFloat(),
|
||||||
|
|
||||||
check('payment_date').exists(),
|
|
||||||
check('amount').exists().isNumeric().toFloat(),
|
check('amount').exists().isNumeric().toFloat(),
|
||||||
|
check('payment_date').exists(),
|
||||||
check('reference_no').optional(),
|
check('reference_no').optional(),
|
||||||
check('deposit_account_id').exists().isNumeric().toInt(),
|
check('deposit_account_id').exists().isNumeric().toInt(),
|
||||||
check('payment_receive_no').optional({ nullable: true }).trim().escape(),
|
check('payment_receive_no').optional({ nullable: true }).trim().escape(),
|
||||||
@@ -160,8 +159,7 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
|
|
||||||
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
check('branch_id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
|
|
||||||
check('entries').isArray(),
|
check('entries').isArray({}),
|
||||||
|
|
||||||
check('entries.*.id').optional({ nullable: true }).isNumeric().toInt(),
|
check('entries.*.id').optional({ nullable: true }).isNumeric().toInt(),
|
||||||
check('entries.*.index').optional().isNumeric().toInt(),
|
check('entries.*.index').optional().isNumeric().toInt(),
|
||||||
check('entries.*.invoice_id').exists().isNumeric().toInt(),
|
check('entries.*.invoice_id').exists().isNumeric().toInt(),
|
||||||
@@ -169,11 +167,6 @@ export default class PaymentReceivesController extends BaseController {
|
|||||||
|
|
||||||
check('attachments').isArray().optional(),
|
check('attachments').isArray().optional(),
|
||||||
check('attachments.*.key').exists().isString(),
|
check('attachments.*.key').exists().isString(),
|
||||||
|
|
||||||
check('unearned_revenue_account_id')
|
|
||||||
.optional({ nullable: true })
|
|
||||||
.isNumeric()
|
|
||||||
.toInt(),
|
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -237,4 +237,8 @@ module.exports = {
|
|||||||
endpoint: process.env.S3_ENDPOINT,
|
endpoint: process.env.S3_ENDPOINT,
|
||||||
bucket: process.env.S3_BUCKET || 'bigcapital-documents',
|
bucket: process.env.S3_BUCKET || 'bigcapital-documents',
|
||||||
},
|
},
|
||||||
|
|
||||||
|
loops: {
|
||||||
|
apiKey: process.env.LOOPS_API_KEY,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,17 +0,0 @@
|
|||||||
exports.up = function (knex) {
|
|
||||||
return knex.schema.table('payment_receives', (table) => {
|
|
||||||
table.decimal('applied_amount', 13, 3).defaultTo(0);
|
|
||||||
table
|
|
||||||
.integer('unearned_revenue_account_id')
|
|
||||||
.unsigned()
|
|
||||||
.references('id')
|
|
||||||
.inTable('accounts');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.down = function (knex) {
|
|
||||||
return knex.schema.table('payment_receives', (table) => {
|
|
||||||
table.dropColumn('applied_amount');
|
|
||||||
table.dropColumn('unearned_revenue_account_id');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
exports.up = function (knex) {
|
|
||||||
return knex.schema.table('bills_payments', (table) => {
|
|
||||||
table.decimal('applied_amount', 13, 3).defaultTo(0);
|
|
||||||
table
|
|
||||||
.integer('prepard_expenses_account_id')
|
|
||||||
.unsigned()
|
|
||||||
.references('id')
|
|
||||||
.inTable('accounts');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
exports.down = function (knex) {
|
|
||||||
return knex.schema.table('bills_payments', (table) => {
|
|
||||||
table.dropColumn('applied_amount');
|
|
||||||
table.dropColumn('prepard_expenses_account_id');
|
|
||||||
});
|
|
||||||
};
|
|
||||||
@@ -166,10 +166,3 @@ export interface IBillOpenedPayload {
|
|||||||
oldBill: IBill;
|
oldBill: IBill;
|
||||||
tenantId: number;
|
tenantId: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export interface IBillPrepardExpensesAppliedEventPayload {
|
|
||||||
tenantId: number;
|
|
||||||
billId: number;
|
|
||||||
trx?: Knex.Transaction;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -29,9 +29,6 @@ export interface IBillPayment {
|
|||||||
|
|
||||||
localAmount?: number;
|
localAmount?: number;
|
||||||
branchId?: number;
|
branchId?: number;
|
||||||
|
|
||||||
prepardExpensesAccountId?: number;
|
|
||||||
isPrepardExpense: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IBillPaymentEntryDTO {
|
export interface IBillPaymentEntryDTO {
|
||||||
@@ -41,7 +38,6 @@ export interface IBillPaymentEntryDTO {
|
|||||||
|
|
||||||
export interface IBillPaymentDTO {
|
export interface IBillPaymentDTO {
|
||||||
vendorId: number;
|
vendorId: number;
|
||||||
amount: number;
|
|
||||||
paymentAccountId: number;
|
paymentAccountId: number;
|
||||||
paymentNumber?: string;
|
paymentNumber?: string;
|
||||||
paymentDate: Date;
|
paymentDate: Date;
|
||||||
@@ -51,7 +47,6 @@ export interface IBillPaymentDTO {
|
|||||||
entries: IBillPaymentEntryDTO[];
|
entries: IBillPaymentEntryDTO[];
|
||||||
branchId?: number;
|
branchId?: number;
|
||||||
attachments?: AttachmentLinkDTO[];
|
attachments?: AttachmentLinkDTO[];
|
||||||
prepardExpensesAccountId?: number;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IBillReceivePageEntry {
|
export interface IBillReceivePageEntry {
|
||||||
@@ -124,11 +119,3 @@ export enum IPaymentMadeAction {
|
|||||||
Delete = 'Delete',
|
Delete = 'Delete',
|
||||||
View = 'View',
|
View = 'View',
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentPrepardExpensesAppliedEventPayload {
|
|
||||||
tenantId: number;
|
|
||||||
billPaymentId: number;
|
|
||||||
billId: number;
|
|
||||||
appliedAmount: number;
|
|
||||||
trx?: Knex.Transaction;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -25,13 +25,8 @@ export interface IPaymentReceive {
|
|||||||
updatedAt: Date;
|
updatedAt: Date;
|
||||||
localAmount?: number;
|
localAmount?: number;
|
||||||
branchId?: number;
|
branchId?: number;
|
||||||
unearnedRevenueAccountId?: number;
|
|
||||||
}
|
}
|
||||||
|
export interface IPaymentReceiveCreateDTO {
|
||||||
interface IPaymentReceivedCommonDTO {
|
|
||||||
unearnedRevenueAccountId?: number;
|
|
||||||
}
|
|
||||||
export interface IPaymentReceiveCreateDTO extends IPaymentReceivedCommonDTO {
|
|
||||||
customerId: number;
|
customerId: number;
|
||||||
paymentDate: Date;
|
paymentDate: Date;
|
||||||
amount: number;
|
amount: number;
|
||||||
@@ -46,7 +41,7 @@ export interface IPaymentReceiveCreateDTO extends IPaymentReceivedCommonDTO {
|
|||||||
attachments?: AttachmentLinkDTO[];
|
attachments?: AttachmentLinkDTO[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IPaymentReceiveEditDTO extends IPaymentReceivedCommonDTO {
|
export interface IPaymentReceiveEditDTO {
|
||||||
customerId: number;
|
customerId: number;
|
||||||
paymentDate: Date;
|
paymentDate: Date;
|
||||||
amount: number;
|
amount: number;
|
||||||
@@ -189,11 +184,3 @@ export interface PaymentReceiveMailPresendEvent {
|
|||||||
paymentReceiveId: number;
|
paymentReceiveId: number;
|
||||||
messageOptions: PaymentReceiveMailOptsDTO;
|
messageOptions: PaymentReceiveMailOptsDTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface PaymentReceiveUnearnedRevenueAppliedEventPayload {
|
|
||||||
tenantId: number;
|
|
||||||
paymentReceiveId: number;
|
|
||||||
saleInvoiceId: number;
|
|
||||||
appliedAmount: number;
|
|
||||||
trx?: Knex.Transaction;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -216,9 +216,3 @@ export interface ISaleInvoiceMailSent {
|
|||||||
saleInvoiceId: number;
|
saleInvoiceId: number;
|
||||||
messageOptions: SendInvoiceMailDTO;
|
messageOptions: SendInvoiceMailDTO;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SaleInvoiceAppliedUnearnedRevenueOnCreatedEventPayload {
|
|
||||||
tenantId: number;
|
|
||||||
saleInvoiceId: number;
|
|
||||||
trx?: Knex.Transaction;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -113,8 +113,7 @@ import { UnlinkBankRuleOnDeleteBankRule } from '@/services/Banking/Rules/events/
|
|||||||
import { DecrementUncategorizedTransactionOnMatching } from '@/services/Banking/Matching/events/DecrementUncategorizedTransactionsOnMatch';
|
import { DecrementUncategorizedTransactionOnMatching } from '@/services/Banking/Matching/events/DecrementUncategorizedTransactionsOnMatch';
|
||||||
import { DecrementUncategorizedTransactionOnExclude } from '@/services/Banking/Exclude/events/DecrementUncategorizedTransactionOnExclude';
|
import { DecrementUncategorizedTransactionOnExclude } from '@/services/Banking/Exclude/events/DecrementUncategorizedTransactionOnExclude';
|
||||||
import { DecrementUncategorizedTransactionOnCategorize } from '@/services/Cashflow/subscribers/DecrementUncategorizedTransactionOnCategorize';
|
import { DecrementUncategorizedTransactionOnCategorize } from '@/services/Cashflow/subscribers/DecrementUncategorizedTransactionOnCategorize';
|
||||||
import { AutoApplyUnearnedRevenueOnInvoiceCreated } from '@/services/Sales/PaymentReceives/events/AutoApplyUnearnedRevenueOnInvoiceCreated';
|
import { LoopsEventsSubscriber } from '@/services/Loops/LoopsEventsSubscriber';
|
||||||
import { AutoApplyPrepardExpensesOnBillCreated } from '@/services/Purchases/Bills/events/AutoApplyPrepardExpensesOnBillCreated';
|
|
||||||
|
|
||||||
export default () => {
|
export default () => {
|
||||||
return new EventPublisher();
|
return new EventPublisher();
|
||||||
@@ -276,5 +275,8 @@ export const susbcribers = () => {
|
|||||||
|
|
||||||
// Plaid
|
// Plaid
|
||||||
RecognizeSyncedBankTranasctions,
|
RecognizeSyncedBankTranasctions,
|
||||||
|
|
||||||
|
// Loops
|
||||||
|
LoopsEventsSubscriber
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -11,8 +11,6 @@ export default class BillPayment extends mixin(TenantModel, [
|
|||||||
CustomViewBaseModel,
|
CustomViewBaseModel,
|
||||||
ModelSearchable,
|
ModelSearchable,
|
||||||
]) {
|
]) {
|
||||||
prepardExpensesAccountId: number;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Table name
|
* Table name
|
||||||
*/
|
*/
|
||||||
@@ -49,14 +47,6 @@ export default class BillPayment extends mixin(TenantModel, [
|
|||||||
return BillPaymentSettings;
|
return BillPaymentSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Detarmines whether the payment is prepard expense.
|
|
||||||
* @returns {boolean}
|
|
||||||
*/
|
|
||||||
get isPrepardExpense() {
|
|
||||||
return !!this.prepardExpensesAccountId;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Relationship mapping.
|
* Relationship mapping.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -38,24 +38,4 @@ export default class ContactTransfromer extends Transformer {
|
|||||||
? this.formatDate(contact.openingBalanceAt)
|
? this.formatDate(contact.openingBalanceAt)
|
||||||
: '';
|
: '';
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the unused credit balance.
|
|
||||||
* @param {IContact} contact
|
|
||||||
* @returns {number}
|
|
||||||
*/
|
|
||||||
protected unusedCredit = (contact: IContact): number => {
|
|
||||||
return contact.balance > 0 ? 0 : Math.abs(contact.balance);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the formatted unused credit balance.
|
|
||||||
* @param {IContact} contact
|
|
||||||
* @returns {string}
|
|
||||||
*/
|
|
||||||
protected formattedUnusedCredit = (contact: IContact): string => {
|
|
||||||
const unusedCredit = this.unusedCredit(contact);
|
|
||||||
|
|
||||||
return formatNumber(unusedCredit, { currencyCode: contact.currencyCode });
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,8 +12,6 @@ export default class CustomerTransfromer extends ContactTransfromer {
|
|||||||
'formattedOpeningBalanceAt',
|
'formattedOpeningBalanceAt',
|
||||||
'customerType',
|
'customerType',
|
||||||
'formattedCustomerType',
|
'formattedCustomerType',
|
||||||
'unusedCredit',
|
|
||||||
'formattedUnusedCredit',
|
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { Service } from 'typedi';
|
||||||
import ContactTransfromer from '../ContactTransformer';
|
import ContactTransfromer from '../ContactTransformer';
|
||||||
|
|
||||||
export default class VendorTransfromer extends ContactTransfromer {
|
export default class VendorTransfromer extends ContactTransfromer {
|
||||||
@@ -9,9 +10,7 @@ export default class VendorTransfromer extends ContactTransfromer {
|
|||||||
return [
|
return [
|
||||||
'formattedBalance',
|
'formattedBalance',
|
||||||
'formattedOpeningBalance',
|
'formattedOpeningBalance',
|
||||||
'formattedOpeningBalanceAt',
|
'formattedOpeningBalanceAt'
|
||||||
'unusedCredit',
|
|
||||||
'formattedUnusedCredit',
|
|
||||||
];
|
];
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ export const DEFAULT_VIEWS = [
|
|||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
||||||
export const ERRORS = {
|
export const ERRORS = {
|
||||||
OPENING_BALANCE_DATE_REQUIRED: 'OPENING_BALANCE_DATE_REQUIRED',
|
OPENING_BALANCE_DATE_REQUIRED: 'OPENING_BALANCE_DATE_REQUIRED',
|
||||||
CONTACT_ALREADY_INACTIVE: 'CONTACT_ALREADY_INACTIVE',
|
CONTACT_ALREADY_INACTIVE: 'CONTACT_ALREADY_INACTIVE',
|
||||||
|
|||||||
51
packages/server/src/services/Loops/LoopsEventsSubscriber.ts
Normal file
51
packages/server/src/services/Loops/LoopsEventsSubscriber.ts
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import config from '@/config';
|
||||||
|
import { IAuthSignUpVerifiedEventPayload } from '@/interfaces';
|
||||||
|
import events from '@/subscribers/events';
|
||||||
|
import { SystemUser } from '@/system/models';
|
||||||
|
|
||||||
|
export class LoopsEventsSubscriber {
|
||||||
|
/**
|
||||||
|
* Constructor method.
|
||||||
|
*/
|
||||||
|
public attach(bus) {
|
||||||
|
bus.subscribe(
|
||||||
|
events.auth.signUpConfirmed,
|
||||||
|
this.triggerEventOnSignupVerified.bind(this)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Once the user verified sends the event to the Loops.
|
||||||
|
* @param {IAuthSignUpVerifiedEventPayload} param0
|
||||||
|
*/
|
||||||
|
public async triggerEventOnSignupVerified({
|
||||||
|
email,
|
||||||
|
userId,
|
||||||
|
}: IAuthSignUpVerifiedEventPayload) {
|
||||||
|
// Can't continue since the Loops the api key is not configured.
|
||||||
|
if (!config.loops.apiKey) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const user = await SystemUser.query().findById(userId);
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
method: 'POST',
|
||||||
|
url: 'https://app.loops.so/api/v1/events/send',
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${config.loops.apiKey}`,
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
email,
|
||||||
|
userId,
|
||||||
|
firstName: user.firstName,
|
||||||
|
lastName: user.lastName,
|
||||||
|
eventName: 'USER_VERIFIED',
|
||||||
|
eventProperties: {},
|
||||||
|
mailingLists: {},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
await axios(options);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,14 +1,8 @@
|
|||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import { sumBy, chain } from 'lodash';
|
import { sumBy } from 'lodash';
|
||||||
import { Service, Inject } from 'typedi';
|
import { Service, Inject } from 'typedi';
|
||||||
import { Knex } from 'knex';
|
import { Knex } from 'knex';
|
||||||
import {
|
import { AccountNormal, IBillPayment, ILedgerEntry } from '@/interfaces';
|
||||||
AccountNormal,
|
|
||||||
IBillPayment,
|
|
||||||
IBillPaymentEntry,
|
|
||||||
ILedger,
|
|
||||||
ILedgerEntry,
|
|
||||||
} from '@/interfaces';
|
|
||||||
import Ledger from '@/services/Accounting/Ledger';
|
import Ledger from '@/services/Accounting/Ledger';
|
||||||
import LedgerStorageService from '@/services/Accounting/LedgerStorageService';
|
import LedgerStorageService from '@/services/Accounting/LedgerStorageService';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
||||||
@@ -27,7 +21,6 @@ export class BillPaymentGLEntries {
|
|||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {number} billPaymentId
|
* @param {number} billPaymentId
|
||||||
* @param {Knex.Transaction} trx
|
* @param {Knex.Transaction} trx
|
||||||
* @returns {Promise<void>}
|
|
||||||
*/
|
*/
|
||||||
public writePaymentGLEntries = async (
|
public writePaymentGLEntries = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
@@ -72,7 +65,6 @@ export class BillPaymentGLEntries {
|
|||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {number} billPaymentId
|
* @param {number} billPaymentId
|
||||||
* @param {Knex.Transaction} trx
|
* @param {Knex.Transaction} trx
|
||||||
* @returns {Promise<void>}
|
|
||||||
*/
|
*/
|
||||||
public rewritePaymentGLEntries = async (
|
public rewritePaymentGLEntries = async (
|
||||||
tenantId: number,
|
tenantId: number,
|
||||||
@@ -110,7 +102,7 @@ export class BillPaymentGLEntries {
|
|||||||
* @param {IBillPayment} billPayment
|
* @param {IBillPayment} billPayment
|
||||||
* @returns {}
|
* @returns {}
|
||||||
*/
|
*/
|
||||||
private getPaymentCommonEntry = (billPayment: IBillPayment): ILedgerEntry => {
|
private getPaymentCommonEntry = (billPayment: IBillPayment) => {
|
||||||
const formattedDate = moment(billPayment.paymentDate).format('YYYY-MM-DD');
|
const formattedDate = moment(billPayment.paymentDate).format('YYYY-MM-DD');
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@@ -206,7 +198,6 @@ export class BillPaymentGLEntries {
|
|||||||
accountId: billPayment.paymentAccountId,
|
accountId: billPayment.paymentAccountId,
|
||||||
accountNormal: AccountNormal.DEBIT,
|
accountNormal: AccountNormal.DEBIT,
|
||||||
index: 2,
|
index: 2,
|
||||||
indexGroup: 10,
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -263,49 +254,6 @@ export class BillPaymentGLEntries {
|
|||||||
return [paymentEntry, payableEntry, ...exGainLossEntries];
|
return [paymentEntry, payableEntry, ...exGainLossEntries];
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* BEFORE APPLYING TO PAYMENT TO BILLS.
|
|
||||||
* -----------------------------------------
|
|
||||||
* - Cash/Bank - Credit.
|
|
||||||
* - Prepard Expenses - Debit
|
|
||||||
*
|
|
||||||
* AFTER APPLYING BILLS TO PAYMENT.
|
|
||||||
* -----------------------------------------
|
|
||||||
* - Prepard Expenses - Credit
|
|
||||||
* - A/P - Debit
|
|
||||||
*
|
|
||||||
* @param {number} APAccountId - A/P account id.
|
|
||||||
* @param {IBillPayment} billPayment
|
|
||||||
*/
|
|
||||||
private getPrepardExpenseGLEntries = (
|
|
||||||
APAccountId: number,
|
|
||||||
billPayment: IBillPayment
|
|
||||||
) => {
|
|
||||||
const prepardExpenseEntry = this.getPrepardExpenseEntry(billPayment);
|
|
||||||
const withdrawalEntry = this.getPaymentGLEntry(billPayment);
|
|
||||||
|
|
||||||
const paymentLinesEntries = chain(billPayment.entries)
|
|
||||||
.map((billPaymentEntry) => {
|
|
||||||
const APEntry = this.getAccountPayablePaymentLineEntry(
|
|
||||||
APAccountId,
|
|
||||||
billPayment,
|
|
||||||
billPaymentEntry
|
|
||||||
);
|
|
||||||
const creditPrepardExpenseEntry = this.getCreditPrepardExpenseEntry(
|
|
||||||
billPayment,
|
|
||||||
billPaymentEntry
|
|
||||||
);
|
|
||||||
return [creditPrepardExpenseEntry, APEntry];
|
|
||||||
})
|
|
||||||
.flatten()
|
|
||||||
.value();
|
|
||||||
const prepardExpenseEntries = [prepardExpenseEntry, withdrawalEntry];
|
|
||||||
const combinedEntries = [...prepardExpenseEntries, ...paymentLinesEntries];
|
|
||||||
|
|
||||||
return combinedEntries;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves the bill payment ledger.
|
* Retrieves the bill payment ledger.
|
||||||
* @param {IBillPayment} billPayment
|
* @param {IBillPayment} billPayment
|
||||||
@@ -318,9 +266,7 @@ export class BillPaymentGLEntries {
|
|||||||
gainLossAccountId: number,
|
gainLossAccountId: number,
|
||||||
baseCurrency: string
|
baseCurrency: string
|
||||||
): Ledger => {
|
): Ledger => {
|
||||||
const entries = billPayment.isPrepardExpense
|
const entries = this.getPaymentGLEntries(
|
||||||
? this.getPrepardExpenseGLEntries(APAccountId, billPayment)
|
|
||||||
: this.getPaymentGLEntries(
|
|
||||||
billPayment,
|
billPayment,
|
||||||
APAccountId,
|
APAccountId,
|
||||||
gainLossAccountId,
|
gainLossAccountId,
|
||||||
@@ -328,69 +274,4 @@ export class BillPaymentGLEntries {
|
|||||||
);
|
);
|
||||||
return new Ledger(entries);
|
return new Ledger(entries);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the prepard expense GL entry.
|
|
||||||
* @param {IBillPayment} billPayment
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getPrepardExpenseEntry = (
|
|
||||||
billPayment: IBillPayment
|
|
||||||
): ILedgerEntry => {
|
|
||||||
const commonJournal = this.getPaymentCommonEntry(billPayment);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
debit: billPayment.localAmount,
|
|
||||||
accountId: billPayment.prepardExpensesAccountId,
|
|
||||||
accountNormal: AccountNormal.DEBIT,
|
|
||||||
indexGroup: 10,
|
|
||||||
index: 1,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the GL entries of credit prepard expense for the give payment line.
|
|
||||||
* @param {IBillPayment} billPayment
|
|
||||||
* @param {IBillPaymentEntry} billPaymentEntry
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getCreditPrepardExpenseEntry = (
|
|
||||||
billPayment: IBillPayment,
|
|
||||||
billPaymentEntry: IBillPaymentEntry
|
|
||||||
) => {
|
|
||||||
const commonJournal = this.getPaymentCommonEntry(billPayment);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
credit: billPaymentEntry.paymentAmount,
|
|
||||||
accountId: billPayment.prepardExpensesAccountId,
|
|
||||||
accountNormal: AccountNormal.DEBIT,
|
|
||||||
index: 2,
|
|
||||||
indexGroup: 20,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the A/P debit of the payment line.
|
|
||||||
* @param {number} APAccountId
|
|
||||||
* @param {IBillPayment} billPayment
|
|
||||||
* @param {IBillPaymentEntry} billPaymentEntry
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getAccountPayablePaymentLineEntry = (
|
|
||||||
APAccountId: number,
|
|
||||||
billPayment: IBillPayment,
|
|
||||||
billPaymentEntry: IBillPaymentEntry
|
|
||||||
): ILedgerEntry => {
|
|
||||||
const commonJournal = this.getPaymentCommonEntry(billPayment);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
debit: billPaymentEntry.paymentAmount,
|
|
||||||
accountId: APAccountId,
|
|
||||||
index: 1,
|
|
||||||
indexGroup: 20,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,10 +17,6 @@ export class PaymentWriteGLEntriesSubscriber {
|
|||||||
*/
|
*/
|
||||||
public attach(bus) {
|
public attach(bus) {
|
||||||
bus.subscribe(events.billPayment.onCreated, this.handleWriteJournalEntries);
|
bus.subscribe(events.billPayment.onCreated, this.handleWriteJournalEntries);
|
||||||
bus.subscribe(
|
|
||||||
events.billPayment.onPrepardExpensesApplied,
|
|
||||||
this.handleWritePrepardExpenseGLEntries
|
|
||||||
);
|
|
||||||
bus.subscribe(
|
bus.subscribe(
|
||||||
events.billPayment.onEdited,
|
events.billPayment.onEdited,
|
||||||
this.handleRewriteJournalEntriesOncePaymentEdited
|
this.handleRewriteJournalEntriesOncePaymentEdited
|
||||||
@@ -32,8 +28,7 @@ export class PaymentWriteGLEntriesSubscriber {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles bill payment writing journal entries once created.
|
* Handle bill payment writing journal entries once created.
|
||||||
* @param {IBillPaymentEventCreatedPayload} payload -
|
|
||||||
*/
|
*/
|
||||||
private handleWriteJournalEntries = async ({
|
private handleWriteJournalEntries = async ({
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -49,22 +44,6 @@ export class PaymentWriteGLEntriesSubscriber {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles rewrite prepard expense GL entries once the bill payment applying to bills.
|
|
||||||
* @param {IBillPaymentEventCreatedPayload} payload -
|
|
||||||
*/
|
|
||||||
private handleWritePrepardExpenseGLEntries = async ({
|
|
||||||
tenantId,
|
|
||||||
billPaymentId,
|
|
||||||
trx,
|
|
||||||
}: IBillPaymentEventCreatedPayload) => {
|
|
||||||
await this.billPaymentGLEntries.rewritePaymentGLEntries(
|
|
||||||
tenantId,
|
|
||||||
billPaymentId,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle bill payment re-writing journal entries once the payment transaction be edited.
|
* Handle bill payment re-writing journal entries once the payment transaction be edited.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -11,9 +11,6 @@ export class CommandBillPaymentDTOTransformer {
|
|||||||
@Inject()
|
@Inject()
|
||||||
private branchDTOTransform: BranchTransactionDTOTransform;
|
private branchDTOTransform: BranchTransactionDTOTransform;
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private tenancy: HasTenancyService;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Transforms create/edit DTO to model.
|
* Transforms create/edit DTO to model.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
@@ -27,27 +24,17 @@ export class CommandBillPaymentDTOTransformer {
|
|||||||
vendor: IVendor,
|
vendor: IVendor,
|
||||||
oldBillPayment?: IBillPayment
|
oldBillPayment?: IBillPayment
|
||||||
): Promise<IBillPayment> {
|
): Promise<IBillPayment> {
|
||||||
const { accountRepository } = this.tenancy.repositories(tenantId);
|
const amount =
|
||||||
const appliedAmount = sumBy(billPaymentDTO.entries, 'paymentAmount');
|
billPaymentDTO.amount ?? sumBy(billPaymentDTO.entries, 'paymentAmount');
|
||||||
|
|
||||||
const hasPrepardExpenses = appliedAmount < billPaymentDTO.amount;
|
|
||||||
const prepardExpensesAccount = hasPrepardExpenses
|
|
||||||
? await accountRepository.findOrCreatePrepardExpenses()
|
|
||||||
: null;
|
|
||||||
const prepardExpensesAccountId =
|
|
||||||
hasPrepardExpenses && prepardExpensesAccount
|
|
||||||
? billPaymentDTO.prepardExpensesAccountId ?? prepardExpensesAccount?.id
|
|
||||||
: billPaymentDTO.prepardExpensesAccountId;
|
|
||||||
|
|
||||||
const initialDTO = {
|
const initialDTO = {
|
||||||
...formatDateFields(omit(billPaymentDTO, ['attachments']), [
|
...formatDateFields(omit(billPaymentDTO, ['attachments']), [
|
||||||
'paymentDate',
|
'paymentDate',
|
||||||
]),
|
]),
|
||||||
appliedAmount,
|
amount,
|
||||||
currencyCode: vendor.currencyCode,
|
currencyCode: vendor.currencyCode,
|
||||||
exchangeRate: billPaymentDTO.exchangeRate || 1,
|
exchangeRate: billPaymentDTO.exchangeRate || 1,
|
||||||
entries: billPaymentDTO.entries,
|
entries: billPaymentDTO.entries,
|
||||||
prepardExpensesAccountId,
|
|
||||||
};
|
};
|
||||||
return R.compose(
|
return R.compose(
|
||||||
this.branchDTOTransform.transformDTO<IBillPayment>(tenantId)
|
this.branchDTOTransform.transformDTO<IBillPayment>(tenantId)
|
||||||
|
|||||||
@@ -1,117 +0,0 @@
|
|||||||
import { Knex } from 'knex';
|
|
||||||
import { Inject, Service } from 'typedi';
|
|
||||||
import PromisePool, { ProcessHandler } from '@supercharge/promise-pool';
|
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
|
||||||
import {
|
|
||||||
IBillPayment,
|
|
||||||
IBillPrepardExpensesAppliedEventPayload,
|
|
||||||
IPaymentPrepardExpensesAppliedEventPayload,
|
|
||||||
} from '@/interfaces';
|
|
||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
|
||||||
import events from '@/subscribers/events';
|
|
||||||
|
|
||||||
@Service()
|
|
||||||
export class AutoApplyPrepardExpenses {
|
|
||||||
@Inject()
|
|
||||||
private tenancy: HasTenancyService;
|
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private eventPublisher: EventPublisher;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Auto apply prepard expenses to the given bill.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} billId
|
|
||||||
* @returns {Promise<void>}
|
|
||||||
*/
|
|
||||||
async autoApplyPrepardExpensesToBill(
|
|
||||||
tenantId: number,
|
|
||||||
billId: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
): Promise<void> {
|
|
||||||
const { BillPayment, Bill } = this.tenancy.models(tenantId);
|
|
||||||
|
|
||||||
const bill = await Bill.query(trx).findById(billId).throwIfNotFound();
|
|
||||||
|
|
||||||
const unappliedPayments = await BillPayment.query(trx)
|
|
||||||
.where('vendorId', bill.vendorId)
|
|
||||||
.whereRaw('amount - applied_amount > 0')
|
|
||||||
.whereNotNull('prepardExpensesAccountId');
|
|
||||||
|
|
||||||
let unappliedAmount = bill.total;
|
|
||||||
let appliedTotalAmount = 0; // Total applied amount after applying.
|
|
||||||
|
|
||||||
const precessHandler: ProcessHandler<IBillPayment, void> = async (
|
|
||||||
unappliedPayment: IBillPayment,
|
|
||||||
index: number,
|
|
||||||
pool
|
|
||||||
) => {
|
|
||||||
const appliedAmount = Math.min(unappliedAmount, unappliedPayment.amount);
|
|
||||||
unappliedAmount = unappliedAmount - appliedAmount;
|
|
||||||
appliedTotalAmount += appliedAmount;
|
|
||||||
|
|
||||||
// Stop applying once the unapplied amount reach zero or less.
|
|
||||||
if (appliedAmount <= 0) {
|
|
||||||
pool.stop();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await this.applyBillToPaymentMade(
|
|
||||||
tenantId,
|
|
||||||
unappliedPayment.id,
|
|
||||||
bill.id,
|
|
||||||
appliedAmount,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
};
|
|
||||||
await PromisePool.withConcurrency(1)
|
|
||||||
.for(unappliedPayments)
|
|
||||||
.process(precessHandler);
|
|
||||||
|
|
||||||
// Increase the paid amount of the purchase invoice.
|
|
||||||
await Bill.changePaymentAmount(billId, appliedTotalAmount, trx);
|
|
||||||
|
|
||||||
// Triggers `onBillPrepardExpensesApplied` event.
|
|
||||||
await this.eventPublisher.emitAsync(events.bill.onPrepardExpensesApplied, {
|
|
||||||
tenantId,
|
|
||||||
billId,
|
|
||||||
trx,
|
|
||||||
} as IBillPrepardExpensesAppliedEventPayload);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Apply the given bill to payment made transaction.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} billPaymentId
|
|
||||||
* @param {number} billId
|
|
||||||
* @param {number} appliedAmount
|
|
||||||
* @param {Knex.Transaction} trx
|
|
||||||
*/
|
|
||||||
public applyBillToPaymentMade = async (
|
|
||||||
tenantId: number,
|
|
||||||
billPaymentId: number,
|
|
||||||
billId: number,
|
|
||||||
appliedAmount: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
) => {
|
|
||||||
const { BillPaymentEntry, BillPayment } = this.tenancy.models(tenantId);
|
|
||||||
|
|
||||||
await BillPaymentEntry.query(trx).insert({
|
|
||||||
billPaymentId,
|
|
||||||
billId,
|
|
||||||
paymentAmount: appliedAmount,
|
|
||||||
});
|
|
||||||
await BillPayment.query(trx).increment('appliedAmount', appliedAmount);
|
|
||||||
|
|
||||||
// Triggers `onBillPaymentPrepardExpensesApplied` event.
|
|
||||||
await this.eventPublisher.emitAsync(
|
|
||||||
events.billPayment.onPrepardExpensesApplied,
|
|
||||||
{
|
|
||||||
tenantId,
|
|
||||||
billPaymentId,
|
|
||||||
billId,
|
|
||||||
appliedAmount,
|
|
||||||
trx,
|
|
||||||
} as IPaymentPrepardExpensesAppliedEventPayload
|
|
||||||
);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -1,36 +0,0 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
|
||||||
import { AutoApplyPrepardExpenses } from '../AutoApplyPrepardExpenses';
|
|
||||||
import events from '@/subscribers/events';
|
|
||||||
import { IBillCreatedPayload } from '@/interfaces';
|
|
||||||
|
|
||||||
@Service()
|
|
||||||
export class AutoApplyPrepardExpensesOnBillCreated {
|
|
||||||
@Inject()
|
|
||||||
private autoApplyPrepardExpenses: AutoApplyPrepardExpenses;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Constructor method.
|
|
||||||
*/
|
|
||||||
public attach(bus) {
|
|
||||||
bus.subscribe(
|
|
||||||
events.bill.onCreated,
|
|
||||||
this.handleAutoApplyPrepardExpensesOnBillCreated.bind(this)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles the auto apply prepard expenses on bill created.
|
|
||||||
* @param {IBillCreatedPayload} payload -
|
|
||||||
*/
|
|
||||||
private async handleAutoApplyPrepardExpensesOnBillCreated({
|
|
||||||
tenantId,
|
|
||||||
billId,
|
|
||||||
trx,
|
|
||||||
}: IBillCreatedPayload) {
|
|
||||||
await this.autoApplyPrepardExpenses.autoApplyPrepardExpensesToBill(
|
|
||||||
tenantId,
|
|
||||||
billId,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,126 +0,0 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
|
||||||
import { Knex } from 'knex';
|
|
||||||
import PromisePool, { ProcessHandler } from '@supercharge/promise-pool';
|
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
|
||||||
import { EventPublisher } from '@/lib/EventPublisher/EventPublisher';
|
|
||||||
import events from '@/subscribers/events';
|
|
||||||
import {
|
|
||||||
IPaymentReceive,
|
|
||||||
PaymentReceiveUnearnedRevenueAppliedEventPayload,
|
|
||||||
SaleInvoiceAppliedUnearnedRevenueOnCreatedEventPayload,
|
|
||||||
} from '@/interfaces';
|
|
||||||
|
|
||||||
@Service()
|
|
||||||
export class AutoApplyUnearnedRevenue {
|
|
||||||
@Inject()
|
|
||||||
private tenancy: HasTenancyService;
|
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private eventPublisher: EventPublisher;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Auto apply invoice to advanced payment received transactions.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} invoiceId
|
|
||||||
* @returns {Promise<void>}
|
|
||||||
*/
|
|
||||||
public async autoApplyUnearnedRevenueToInvoice(
|
|
||||||
tenantId: number,
|
|
||||||
saleInvoiceId: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
): Promise<void> {
|
|
||||||
const { PaymentReceive, SaleInvoice } = this.tenancy.models(tenantId);
|
|
||||||
|
|
||||||
const invoice = await SaleInvoice.query(trx)
|
|
||||||
.findById(saleInvoiceId)
|
|
||||||
.throwIfNotFound();
|
|
||||||
|
|
||||||
const unappliedPayments = await PaymentReceive.query(trx)
|
|
||||||
.where('customerId', invoice.customerId)
|
|
||||||
.whereRaw('amount - applied_amount > 0')
|
|
||||||
.whereNotNull('unearnedRevenueAccountId');
|
|
||||||
|
|
||||||
let unappliedAmount = invoice.total;
|
|
||||||
let appliedTotalAmount = 0; // Total applied amount after applying.
|
|
||||||
|
|
||||||
const processHandler: ProcessHandler<
|
|
||||||
IPaymentReceive,
|
|
||||||
Promise<void>
|
|
||||||
> = async (unappliedPayment: IPaymentReceive, index: number, pool) => {
|
|
||||||
const appliedAmount = Math.min(unappliedAmount, unappliedPayment.amount);
|
|
||||||
unappliedAmount = unappliedAmount - appliedAmount;
|
|
||||||
appliedTotalAmount += appliedAmount;
|
|
||||||
|
|
||||||
// Stop applying once the unapplied amount reache zero or less.
|
|
||||||
if (appliedAmount <= 0) {
|
|
||||||
pool.stop();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await this.applyInvoiceToPaymentReceived(
|
|
||||||
tenantId,
|
|
||||||
unappliedPayment.id,
|
|
||||||
invoice.id,
|
|
||||||
appliedAmount,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
};
|
|
||||||
await PromisePool.withConcurrency(1)
|
|
||||||
.for(unappliedPayments)
|
|
||||||
.process(processHandler);
|
|
||||||
|
|
||||||
// Increase the paid amount of the sale invoice.
|
|
||||||
await SaleInvoice.changePaymentAmount(
|
|
||||||
saleInvoiceId,
|
|
||||||
appliedTotalAmount,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
// Triggers event `onSaleInvoiceUnearnedRevenue`.
|
|
||||||
await this.eventPublisher.emitAsync(
|
|
||||||
events.saleInvoice.onUnearnedRevenueApplied,
|
|
||||||
{
|
|
||||||
tenantId,
|
|
||||||
saleInvoiceId,
|
|
||||||
trx,
|
|
||||||
} as SaleInvoiceAppliedUnearnedRevenueOnCreatedEventPayload
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Apply the given invoice to payment received transaction.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} paymentReceivedId
|
|
||||||
* @param {number} invoiceId
|
|
||||||
* @param {number} appliedAmount
|
|
||||||
* @param {Knex.Transaction} trx
|
|
||||||
* @returns {Promise<void>}
|
|
||||||
*/
|
|
||||||
public applyInvoiceToPaymentReceived = async (
|
|
||||||
tenantId: number,
|
|
||||||
paymentReceiveId: number,
|
|
||||||
invoiceId: number,
|
|
||||||
appliedAmount: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
): Promise<void> => {
|
|
||||||
const { PaymentReceiveEntry, PaymentReceive } =
|
|
||||||
this.tenancy.models(tenantId);
|
|
||||||
|
|
||||||
await PaymentReceiveEntry.query(trx).insert({
|
|
||||||
paymentReceiveId,
|
|
||||||
invoiceId,
|
|
||||||
paymentAmount: appliedAmount,
|
|
||||||
});
|
|
||||||
await PaymentReceive.query(trx).increment('appliedAmount', appliedAmount);
|
|
||||||
|
|
||||||
// Triggers the event `onPaymentReceivedUnearnedRevenue`.
|
|
||||||
await this.eventPublisher.emitAsync(
|
|
||||||
events.paymentReceive.onUnearnedRevenueApplied,
|
|
||||||
{
|
|
||||||
tenantId,
|
|
||||||
paymentReceiveId,
|
|
||||||
saleInvoiceId: invoiceId,
|
|
||||||
appliedAmount,
|
|
||||||
trx,
|
|
||||||
} as PaymentReceiveUnearnedRevenueAppliedEventPayload
|
|
||||||
);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -11,7 +11,6 @@ import { PaymentReceiveValidators } from './PaymentReceiveValidators';
|
|||||||
import { PaymentReceiveIncrement } from './PaymentReceiveIncrement';
|
import { PaymentReceiveIncrement } from './PaymentReceiveIncrement';
|
||||||
import { BranchTransactionDTOTransform } from '@/services/Branches/Integrations/BranchTransactionDTOTransform';
|
import { BranchTransactionDTOTransform } from '@/services/Branches/Integrations/BranchTransactionDTOTransform';
|
||||||
import { formatDateFields } from '@/utils';
|
import { formatDateFields } from '@/utils';
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class PaymentReceiveDTOTransformer {
|
export class PaymentReceiveDTOTransformer {
|
||||||
@@ -24,9 +23,6 @@ export class PaymentReceiveDTOTransformer {
|
|||||||
@Inject()
|
@Inject()
|
||||||
private branchDTOTransform: BranchTransactionDTOTransform;
|
private branchDTOTransform: BranchTransactionDTOTransform;
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private tenancy: HasTenancyService;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Transformes the create payment receive DTO to model object.
|
* Transformes the create payment receive DTO to model object.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
@@ -40,8 +36,9 @@ export class PaymentReceiveDTOTransformer {
|
|||||||
paymentReceiveDTO: IPaymentReceiveCreateDTO | IPaymentReceiveEditDTO,
|
paymentReceiveDTO: IPaymentReceiveCreateDTO | IPaymentReceiveEditDTO,
|
||||||
oldPaymentReceive?: IPaymentReceive
|
oldPaymentReceive?: IPaymentReceive
|
||||||
): Promise<IPaymentReceive> {
|
): Promise<IPaymentReceive> {
|
||||||
const { accountRepository } = this.tenancy.repositories(tenantId);
|
const amount =
|
||||||
const appliedAmount = sumBy(paymentReceiveDTO.entries, 'paymentAmount');
|
paymentReceiveDTO.amount ??
|
||||||
|
sumBy(paymentReceiveDTO.entries, 'paymentAmount');
|
||||||
|
|
||||||
// Retreive the next invoice number.
|
// Retreive the next invoice number.
|
||||||
const autoNextNumber =
|
const autoNextNumber =
|
||||||
@@ -55,29 +52,17 @@ export class PaymentReceiveDTOTransformer {
|
|||||||
|
|
||||||
this.validators.validatePaymentNoRequire(paymentReceiveNo);
|
this.validators.validatePaymentNoRequire(paymentReceiveNo);
|
||||||
|
|
||||||
const hasUnearnedPayment = appliedAmount < paymentReceiveDTO.amount;
|
|
||||||
const unearnedRevenueAccount = hasUnearnedPayment
|
|
||||||
? await accountRepository.findOrCreateUnearnedRevenue()
|
|
||||||
: null;
|
|
||||||
|
|
||||||
const unearnedRevenueAccountId =
|
|
||||||
hasUnearnedPayment && unearnedRevenueAccount
|
|
||||||
? paymentReceiveDTO.unearnedRevenueAccountId ??
|
|
||||||
unearnedRevenueAccount?.id
|
|
||||||
: paymentReceiveDTO.unearnedRevenueAccountId;
|
|
||||||
|
|
||||||
const initialDTO = {
|
const initialDTO = {
|
||||||
...formatDateFields(omit(paymentReceiveDTO, ['entries', 'attachments']), [
|
...formatDateFields(omit(paymentReceiveDTO, ['entries', 'attachments']), [
|
||||||
'paymentDate',
|
'paymentDate',
|
||||||
]),
|
]),
|
||||||
appliedAmount,
|
amount,
|
||||||
currencyCode: customer.currencyCode,
|
currencyCode: customer.currencyCode,
|
||||||
...(paymentReceiveNo ? { paymentReceiveNo } : {}),
|
...(paymentReceiveNo ? { paymentReceiveNo } : {}),
|
||||||
exchangeRate: paymentReceiveDTO.exchangeRate || 1,
|
exchangeRate: paymentReceiveDTO.exchangeRate || 1,
|
||||||
entries: paymentReceiveDTO.entries.map((entry) => ({
|
entries: paymentReceiveDTO.entries.map((entry) => ({
|
||||||
...entry,
|
...entry,
|
||||||
})),
|
})),
|
||||||
unearnedRevenueAccountId,
|
|
||||||
};
|
};
|
||||||
return R.compose(
|
return R.compose(
|
||||||
this.branchDTOTransform.transformDTO<IPaymentReceive>(tenantId)
|
this.branchDTOTransform.transformDTO<IPaymentReceive>(tenantId)
|
||||||
|
|||||||
@@ -1,14 +1,19 @@
|
|||||||
import { Service, Inject } from 'typedi';
|
import { Service, Inject } from 'typedi';
|
||||||
|
import { sumBy } from 'lodash';
|
||||||
import { Knex } from 'knex';
|
import { Knex } from 'knex';
|
||||||
import Ledger from '@/services/Accounting/Ledger';
|
import Ledger from '@/services/Accounting/Ledger';
|
||||||
import TenancyService from '@/services/Tenancy/TenancyService';
|
import TenancyService from '@/services/Tenancy/TenancyService';
|
||||||
import { IPaymentReceive, ILedgerEntry, AccountNormal } from '@/interfaces';
|
import {
|
||||||
|
IPaymentReceive,
|
||||||
|
ILedgerEntry,
|
||||||
|
AccountNormal,
|
||||||
|
IPaymentReceiveGLCommonEntry,
|
||||||
|
} from '@/interfaces';
|
||||||
import LedgerStorageService from '@/services/Accounting/LedgerStorageService';
|
import LedgerStorageService from '@/services/Accounting/LedgerStorageService';
|
||||||
import { TenantMetadata } from '@/system/models';
|
import { TenantMetadata } from '@/system/models';
|
||||||
import { PaymentReceivedGLCommon } from './PaymentReceivedGLCommon';
|
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export class PaymentReceiveGLEntries extends PaymentReceivedGLCommon {
|
export class PaymentReceiveGLEntries {
|
||||||
@Inject()
|
@Inject()
|
||||||
private tenancy: TenancyService;
|
private tenancy: TenancyService;
|
||||||
|
|
||||||
@@ -17,9 +22,9 @@ export class PaymentReceiveGLEntries extends PaymentReceivedGLCommon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Writes payment GL entries to the storage.
|
* Writes payment GL entries to the storage.
|
||||||
* @param {number} tenantId - Tenant id.
|
* @param {number} tenantId
|
||||||
* @param {number} paymentReceiveId - Payment received id.
|
* @param {number} paymentReceiveId
|
||||||
* @param {Knex.Transaction} trx - Knex transaction.
|
* @param {Knex.Transaction} trx
|
||||||
* @returns {Promise<void>}
|
* @returns {Promise<void>}
|
||||||
*/
|
*/
|
||||||
public writePaymentGLEntries = async (
|
public writePaymentGLEntries = async (
|
||||||
@@ -29,19 +34,14 @@ export class PaymentReceiveGLEntries extends PaymentReceivedGLCommon {
|
|||||||
): Promise<void> => {
|
): Promise<void> => {
|
||||||
const { PaymentReceive } = this.tenancy.models(tenantId);
|
const { PaymentReceive } = this.tenancy.models(tenantId);
|
||||||
|
|
||||||
|
// Retrieves the given tenant metadata.
|
||||||
|
const tenantMeta = await TenantMetadata.query().findOne({ tenantId });
|
||||||
|
|
||||||
// Retrieves the payment receive with associated entries.
|
// Retrieves the payment receive with associated entries.
|
||||||
const paymentReceive = await PaymentReceive.query(trx)
|
const paymentReceive = await PaymentReceive.query(trx)
|
||||||
.findById(paymentReceiveId)
|
.findById(paymentReceiveId)
|
||||||
.withGraphFetched('entries.invoice');
|
.withGraphFetched('entries.invoice');
|
||||||
|
|
||||||
// Cannot continue if the received payment is unearned revenue type,
|
|
||||||
// that type of transactions have different type of GL entries.
|
|
||||||
if (paymentReceive.unearnedRevenueAccountId) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Retrieves the given tenant metadata.
|
|
||||||
const tenantMeta = await TenantMetadata.query().findOne({ tenantId });
|
|
||||||
|
|
||||||
// Retrives the payment receive ledger.
|
// Retrives the payment receive ledger.
|
||||||
const ledger = await this.getPaymentReceiveGLedger(
|
const ledger = await this.getPaymentReceiveGLedger(
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -53,6 +53,25 @@ export class PaymentReceiveGLEntries extends PaymentReceivedGLCommon {
|
|||||||
await this.ledgerStorage.commit(tenantId, ledger, trx);
|
await this.ledgerStorage.commit(tenantId, ledger, trx);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reverts the given payment receive GL entries.
|
||||||
|
* @param {number} tenantId
|
||||||
|
* @param {number} paymentReceiveId
|
||||||
|
* @param {Knex.Transaction} trx
|
||||||
|
*/
|
||||||
|
public revertPaymentGLEntries = async (
|
||||||
|
tenantId: number,
|
||||||
|
paymentReceiveId: number,
|
||||||
|
trx?: Knex.Transaction
|
||||||
|
) => {
|
||||||
|
await this.ledgerStorage.deleteByReference(
|
||||||
|
tenantId,
|
||||||
|
paymentReceiveId,
|
||||||
|
'PaymentReceive',
|
||||||
|
trx
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Rewrites the given payment receive GL entries.
|
* Rewrites the given payment receive GL entries.
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
@@ -107,6 +126,97 @@ export class PaymentReceiveGLEntries extends PaymentReceivedGLCommon {
|
|||||||
return new Ledger(ledgerEntries);
|
return new Ledger(ledgerEntries);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Calculates the payment total exchange gain/loss.
|
||||||
|
* @param {IBillPayment} paymentReceive - Payment receive with entries.
|
||||||
|
* @returns {number}
|
||||||
|
*/
|
||||||
|
private getPaymentExGainOrLoss = (
|
||||||
|
paymentReceive: IPaymentReceive
|
||||||
|
): number => {
|
||||||
|
return sumBy(paymentReceive.entries, (entry) => {
|
||||||
|
const paymentLocalAmount =
|
||||||
|
entry.paymentAmount * paymentReceive.exchangeRate;
|
||||||
|
const invoicePayment = entry.paymentAmount * entry.invoice.exchangeRate;
|
||||||
|
|
||||||
|
return paymentLocalAmount - invoicePayment;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the common entry of payment receive.
|
||||||
|
* @param {IPaymentReceive} paymentReceive
|
||||||
|
* @returns {}
|
||||||
|
*/
|
||||||
|
private getPaymentReceiveCommonEntry = (
|
||||||
|
paymentReceive: IPaymentReceive
|
||||||
|
): IPaymentReceiveGLCommonEntry => {
|
||||||
|
return {
|
||||||
|
debit: 0,
|
||||||
|
credit: 0,
|
||||||
|
|
||||||
|
currencyCode: paymentReceive.currencyCode,
|
||||||
|
exchangeRate: paymentReceive.exchangeRate,
|
||||||
|
|
||||||
|
transactionId: paymentReceive.id,
|
||||||
|
transactionType: 'PaymentReceive',
|
||||||
|
|
||||||
|
transactionNumber: paymentReceive.paymentReceiveNo,
|
||||||
|
referenceNumber: paymentReceive.referenceNo,
|
||||||
|
|
||||||
|
date: paymentReceive.paymentDate,
|
||||||
|
userId: paymentReceive.userId,
|
||||||
|
createdAt: paymentReceive.createdAt,
|
||||||
|
|
||||||
|
branchId: paymentReceive.branchId,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Retrieves the payment exchange gain/loss entry.
|
||||||
|
* @param {IPaymentReceive} paymentReceive -
|
||||||
|
* @param {number} ARAccountId -
|
||||||
|
* @param {number} exchangeGainOrLossAccountId -
|
||||||
|
* @param {string} baseCurrencyCode -
|
||||||
|
* @returns {ILedgerEntry[]}
|
||||||
|
*/
|
||||||
|
private getPaymentExchangeGainLossEntry = (
|
||||||
|
paymentReceive: IPaymentReceive,
|
||||||
|
ARAccountId: number,
|
||||||
|
exchangeGainOrLossAccountId: number,
|
||||||
|
baseCurrencyCode: string
|
||||||
|
): ILedgerEntry[] => {
|
||||||
|
const commonJournal = this.getPaymentReceiveCommonEntry(paymentReceive);
|
||||||
|
const gainOrLoss = this.getPaymentExGainOrLoss(paymentReceive);
|
||||||
|
const absGainOrLoss = Math.abs(gainOrLoss);
|
||||||
|
|
||||||
|
return gainOrLoss
|
||||||
|
? [
|
||||||
|
{
|
||||||
|
...commonJournal,
|
||||||
|
currencyCode: baseCurrencyCode,
|
||||||
|
exchangeRate: 1,
|
||||||
|
debit: gainOrLoss > 0 ? absGainOrLoss : 0,
|
||||||
|
credit: gainOrLoss < 0 ? absGainOrLoss : 0,
|
||||||
|
accountId: ARAccountId,
|
||||||
|
contactId: paymentReceive.customerId,
|
||||||
|
index: 3,
|
||||||
|
accountNormal: AccountNormal.CREDIT,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
...commonJournal,
|
||||||
|
currencyCode: baseCurrencyCode,
|
||||||
|
exchangeRate: 1,
|
||||||
|
credit: gainOrLoss > 0 ? absGainOrLoss : 0,
|
||||||
|
debit: gainOrLoss < 0 ? absGainOrLoss : 0,
|
||||||
|
accountId: exchangeGainOrLossAccountId,
|
||||||
|
index: 3,
|
||||||
|
accountNormal: AccountNormal.DEBIT,
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: [];
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves the payment deposit GL entry.
|
* Retrieves the payment deposit GL entry.
|
||||||
* @param {IPaymentReceive} paymentReceive
|
* @param {IPaymentReceive} paymentReceive
|
||||||
@@ -152,9 +262,9 @@ export class PaymentReceiveGLEntries extends PaymentReceivedGLCommon {
|
|||||||
* Records payment receive journal transactions.
|
* Records payment receive journal transactions.
|
||||||
*
|
*
|
||||||
* Invoice payment journals.
|
* Invoice payment journals.
|
||||||
* ------------
|
* --------
|
||||||
* - Account Receivable -> Debit
|
* - Account receivable -> Debit
|
||||||
* - Payment Account [current asset] -> Credit
|
* - Payment account [current asset] -> Credit
|
||||||
*
|
*
|
||||||
* @param {number} tenantId
|
* @param {number} tenantId
|
||||||
* @param {IPaymentReceive} paymentRecieve - Payment receive model.
|
* @param {IPaymentReceive} paymentRecieve - Payment receive model.
|
||||||
|
|||||||
@@ -107,6 +107,7 @@ export class PaymentReceiveValidators {
|
|||||||
const invoicesIds = paymentReceiveEntries.map(
|
const invoicesIds = paymentReceiveEntries.map(
|
||||||
(e: IPaymentReceiveEntryDTO) => e.invoiceId
|
(e: IPaymentReceiveEntryDTO) => e.invoiceId
|
||||||
);
|
);
|
||||||
|
|
||||||
const storedInvoices = await SaleInvoice.query().whereIn('id', invoicesIds);
|
const storedInvoices = await SaleInvoice.query().whereIn('id', invoicesIds);
|
||||||
|
|
||||||
const storedInvoicesMap = new Map(
|
const storedInvoicesMap = new Map(
|
||||||
|
|||||||
@@ -1,123 +0,0 @@
|
|||||||
import { Knex } from 'knex';
|
|
||||||
import { sumBy } from 'lodash';
|
|
||||||
import {
|
|
||||||
AccountNormal,
|
|
||||||
ILedgerEntry,
|
|
||||||
IPaymentReceive,
|
|
||||||
IPaymentReceiveGLCommonEntry,
|
|
||||||
} from '@/interfaces';
|
|
||||||
import LedgerStorageService from '@/services/Accounting/LedgerStorageService';
|
|
||||||
|
|
||||||
export class PaymentReceivedGLCommon {
|
|
||||||
private ledgerStorage: LedgerStorageService;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the common entry of payment receive.
|
|
||||||
* @param {IPaymentReceive} paymentReceive
|
|
||||||
* @returns {IPaymentReceiveGLCommonEntry}
|
|
||||||
*/
|
|
||||||
protected getPaymentReceiveCommonEntry = (
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
): IPaymentReceiveGLCommonEntry => {
|
|
||||||
return {
|
|
||||||
debit: 0,
|
|
||||||
credit: 0,
|
|
||||||
|
|
||||||
currencyCode: paymentReceive.currencyCode,
|
|
||||||
exchangeRate: paymentReceive.exchangeRate,
|
|
||||||
|
|
||||||
transactionId: paymentReceive.id,
|
|
||||||
transactionType: 'PaymentReceive',
|
|
||||||
|
|
||||||
transactionNumber: paymentReceive.paymentReceiveNo,
|
|
||||||
referenceNumber: paymentReceive.referenceNo,
|
|
||||||
|
|
||||||
date: paymentReceive.paymentDate,
|
|
||||||
userId: paymentReceive.userId,
|
|
||||||
createdAt: paymentReceive.createdAt,
|
|
||||||
|
|
||||||
branchId: paymentReceive.branchId,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the payment exchange gain/loss entry.
|
|
||||||
* @param {IPaymentReceive} paymentReceive -
|
|
||||||
* @param {number} ARAccountId -
|
|
||||||
* @param {number} exchangeGainOrLossAccountId -
|
|
||||||
* @param {string} baseCurrencyCode -
|
|
||||||
* @returns {ILedgerEntry[]}
|
|
||||||
*/
|
|
||||||
protected getPaymentExchangeGainLossEntry = (
|
|
||||||
paymentReceive: IPaymentReceive,
|
|
||||||
ARAccountId: number,
|
|
||||||
exchangeGainOrLossAccountId: number,
|
|
||||||
baseCurrencyCode: string
|
|
||||||
): ILedgerEntry[] => {
|
|
||||||
const commonJournal = this.getPaymentReceiveCommonEntry(paymentReceive);
|
|
||||||
const gainOrLoss = this.getPaymentExGainOrLoss(paymentReceive);
|
|
||||||
const absGainOrLoss = Math.abs(gainOrLoss);
|
|
||||||
|
|
||||||
return gainOrLoss
|
|
||||||
? [
|
|
||||||
{
|
|
||||||
...commonJournal,
|
|
||||||
currencyCode: baseCurrencyCode,
|
|
||||||
exchangeRate: 1,
|
|
||||||
debit: gainOrLoss > 0 ? absGainOrLoss : 0,
|
|
||||||
credit: gainOrLoss < 0 ? absGainOrLoss : 0,
|
|
||||||
accountId: ARAccountId,
|
|
||||||
contactId: paymentReceive.customerId,
|
|
||||||
index: 3,
|
|
||||||
accountNormal: AccountNormal.CREDIT,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
...commonJournal,
|
|
||||||
currencyCode: baseCurrencyCode,
|
|
||||||
exchangeRate: 1,
|
|
||||||
credit: gainOrLoss > 0 ? absGainOrLoss : 0,
|
|
||||||
debit: gainOrLoss < 0 ? absGainOrLoss : 0,
|
|
||||||
accountId: exchangeGainOrLossAccountId,
|
|
||||||
index: 3,
|
|
||||||
accountNormal: AccountNormal.DEBIT,
|
|
||||||
},
|
|
||||||
]
|
|
||||||
: [];
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Calculates the payment total exchange gain/loss.
|
|
||||||
* @param {IBillPayment} paymentReceive - Payment receive with entries.
|
|
||||||
* @returns {number}
|
|
||||||
*/
|
|
||||||
private getPaymentExGainOrLoss = (
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
): number => {
|
|
||||||
return sumBy(paymentReceive.entries, (entry) => {
|
|
||||||
const paymentLocalAmount =
|
|
||||||
entry.paymentAmount * paymentReceive.exchangeRate;
|
|
||||||
const invoicePayment = entry.paymentAmount * entry.invoice.exchangeRate;
|
|
||||||
|
|
||||||
return paymentLocalAmount - invoicePayment;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Reverts the given payment receive GL entries.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} paymentReceiveId
|
|
||||||
* @param {Knex.Transaction} trx
|
|
||||||
*/
|
|
||||||
public revertPaymentGLEntries = async (
|
|
||||||
tenantId: number,
|
|
||||||
paymentReceiveId: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
) => {
|
|
||||||
await this.ledgerStorage.deleteByReference(
|
|
||||||
tenantId,
|
|
||||||
paymentReceiveId,
|
|
||||||
'PaymentReceive',
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -1,252 +0,0 @@
|
|||||||
import * as R from 'ramda';
|
|
||||||
import { Knex } from 'knex';
|
|
||||||
import { Inject, Service } from 'typedi';
|
|
||||||
import { flatten } from 'lodash';
|
|
||||||
import HasTenancyService from '@/services/Tenancy/TenancyService';
|
|
||||||
import { TenantMetadata } from '@/system/models';
|
|
||||||
import { PaymentReceivedGLCommon } from './PaymentReceivedGLCommon';
|
|
||||||
import {
|
|
||||||
AccountNormal,
|
|
||||||
ILedgerEntry,
|
|
||||||
IPaymentReceive,
|
|
||||||
IPaymentReceiveEntry,
|
|
||||||
} from '@/interfaces';
|
|
||||||
import LedgerStorageService from '@/services/Accounting/LedgerStorageService';
|
|
||||||
import Ledger from '@/services/Accounting/Ledger';
|
|
||||||
|
|
||||||
@Service()
|
|
||||||
export class PaymentReceivedUnearnedGLEntries extends PaymentReceivedGLCommon {
|
|
||||||
@Inject()
|
|
||||||
private tenancy: HasTenancyService;
|
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private ledgerStorage: LedgerStorageService;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Writes payment GL entries to the storage.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} paymentReceiveId
|
|
||||||
* @param {Knex.Transaction} trx
|
|
||||||
* @returns {Promise<void>}
|
|
||||||
*/
|
|
||||||
public writePaymentGLEntries = async (
|
|
||||||
tenantId: number,
|
|
||||||
paymentReceiveId: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
): Promise<void> => {
|
|
||||||
const { PaymentReceive } = this.tenancy.models(tenantId);
|
|
||||||
|
|
||||||
// Retrieves the payment receive with associated entries.
|
|
||||||
const paymentReceive = await PaymentReceive.query(trx)
|
|
||||||
.findById(paymentReceiveId)
|
|
||||||
.withGraphFetched('entries.invoice');
|
|
||||||
|
|
||||||
// Stop early if
|
|
||||||
if (!paymentReceive.unearnedRevenueAccountId) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Retrieves the given tenant metadata.
|
|
||||||
const tenantMeta = await TenantMetadata.query().findOne({ tenantId });
|
|
||||||
|
|
||||||
const ledger = await this.getPaymentReceiveGLedger(
|
|
||||||
tenantId,
|
|
||||||
paymentReceive
|
|
||||||
);
|
|
||||||
// Commit the ledger entries to the storage.
|
|
||||||
await this.ledgerStorage.commit(tenantId, ledger, trx);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Rewrites the given payment receive GL entries.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {number} paymentReceiveId
|
|
||||||
* @param {Knex.Transaction} trx
|
|
||||||
*/
|
|
||||||
public rewritePaymentGLEntries = async (
|
|
||||||
tenantId: number,
|
|
||||||
paymentReceiveId: number,
|
|
||||||
trx?: Knex.Transaction
|
|
||||||
) => {
|
|
||||||
// Reverts the payment GL entries.
|
|
||||||
await this.revertPaymentGLEntries(tenantId, paymentReceiveId, trx);
|
|
||||||
|
|
||||||
// Writes the payment GL entries.
|
|
||||||
await this.writePaymentGLEntries(tenantId, paymentReceiveId, trx);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the payment received GL entries.
|
|
||||||
* @param {number} tenantId
|
|
||||||
* @param {IPaymentReceive} paymentReceive
|
|
||||||
* @returns {Promise<Ledger>}
|
|
||||||
*/
|
|
||||||
private getPaymentReceiveGLedger = async (
|
|
||||||
tenantId: number,
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
) => {
|
|
||||||
const { accountRepository } = this.tenancy.repositories(tenantId);
|
|
||||||
|
|
||||||
// Retrieve the A/R account of the given currency.
|
|
||||||
const receivableAccount =
|
|
||||||
await accountRepository.findOrCreateAccountReceivable(
|
|
||||||
paymentReceive.currencyCode
|
|
||||||
);
|
|
||||||
// Retrieve the payment GL entries.
|
|
||||||
const entries = this.getPaymentGLEntries(
|
|
||||||
receivableAccount.id,
|
|
||||||
paymentReceive
|
|
||||||
);
|
|
||||||
const unearnedRevenueEntries =
|
|
||||||
this.getUnearnedRevenueEntries(paymentReceive);
|
|
||||||
|
|
||||||
const combinedEntries = [...unearnedRevenueEntries, ...entries];
|
|
||||||
|
|
||||||
return new Ledger(combinedEntries);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieve the payment received GL entries.
|
|
||||||
* @param {number} ARAccountId - A/R account id.
|
|
||||||
* @param {IPaymentReceive} paymentReceive -
|
|
||||||
* @returns {Array<ILedgerEntry>}
|
|
||||||
*/
|
|
||||||
private getPaymentGLEntries = R.curry(
|
|
||||||
(
|
|
||||||
ARAccountId: number,
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
): Array<ILedgerEntry> => {
|
|
||||||
const getPaymentEntryGLEntries = this.getPaymentEntryGLEntries(
|
|
||||||
ARAccountId,
|
|
||||||
paymentReceive
|
|
||||||
);
|
|
||||||
const entriesGroup = paymentReceive.entries.map((paymentEntry) => {
|
|
||||||
return getPaymentEntryGLEntries(paymentEntry);
|
|
||||||
});
|
|
||||||
return flatten(entriesGroup);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieve the payment entry GL entries.
|
|
||||||
* @param {IPaymentReceiveEntry} paymentReceivedEntry -
|
|
||||||
* @param {IPaymentReceive} paymentReceive -
|
|
||||||
* @returns {Array<ILedgerEntry>}
|
|
||||||
*/
|
|
||||||
private getPaymentEntryGLEntries = R.curry(
|
|
||||||
(
|
|
||||||
ARAccountId: number,
|
|
||||||
paymentReceive: IPaymentReceive,
|
|
||||||
paymentReceivedEntry: IPaymentReceiveEntry
|
|
||||||
): Array<ILedgerEntry> => {
|
|
||||||
const unearnedRevenueEntry = this.getDebitUnearnedRevenueGLEntry(
|
|
||||||
paymentReceivedEntry.paymentAmount,
|
|
||||||
paymentReceive
|
|
||||||
);
|
|
||||||
const AREntry = this.getPaymentReceivableEntry(
|
|
||||||
paymentReceivedEntry.paymentAmount,
|
|
||||||
paymentReceive,
|
|
||||||
ARAccountId
|
|
||||||
);
|
|
||||||
return [unearnedRevenueEntry, AREntry];
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the payment deposit GL entry.
|
|
||||||
* @param {IPaymentReceive} paymentReceive
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getDebitUnearnedRevenueGLEntry = (
|
|
||||||
amount: number,
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
): ILedgerEntry => {
|
|
||||||
const commonJournal = this.getPaymentReceiveCommonEntry(paymentReceive);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
debit: amount,
|
|
||||||
accountId: paymentReceive.unearnedRevenueAccountId,
|
|
||||||
accountNormal: AccountNormal.CREDIT,
|
|
||||||
index: 2,
|
|
||||||
indexGroup: 20,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the payment receivable entry.
|
|
||||||
* @param {IPaymentReceive} paymentReceive
|
|
||||||
* @param {number} ARAccountId
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getPaymentReceivableEntry = (
|
|
||||||
amount: number,
|
|
||||||
paymentReceive: IPaymentReceive,
|
|
||||||
ARAccountId: number
|
|
||||||
): ILedgerEntry => {
|
|
||||||
const commonJournal = this.getPaymentReceiveCommonEntry(paymentReceive);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
credit: amount,
|
|
||||||
contactId: paymentReceive.customerId,
|
|
||||||
accountId: ARAccountId,
|
|
||||||
accountNormal: AccountNormal.DEBIT,
|
|
||||||
index: 1,
|
|
||||||
indexGroup: 20,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieves the unearned revenue entries.
|
|
||||||
* @param {IPaymentReceive} paymentReceived -
|
|
||||||
* @returns {Array<ILedgerEntry>}
|
|
||||||
*/
|
|
||||||
private getUnearnedRevenueEntries = (
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
): Array<ILedgerEntry> => {
|
|
||||||
const depositEntry = this.getDepositPaymentGLEntry(paymentReceive);
|
|
||||||
const unearnedEntry = this.getUnearnedRevenueEntry(paymentReceive);
|
|
||||||
|
|
||||||
return [depositEntry, unearnedEntry];
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieve the payment deposit entry.
|
|
||||||
* @param {IPaymentReceive} paymentReceived -
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getDepositPaymentGLEntry = (
|
|
||||||
paymentReceive: IPaymentReceive
|
|
||||||
): ILedgerEntry => {
|
|
||||||
const commonJournal = this.getPaymentReceiveCommonEntry(paymentReceive);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
debit: paymentReceive.amount,
|
|
||||||
accountId: paymentReceive.depositAccountId,
|
|
||||||
accountNormal: AccountNormal.DEBIT,
|
|
||||||
indexGroup: 10,
|
|
||||||
index: 1,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Retrieve the unearned revenue entry.
|
|
||||||
* @param {IPaymentReceive} paymentReceived -
|
|
||||||
* @returns {ILedgerEntry}
|
|
||||||
*/
|
|
||||||
private getUnearnedRevenueEntry = (
|
|
||||||
paymentReceived: IPaymentReceive
|
|
||||||
): ILedgerEntry => {
|
|
||||||
const commonJournal = this.getPaymentReceiveCommonEntry(paymentReceived);
|
|
||||||
|
|
||||||
return {
|
|
||||||
...commonJournal,
|
|
||||||
credit: paymentReceived.amount,
|
|
||||||
accountId: paymentReceived.unearnedRevenueAccountId,
|
|
||||||
accountNormal: AccountNormal.CREDIT,
|
|
||||||
indexGroup: 10,
|
|
||||||
index: 1,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
import { Inject, Service } from 'typedi';
|
|
||||||
import events from '@/subscribers/events';
|
|
||||||
import { AutoApplyUnearnedRevenue } from '../AutoApplyUnearnedRevenue';
|
|
||||||
|
|
||||||
@Service()
|
|
||||||
export class AutoApplyUnearnedRevenueOnInvoiceCreated {
|
|
||||||
@Inject()
|
|
||||||
private autoApplyUnearnedRevenue: AutoApplyUnearnedRevenue;
|
|
||||||
/**
|
|
||||||
* Constructor method.
|
|
||||||
*/
|
|
||||||
public attach(bus) {
|
|
||||||
bus.subscribe(
|
|
||||||
events.saleInvoice.onCreated,
|
|
||||||
this.handleAutoApplyUnearnedRevenueOnInvoiceCreated.bind(this)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles the auto apply unearned revenue on invoice creating.
|
|
||||||
* @param
|
|
||||||
*/
|
|
||||||
private async handleAutoApplyUnearnedRevenueOnInvoiceCreated({
|
|
||||||
tenantId,
|
|
||||||
saleInvoice,
|
|
||||||
trx,
|
|
||||||
}) {
|
|
||||||
await this.autoApplyUnearnedRevenue.autoApplyUnearnedRevenueToInvoice(
|
|
||||||
tenantId,
|
|
||||||
saleInvoice.id,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,20 +3,15 @@ import {
|
|||||||
IPaymentReceiveCreatedPayload,
|
IPaymentReceiveCreatedPayload,
|
||||||
IPaymentReceiveDeletedPayload,
|
IPaymentReceiveDeletedPayload,
|
||||||
IPaymentReceiveEditedPayload,
|
IPaymentReceiveEditedPayload,
|
||||||
PaymentReceiveUnearnedRevenueAppliedEventPayload,
|
|
||||||
} from '@/interfaces';
|
} from '@/interfaces';
|
||||||
import events from '@/subscribers/events';
|
import events from '@/subscribers/events';
|
||||||
import { PaymentReceiveGLEntries } from '@/services/Sales/PaymentReceives/PaymentReceiveGLEntries';
|
import { PaymentReceiveGLEntries } from '@/services/Sales/PaymentReceives/PaymentReceiveGLEntries';
|
||||||
import { PaymentReceivedUnearnedGLEntries } from '@/services/Sales/PaymentReceives/PaymentReceivedUnearnedGLEntries';
|
|
||||||
|
|
||||||
@Service()
|
@Service()
|
||||||
export default class PaymentReceivesWriteGLEntriesSubscriber {
|
export default class PaymentReceivesWriteGLEntriesSubscriber {
|
||||||
@Inject()
|
@Inject()
|
||||||
private paymentReceiveGLEntries: PaymentReceiveGLEntries;
|
private paymentReceiveGLEntries: PaymentReceiveGLEntries;
|
||||||
|
|
||||||
@Inject()
|
|
||||||
private paymentReceivedUnearnedGLEntries: PaymentReceivedUnearnedGLEntries;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Attaches events with handlers.
|
* Attaches events with handlers.
|
||||||
*/
|
*/
|
||||||
@@ -37,7 +32,6 @@ export default class PaymentReceivesWriteGLEntriesSubscriber {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle journal entries writing once the payment receive created.
|
* Handle journal entries writing once the payment receive created.
|
||||||
* @param {IPaymentReceiveCreatedPayload} payload -
|
|
||||||
*/
|
*/
|
||||||
private handleWriteJournalEntriesOnceCreated = async ({
|
private handleWriteJournalEntriesOnceCreated = async ({
|
||||||
tenantId,
|
tenantId,
|
||||||
@@ -49,21 +43,14 @@ export default class PaymentReceivesWriteGLEntriesSubscriber {
|
|||||||
paymentReceiveId,
|
paymentReceiveId,
|
||||||
trx
|
trx
|
||||||
);
|
);
|
||||||
await this.paymentReceivedUnearnedGLEntries.writePaymentGLEntries(
|
|
||||||
tenantId,
|
|
||||||
paymentReceiveId,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handle journal entries writing once the payment receive edited.
|
* Handle journal entries writing once the payment receive edited.
|
||||||
* @param {IPaymentReceiveEditedPayload} payload -
|
|
||||||
*/
|
*/
|
||||||
private handleOverwriteJournalEntriesOnceEdited = async ({
|
private handleOverwriteJournalEntriesOnceEdited = async ({
|
||||||
tenantId,
|
tenantId,
|
||||||
paymentReceive,
|
paymentReceive,
|
||||||
paymentReceiveId,
|
|
||||||
trx,
|
trx,
|
||||||
}: IPaymentReceiveEditedPayload) => {
|
}: IPaymentReceiveEditedPayload) => {
|
||||||
await this.paymentReceiveGLEntries.rewritePaymentGLEntries(
|
await this.paymentReceiveGLEntries.rewritePaymentGLEntries(
|
||||||
@@ -71,16 +58,10 @@ export default class PaymentReceivesWriteGLEntriesSubscriber {
|
|||||||
paymentReceive.id,
|
paymentReceive.id,
|
||||||
trx
|
trx
|
||||||
);
|
);
|
||||||
await this.paymentReceivedUnearnedGLEntries.rewritePaymentGLEntries(
|
|
||||||
tenantId,
|
|
||||||
paymentReceiveId,
|
|
||||||
trx
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles revert journal entries once deleted.
|
* Handles revert journal entries once deleted.
|
||||||
* @param {IPaymentReceiveDeletedPayload} payload -
|
|
||||||
*/
|
*/
|
||||||
private handleRevertJournalEntriesOnceDeleted = async ({
|
private handleRevertJournalEntriesOnceDeleted = async ({
|
||||||
tenantId,
|
tenantId,
|
||||||
|
|||||||
@@ -40,6 +40,13 @@ export default {
|
|||||||
baseCurrencyUpdated: 'onOrganizationBaseCurrencyUpdated',
|
baseCurrencyUpdated: 'onOrganizationBaseCurrencyUpdated',
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* User subscription events.
|
||||||
|
*/
|
||||||
|
subscription: {
|
||||||
|
onSubscribed: 'onOrganizationSubscribed',
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tenants managment service.
|
* Tenants managment service.
|
||||||
*/
|
*/
|
||||||
@@ -142,8 +149,6 @@ export default {
|
|||||||
|
|
||||||
onMailReminderSend: 'onSaleInvoiceMailReminderSend',
|
onMailReminderSend: 'onSaleInvoiceMailReminderSend',
|
||||||
onMailReminderSent: 'onSaleInvoiceMailReminderSent',
|
onMailReminderSent: 'onSaleInvoiceMailReminderSent',
|
||||||
|
|
||||||
onUnearnedRevenueApplied: 'onSaleInvoiceUnearnedRevenue',
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -232,8 +237,6 @@ export default {
|
|||||||
onPreMailSend: 'onPaymentReceivePreMailSend',
|
onPreMailSend: 'onPaymentReceivePreMailSend',
|
||||||
onMailSend: 'onPaymentReceiveMailSend',
|
onMailSend: 'onPaymentReceiveMailSend',
|
||||||
onMailSent: 'onPaymentReceiveMailSent',
|
onMailSent: 'onPaymentReceiveMailSent',
|
||||||
|
|
||||||
onUnearnedRevenueApplied: 'onPaymentReceivedUnearnedRevenue',
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -254,8 +257,6 @@ export default {
|
|||||||
|
|
||||||
onOpening: 'onBillOpening',
|
onOpening: 'onBillOpening',
|
||||||
onOpened: 'onBillOpened',
|
onOpened: 'onBillOpened',
|
||||||
|
|
||||||
onPrepardExpensesApplied: 'onBillPrepardExpensesApplied'
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -273,8 +274,6 @@ export default {
|
|||||||
|
|
||||||
onPublishing: 'onBillPaymentPublishing',
|
onPublishing: 'onBillPaymentPublishing',
|
||||||
onPublished: 'onBillPaymentPublished',
|
onPublished: 'onBillPaymentPublished',
|
||||||
|
|
||||||
onPrepardExpensesApplied: 'onBillPaymentPrepardExpensesApplied'
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import { DashboardPageContent } from '@/components';
|
|||||||
import { transformTableStateToQuery, compose } from '@/utils';
|
import { transformTableStateToQuery, compose } from '@/utils';
|
||||||
|
|
||||||
import { ManualJournalsListProvider } from './ManualJournalsListProvider';
|
import { ManualJournalsListProvider } from './ManualJournalsListProvider';
|
||||||
import ManualJournalsViewTabs from './ManualJournalsViewTabs';
|
|
||||||
import ManualJournalsDataTable from './ManualJournalsDataTable';
|
import ManualJournalsDataTable from './ManualJournalsDataTable';
|
||||||
import ManualJournalsActionsBar from './ManualJournalActionsBar';
|
import ManualJournalsActionsBar from './ManualJournalActionsBar';
|
||||||
import withManualJournals from './withManualJournals';
|
import withManualJournals from './withManualJournals';
|
||||||
@@ -29,7 +28,6 @@ function ManualJournalsTable({
|
|||||||
<ManualJournalsActionsBar />
|
<ManualJournalsActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<ManualJournalsViewTabs />
|
|
||||||
<ManualJournalsDataTable />
|
<ManualJournalsDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</ManualJournalsListProvider>
|
</ManualJournalsListProvider>
|
||||||
|
|||||||
@@ -2,15 +2,15 @@
|
|||||||
import React, { useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
|
|
||||||
import '@/style/pages/Accounts/List.scss';
|
import '@/style/pages/Accounts/List.scss';
|
||||||
import { DashboardPageContent, DashboardContentTable } from '@/components';
|
|
||||||
|
|
||||||
|
import { DashboardPageContent, DashboardContentTable } from '@/components';
|
||||||
import { AccountsChartProvider } from './AccountsChartProvider';
|
import { AccountsChartProvider } from './AccountsChartProvider';
|
||||||
import AccountsViewsTabs from './AccountsViewsTabs';
|
|
||||||
import AccountsActionsBar from './AccountsActionsBar';
|
import AccountsActionsBar from './AccountsActionsBar';
|
||||||
import AccountsDataTable from './AccountsDataTable';
|
import AccountsDataTable from './AccountsDataTable';
|
||||||
|
|
||||||
import withAccounts from '@/containers/Accounts/withAccounts';
|
import withAccounts from '@/containers/Accounts/withAccounts';
|
||||||
import withAccountsTableActions from './withAccountsTableActions';
|
import withAccountsTableActions from './withAccountsTableActions';
|
||||||
|
|
||||||
import { transformAccountsStateToQuery } from './utils';
|
import { transformAccountsStateToQuery } from './utils';
|
||||||
import { compose } from '@/utils';
|
import { compose } from '@/utils';
|
||||||
|
|
||||||
@@ -41,8 +41,6 @@ function AccountsChart({
|
|||||||
<AccountsActionsBar />
|
<AccountsActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<AccountsViewsTabs />
|
|
||||||
|
|
||||||
<DashboardContentTable>
|
<DashboardContentTable>
|
||||||
<AccountsDataTable />
|
<AccountsDataTable />
|
||||||
</DashboardContentTable>
|
</DashboardContentTable>
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import '@/style/pages/Customers/List.scss';
|
|||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
|
|
||||||
import CustomersActionsBar from './CustomersActionsBar';
|
import CustomersActionsBar from './CustomersActionsBar';
|
||||||
import CustomersViewsTabs from './CustomersViewsTabs';
|
|
||||||
import CustomersTable from './CustomersTable';
|
import CustomersTable from './CustomersTable';
|
||||||
import { CustomersListProvider } from './CustomersListProvider';
|
import { CustomersListProvider } from './CustomersListProvider';
|
||||||
|
|
||||||
@@ -42,7 +41,6 @@ function CustomersList({
|
|||||||
<CustomersActionsBar />
|
<CustomersActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<CustomersViewsTabs />
|
|
||||||
<CustomersTable />
|
<CustomersTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</CustomersListProvider>
|
</CustomersListProvider>
|
||||||
|
|||||||
@@ -160,13 +160,6 @@ export function useCustomersTableColumns() {
|
|||||||
width: 85,
|
width: 85,
|
||||||
clickable: true,
|
clickable: true,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
id: 'credit_balance',
|
|
||||||
Header: 'Credit Balance',
|
|
||||||
accessor: 'formatted_unused_credit',
|
|
||||||
width: 100,
|
|
||||||
align: 'right',
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
id: 'balance',
|
id: 'balance',
|
||||||
Header: intl.get('receivable_balance'),
|
Header: intl.get('receivable_balance'),
|
||||||
|
|||||||
@@ -3,15 +3,14 @@ import React from 'react';
|
|||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { Formik } from 'formik';
|
import { Formik } from 'formik';
|
||||||
import { Intent } from '@blueprintjs/core';
|
import { Intent } from '@blueprintjs/core';
|
||||||
import { pick } from 'lodash';
|
import { omit } from 'lodash';
|
||||||
|
|
||||||
import { AppToaster } from '@/components';
|
import { AppToaster } from '@/components';
|
||||||
import { CreateQuickPaymentMadeFormSchema } from './QuickPaymentMade.schema';
|
import { CreateQuickPaymentMadeFormSchema } from './QuickPaymentMade.schema';
|
||||||
import { useQuickPaymentMadeContext } from './QuickPaymentMadeFormProvider';
|
import { useQuickPaymentMadeContext } from './QuickPaymentMadeFormProvider';
|
||||||
import QuickPaymentMadeFormContent from './QuickPaymentMadeFormContent';
|
import QuickPaymentMadeFormContent from './QuickPaymentMadeFormContent';
|
||||||
|
|
||||||
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
||||||
import { defaultPaymentMade, transformErrors } from './utils';
|
import { defaultPaymentMade, transformBillToForm, transformErrors } from './utils';
|
||||||
import { compose } from '@/utils';
|
import { compose } from '@/utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -21,31 +20,24 @@ function QuickPaymentMadeForm({
|
|||||||
// #withDialogActions
|
// #withDialogActions
|
||||||
closeDialog,
|
closeDialog,
|
||||||
}) {
|
}) {
|
||||||
|
const { bill, dialogName, createPaymentMadeMutate } =
|
||||||
|
useQuickPaymentMadeContext();
|
||||||
|
|
||||||
const {
|
// Initial form values.
|
||||||
bill,
|
|
||||||
dialogName,
|
|
||||||
createPaymentMadeMutate,
|
|
||||||
} = useQuickPaymentMadeContext();
|
|
||||||
|
|
||||||
// Initial form values
|
|
||||||
const initialValues = {
|
const initialValues = {
|
||||||
...defaultPaymentMade,
|
...defaultPaymentMade,
|
||||||
...bill,
|
...transformBillToForm(bill),
|
||||||
};
|
};
|
||||||
|
|
||||||
// Handles the form submit.
|
// Handles the form submit.
|
||||||
const handleFormSubmit = (values, { setSubmitting, setFieldError }) => {
|
const handleFormSubmit = (values, { setSubmitting, setFieldError }) => {
|
||||||
const entries = [values]
|
const entries = [
|
||||||
.filter((entry) => entry.id && entry.payment_amount)
|
{
|
||||||
.map((entry) => ({
|
payment_amount: values.amount,
|
||||||
bill_id: entry.id,
|
bill_id: values.bill_id,
|
||||||
...pick(entry, ['payment_amount']),
|
},
|
||||||
}));
|
];
|
||||||
|
|
||||||
const form = {
|
const form = {
|
||||||
...values,
|
...omit(values, ['bill_id']),
|
||||||
vendor_id: values?.vendor?.id,
|
|
||||||
entries,
|
entries,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -124,7 +124,7 @@ function QuickPaymentMadeFormFields({
|
|||||||
</Col>
|
</Col>
|
||||||
</Row>
|
</Row>
|
||||||
{/*------------ Amount Received -----------*/}
|
{/*------------ Amount Received -----------*/}
|
||||||
<FastField name={'payment_amount'}>
|
<FastField name={'amount'}>
|
||||||
{({
|
{({
|
||||||
form: { values, setFieldValue },
|
form: { values, setFieldValue },
|
||||||
field: { value },
|
field: { value },
|
||||||
@@ -135,7 +135,7 @@ function QuickPaymentMadeFormFields({
|
|||||||
labelInfo={<FieldRequiredHint />}
|
labelInfo={<FieldRequiredHint />}
|
||||||
className={classNames('form-group--payment_amount', CLASSES.FILL)}
|
className={classNames('form-group--payment_amount', CLASSES.FILL)}
|
||||||
intent={inputIntent({ error, touched })}
|
intent={inputIntent({ error, touched })}
|
||||||
helperText={<ErrorMessage name="payment_amount" />}
|
helperText={<ErrorMessage name="amount" />}
|
||||||
>
|
>
|
||||||
<ControlGroup>
|
<ControlGroup>
|
||||||
<InputPrependText text={values.currency_code} />
|
<InputPrependText text={values.currency_code} />
|
||||||
@@ -144,7 +144,7 @@ function QuickPaymentMadeFormFields({
|
|||||||
value={value}
|
value={value}
|
||||||
minimal={true}
|
minimal={true}
|
||||||
onChange={(amount) => {
|
onChange={(amount) => {
|
||||||
setFieldValue('payment_amount', amount);
|
setFieldValue('amount', amount);
|
||||||
}}
|
}}
|
||||||
intent={inputIntent({ error, touched })}
|
intent={inputIntent({ error, touched })}
|
||||||
inputRef={(ref) => (paymentMadeFieldRef.current = ref)}
|
inputRef={(ref) => (paymentMadeFieldRef.current = ref)}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import React from 'react';
|
import React, { useMemo } from 'react';
|
||||||
import { DialogContent } from '@/components';
|
import { DialogContent } from '@/components';
|
||||||
import {
|
import {
|
||||||
useBill,
|
useBill,
|
||||||
@@ -11,7 +11,6 @@ import { Features } from '@/constants';
|
|||||||
import { useFeatureCan } from '@/hooks/state';
|
import { useFeatureCan } from '@/hooks/state';
|
||||||
import { pick } from 'lodash';
|
import { pick } from 'lodash';
|
||||||
|
|
||||||
|
|
||||||
const QuickPaymentMadeContext = React.createContext();
|
const QuickPaymentMadeContext = React.createContext();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -40,13 +39,14 @@ function QuickPaymentMadeFormProvider({ query, billId, dialogName, ...props }) {
|
|||||||
isSuccess: isBranchesSuccess,
|
isSuccess: isBranchesSuccess,
|
||||||
} = useBranches(query, { enabled: isBranchFeatureCan });
|
} = useBranches(query, { enabled: isBranchFeatureCan });
|
||||||
|
|
||||||
|
const paymentBill = useMemo(
|
||||||
|
() => pick(bill, ['id', 'due_amount', 'vendor_id', 'currency_code']),
|
||||||
|
[bill],
|
||||||
|
);
|
||||||
|
|
||||||
// State provider.
|
// State provider.
|
||||||
const provider = {
|
const provider = {
|
||||||
bill: {
|
bill: paymentBill,
|
||||||
...pick(bill, ['id', 'due_amount', 'vendor', 'currency_code']),
|
|
||||||
vendor_id: bill?.vendor?.display_name,
|
|
||||||
payment_amount: bill?.due_amount,
|
|
||||||
},
|
|
||||||
accounts,
|
accounts,
|
||||||
branches,
|
branches,
|
||||||
dialogName,
|
dialogName,
|
||||||
|
|||||||
@@ -2,24 +2,25 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { first } from 'lodash';
|
import { first, pick } from 'lodash';
|
||||||
|
import { useFormikContext } from 'formik';
|
||||||
import { Intent } from '@blueprintjs/core';
|
import { Intent } from '@blueprintjs/core';
|
||||||
import { AppToaster } from '@/components';
|
import { AppToaster } from '@/components';
|
||||||
import { useFormikContext } from 'formik';
|
|
||||||
import { useQuickPaymentMadeContext } from './QuickPaymentMadeFormProvider';
|
import { useQuickPaymentMadeContext } from './QuickPaymentMadeFormProvider';
|
||||||
import { PAYMENT_MADE_ERRORS } from '@/containers/Purchases/PaymentMades/constants';
|
import { PAYMENT_MADE_ERRORS } from '@/containers/Purchases/PaymentMades/constants';
|
||||||
|
|
||||||
// Default initial values of payment made.
|
// Default initial values of payment made.
|
||||||
export const defaultPaymentMade = {
|
export const defaultPaymentMade = {
|
||||||
|
bill_id: '',
|
||||||
vendor_id: '',
|
vendor_id: '',
|
||||||
payment_account_id: '',
|
payment_account_id: '',
|
||||||
payment_date: moment(new Date()).format('YYYY-MM-DD'),
|
payment_date: moment(new Date()).format('YYYY-MM-DD'),
|
||||||
reference: '',
|
reference: '',
|
||||||
payment_number: '',
|
payment_number: '',
|
||||||
|
amount: '',
|
||||||
// statement: '',
|
// statement: '',
|
||||||
exchange_rate: 1,
|
exchange_rate: 1,
|
||||||
branch_id: '',
|
branch_id: '',
|
||||||
entries: [{ bill_id: '', payment_amount: '' }],
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const transformErrors = (errors, { setFieldError }) => {
|
export const transformErrors = (errors, { setFieldError }) => {
|
||||||
@@ -58,3 +59,11 @@ export const useSetPrimaryBranchToForm = () => {
|
|||||||
}
|
}
|
||||||
}, [isBranchesSuccess, setFieldValue, branches]);
|
}, [isBranchesSuccess, setFieldValue, branches]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const transformBillToForm = (bill) => {
|
||||||
|
return {
|
||||||
|
...pick(bill, ['vendor_id', 'currency_code']),
|
||||||
|
amount: bill.due_amount,
|
||||||
|
bill_id: bill.id,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import React from 'react';
|
|||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { Formik } from 'formik';
|
import { Formik } from 'formik';
|
||||||
import { Intent } from '@blueprintjs/core';
|
import { Intent } from '@blueprintjs/core';
|
||||||
import { pick, defaultTo, omit } from 'lodash';
|
import { defaultTo, omit } from 'lodash';
|
||||||
|
|
||||||
import { AppToaster } from '@/components';
|
import { AppToaster } from '@/components';
|
||||||
import { useQuickPaymentReceiveContext } from './QuickPaymentReceiveFormProvider';
|
import { useQuickPaymentReceiveContext } from './QuickPaymentReceiveFormProvider';
|
||||||
@@ -12,7 +12,11 @@ import QuickPaymentReceiveFormContent from './QuickPaymentReceiveFormContent';
|
|||||||
|
|
||||||
import withSettings from '@/containers/Settings/withSettings';
|
import withSettings from '@/containers/Settings/withSettings';
|
||||||
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
||||||
import { defaultInitialValues, transformErrors } from './utils';
|
import {
|
||||||
|
defaultInitialValues,
|
||||||
|
transformErrors,
|
||||||
|
transformInvoiceToForm,
|
||||||
|
} from './utils';
|
||||||
import { compose, transactionNumber } from '@/utils';
|
import { compose, transactionNumber } from '@/utils';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -26,14 +30,10 @@ function QuickPaymentReceiveForm({
|
|||||||
paymentReceiveAutoIncrement,
|
paymentReceiveAutoIncrement,
|
||||||
paymentReceiveNumberPrefix,
|
paymentReceiveNumberPrefix,
|
||||||
paymentReceiveNextNumber,
|
paymentReceiveNextNumber,
|
||||||
preferredDepositAccount
|
preferredDepositAccount,
|
||||||
}) {
|
}) {
|
||||||
|
const { dialogName, invoice, createPaymentReceiveMutate } =
|
||||||
const {
|
useQuickPaymentReceiveContext();
|
||||||
dialogName,
|
|
||||||
invoice,
|
|
||||||
createPaymentReceiveMutate,
|
|
||||||
} = useQuickPaymentReceiveContext();
|
|
||||||
|
|
||||||
// Payment receive number.
|
// Payment receive number.
|
||||||
const nextPaymentNumber = transactionNumber(
|
const nextPaymentNumber = transactionNumber(
|
||||||
@@ -48,24 +48,22 @@ function QuickPaymentReceiveForm({
|
|||||||
payment_receive_no: nextPaymentNumber,
|
payment_receive_no: nextPaymentNumber,
|
||||||
}),
|
}),
|
||||||
deposit_account_id: defaultTo(preferredDepositAccount, ''),
|
deposit_account_id: defaultTo(preferredDepositAccount, ''),
|
||||||
...invoice,
|
...transformInvoiceToForm(invoice),
|
||||||
};
|
};
|
||||||
|
|
||||||
// Handles the form submit.
|
// Handles the form submit.
|
||||||
const handleFormSubmit = (values, { setSubmitting, setFieldError }) => {
|
const handleFormSubmit = (values, { setSubmitting, setFieldError }) => {
|
||||||
const entries = [values]
|
const entries = [
|
||||||
.filter((entry) => entry.id && entry.payment_amount)
|
{
|
||||||
.map((entry) => ({
|
invoice_id: values.invoice_id,
|
||||||
invoice_id: entry.id,
|
payment_amount: values.amount,
|
||||||
...pick(entry, ['payment_amount']),
|
},
|
||||||
}));
|
];
|
||||||
|
|
||||||
const form = {
|
const form = {
|
||||||
...omit(values, ['payment_receive_no']),
|
...omit(values, ['payment_receive_no', 'invoice_id']),
|
||||||
...(!paymentReceiveAutoIncrement && {
|
...(!paymentReceiveAutoIncrement && {
|
||||||
payment_receive_no: values.payment_receive_no,
|
payment_receive_no: values.payment_receive_no,
|
||||||
}),
|
}),
|
||||||
customer_id: values.customer.id,
|
|
||||||
entries,
|
entries,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -128,7 +128,7 @@ function QuickPaymentReceiveFormFields({
|
|||||||
</Col>
|
</Col>
|
||||||
</Row>
|
</Row>
|
||||||
{/*------------ Amount Received -----------*/}
|
{/*------------ Amount Received -----------*/}
|
||||||
<FastField name={'payment_amount'}>
|
<FastField name={'amount'}>
|
||||||
{({
|
{({
|
||||||
form: { values, setFieldValue },
|
form: { values, setFieldValue },
|
||||||
field: { value },
|
field: { value },
|
||||||
@@ -139,7 +139,7 @@ function QuickPaymentReceiveFormFields({
|
|||||||
labelInfo={<FieldRequiredHint />}
|
labelInfo={<FieldRequiredHint />}
|
||||||
className={classNames('form-group--payment_amount', CLASSES.FILL)}
|
className={classNames('form-group--payment_amount', CLASSES.FILL)}
|
||||||
intent={inputIntent({ error, touched })}
|
intent={inputIntent({ error, touched })}
|
||||||
helperText={<ErrorMessage name="payment_amount" />}
|
helperText={<ErrorMessage name="amount" />}
|
||||||
>
|
>
|
||||||
<ControlGroup>
|
<ControlGroup>
|
||||||
<InputPrependText text={values.currency_code} />
|
<InputPrependText text={values.currency_code} />
|
||||||
@@ -148,7 +148,7 @@ function QuickPaymentReceiveFormFields({
|
|||||||
value={value}
|
value={value}
|
||||||
minimal={true}
|
minimal={true}
|
||||||
onChange={(amount) => {
|
onChange={(amount) => {
|
||||||
setFieldValue('payment_amount', amount);
|
setFieldValue('amount', amount);
|
||||||
}}
|
}}
|
||||||
intent={inputIntent({ error, touched })}
|
intent={inputIntent({ error, touched })}
|
||||||
inputRef={(ref) => (paymentReceiveFieldRef.current = ref)}
|
inputRef={(ref) => (paymentReceiveFieldRef.current = ref)}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import React, { useContext, createContext } from 'react';
|
import React, { useContext, createContext, useMemo } from 'react';
|
||||||
import { pick } from 'lodash';
|
import { pick } from 'lodash';
|
||||||
import { DialogContent } from '@/components';
|
import { DialogContent } from '@/components';
|
||||||
import { Features } from '@/constants';
|
import { Features } from '@/constants';
|
||||||
@@ -47,15 +47,16 @@ function QuickPaymentReceiveFormProvider({
|
|||||||
isSuccess: isBranchesSuccess,
|
isSuccess: isBranchesSuccess,
|
||||||
} = useBranches(query, { enabled: isBranchFeatureCan });
|
} = useBranches(query, { enabled: isBranchFeatureCan });
|
||||||
|
|
||||||
|
const invoicePayment = useMemo(
|
||||||
|
() => pick(invoice, ['id', 'due_amount', 'customer_id', 'currency_code']),
|
||||||
|
[invoice],
|
||||||
|
);
|
||||||
|
|
||||||
// State provider.
|
// State provider.
|
||||||
const provider = {
|
const provider = {
|
||||||
accounts,
|
accounts,
|
||||||
branches,
|
branches,
|
||||||
invoice: {
|
invoice: invoicePayment,
|
||||||
...pick(invoice, ['id', 'due_amount', 'customer', 'currency_code']),
|
|
||||||
customer_id: invoice?.customer?.display_name,
|
|
||||||
payment_amount: invoice.due_amount,
|
|
||||||
},
|
|
||||||
isAccountsLoading,
|
isAccountsLoading,
|
||||||
isSettingsLoading,
|
isSettingsLoading,
|
||||||
isBranchesSuccess,
|
isBranchesSuccess,
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import moment from 'moment';
|
import moment from 'moment';
|
||||||
import intl from 'react-intl-universal';
|
import intl from 'react-intl-universal';
|
||||||
import { first } from 'lodash';
|
import { first, pick } from 'lodash';
|
||||||
import { Intent } from '@blueprintjs/core';
|
import { Intent } from '@blueprintjs/core';
|
||||||
import { AppToaster } from '@/components';
|
import { AppToaster } from '@/components';
|
||||||
|
|
||||||
@@ -10,15 +10,16 @@ import { useFormikContext } from 'formik';
|
|||||||
import { useQuickPaymentReceiveContext } from './QuickPaymentReceiveFormProvider';
|
import { useQuickPaymentReceiveContext } from './QuickPaymentReceiveFormProvider';
|
||||||
|
|
||||||
export const defaultInitialValues = {
|
export const defaultInitialValues = {
|
||||||
|
invoice_id: '',
|
||||||
customer_id: '',
|
customer_id: '',
|
||||||
deposit_account_id: '',
|
deposit_account_id: '',
|
||||||
payment_receive_no: '',
|
payment_receive_no: '',
|
||||||
payment_date: moment(new Date()).format('YYYY-MM-DD'),
|
payment_date: moment(new Date()).format('YYYY-MM-DD'),
|
||||||
reference_no: '',
|
reference_no: '',
|
||||||
|
amount: '',
|
||||||
// statement: '',
|
// statement: '',
|
||||||
exchange_rate: 1,
|
exchange_rate: 1,
|
||||||
branch_id: '',
|
branch_id: '',
|
||||||
entries: [{ invoice_id: '', payment_amount: '' }],
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const transformErrors = (errors, { setFieldError }) => {
|
export const transformErrors = (errors, { setFieldError }) => {
|
||||||
@@ -44,7 +45,9 @@ export const transformErrors = (errors, { setFieldError }) => {
|
|||||||
}
|
}
|
||||||
if (getError('PAYMENT_ACCOUNT_CURRENCY_INVALID')) {
|
if (getError('PAYMENT_ACCOUNT_CURRENCY_INVALID')) {
|
||||||
AppToaster.show({
|
AppToaster.show({
|
||||||
message: intl.get('payment_Receive.error.payment_account_currency_invalid'),
|
message: intl.get(
|
||||||
|
'payment_Receive.error.payment_account_currency_invalid',
|
||||||
|
),
|
||||||
intent: Intent.DANGER,
|
intent: Intent.DANGER,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -64,3 +67,11 @@ export const useSetPrimaryBranchToForm = () => {
|
|||||||
}
|
}
|
||||||
}, [isBranchesSuccess, setFieldValue, branches]);
|
}, [isBranchesSuccess, setFieldValue, branches]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const transformInvoiceToForm = (invoice) => {
|
||||||
|
return {
|
||||||
|
...pick(invoice, ['customer_id', 'currency_code']),
|
||||||
|
amount: invoice.due_amount,
|
||||||
|
invoice_id: invoice.id,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import '@/style/pages/Expense/List.scss';
|
|||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
|
|
||||||
import ExpenseActionsBar from './ExpenseActionsBar';
|
import ExpenseActionsBar from './ExpenseActionsBar';
|
||||||
import ExpenseViewTabs from './ExpenseViewTabs';
|
|
||||||
import ExpenseDataTable from './ExpenseDataTable';
|
import ExpenseDataTable from './ExpenseDataTable';
|
||||||
|
|
||||||
import withExpenses from './withExpenses';
|
import withExpenses from './withExpenses';
|
||||||
@@ -42,7 +41,6 @@ function ExpensesList({
|
|||||||
<ExpenseActionsBar />
|
<ExpenseActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<ExpenseViewTabs />
|
|
||||||
<ExpenseDataTable />
|
<ExpenseDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</ExpensesListProvider>
|
</ExpensesListProvider>
|
||||||
|
|||||||
@@ -8,7 +8,6 @@ import { DashboardPageContent } from '@/components';
|
|||||||
import { ItemsListProvider } from './ItemsListProvider';
|
import { ItemsListProvider } from './ItemsListProvider';
|
||||||
|
|
||||||
import ItemsActionsBar from './ItemsActionsBar';
|
import ItemsActionsBar from './ItemsActionsBar';
|
||||||
import ItemsViewsTabs from './ItemsViewsTabs';
|
|
||||||
import ItemsDataTable from './ItemsDataTable';
|
import ItemsDataTable from './ItemsDataTable';
|
||||||
|
|
||||||
import withItems from './withItems';
|
import withItems from './withItems';
|
||||||
@@ -41,7 +40,6 @@ function ItemsList({
|
|||||||
<ItemsActionsBar />
|
<ItemsActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<ItemsViewsTabs />
|
|
||||||
<ItemsDataTable />
|
<ItemsDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</ItemsListProvider>
|
</ItemsListProvider>
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import '@/style/pages/Bills/List.scss';
|
|||||||
import { BillsListProvider } from './BillsListProvider';
|
import { BillsListProvider } from './BillsListProvider';
|
||||||
|
|
||||||
import BillsActionsBar from './BillsActionsBar';
|
import BillsActionsBar from './BillsActionsBar';
|
||||||
import BillsViewsTabs from './BillsViewsTabs';
|
|
||||||
import BillsTable from './BillsTable';
|
import BillsTable from './BillsTable';
|
||||||
|
|
||||||
import withBills from './withBills';
|
import withBills from './withBills';
|
||||||
@@ -42,7 +41,6 @@ function BillsList({
|
|||||||
<BillsActionsBar />
|
<BillsActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<BillsViewsTabs />
|
|
||||||
<BillsTable />
|
<BillsTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</BillsListProvider>
|
</BillsListProvider>
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import '@/style/pages/VendorsCreditNote/List.scss';
|
|||||||
|
|
||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
import VendorsCreditNoteActionsBar from './VendorsCreditNoteActionsBar';
|
import VendorsCreditNoteActionsBar from './VendorsCreditNoteActionsBar';
|
||||||
import VendorsCreditNoteViewTabs from './VendorsCreditNoteViewTabs';
|
|
||||||
import VendorsCreditNoteDataTable from './VendorsCreditNoteDataTable';
|
import VendorsCreditNoteDataTable from './VendorsCreditNoteDataTable';
|
||||||
|
|
||||||
import withVendorsCreditNotes from './withVendorsCreditNotes';
|
import withVendorsCreditNotes from './withVendorsCreditNotes';
|
||||||
@@ -37,7 +36,6 @@ function VendorsCreditNotesList({
|
|||||||
>
|
>
|
||||||
<VendorsCreditNoteActionsBar />
|
<VendorsCreditNoteActionsBar />
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<VendorsCreditNoteViewTabs />
|
|
||||||
<VendorsCreditNoteDataTable />
|
<VendorsCreditNoteDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</VendorsCreditNoteListProvider>
|
</VendorsCreditNoteListProvider>
|
||||||
|
|||||||
@@ -1,32 +1,16 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import * as R from 'ramda';
|
import * as R from 'ramda';
|
||||||
import * as Yup from 'yup';
|
import React from 'react';
|
||||||
import React, { useMemo } from 'react';
|
|
||||||
import { Button, Classes, Intent } from '@blueprintjs/core';
|
import { Button, Classes, Intent } from '@blueprintjs/core';
|
||||||
import { Form, Formik, FormikHelpers, useFormikContext } from 'formik';
|
import { Form, Formik, FormikHelpers, useFormikContext } from 'formik';
|
||||||
import { AccountsSelect, FFormGroup, FormatNumber } from '@/components';
|
import { FormatNumber } from '@/components';
|
||||||
import { usePaymentMadeFormContext } from '../../PaymentMadeFormProvider';
|
import { usePaymentMadeFormContext } from '../../PaymentMadeFormProvider';
|
||||||
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
||||||
import { ACCOUNT_TYPE } from '@/constants';
|
|
||||||
import { usePaymentMadeExcessAmount } from '../../utils';
|
import { usePaymentMadeExcessAmount } from '../../utils';
|
||||||
|
|
||||||
interface ExcessPaymentValues {
|
interface ExcessPaymentValues {}
|
||||||
accountId: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialValues = {
|
|
||||||
accountId: '',
|
|
||||||
} as ExcessPaymentValues;
|
|
||||||
|
|
||||||
const Schema = Yup.object().shape({
|
|
||||||
accountId: Yup.number().required(),
|
|
||||||
});
|
|
||||||
|
|
||||||
const DEFAULT_ACCOUNT_SLUG = 'prepaid-expenses';
|
|
||||||
|
|
||||||
function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
||||||
const {
|
const {
|
||||||
setFieldValue,
|
|
||||||
submitForm,
|
submitForm,
|
||||||
values: { currency_code: currencyCode },
|
values: { currency_code: currencyCode },
|
||||||
} = useFormikContext();
|
} = useFormikContext();
|
||||||
@@ -37,7 +21,6 @@ function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
|||||||
values: ExcessPaymentValues,
|
values: ExcessPaymentValues,
|
||||||
{ setSubmitting }: FormikHelpers<ExcessPaymentValues>,
|
{ setSubmitting }: FormikHelpers<ExcessPaymentValues>,
|
||||||
) => {
|
) => {
|
||||||
setFieldValue(values.accountId);
|
|
||||||
setSubmitting(true);
|
setSubmitting(true);
|
||||||
setIsExcessConfirmed(true);
|
setIsExcessConfirmed(true);
|
||||||
|
|
||||||
@@ -50,20 +33,10 @@ function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
|||||||
const handleCloseBtn = () => {
|
const handleCloseBtn = () => {
|
||||||
closeDialog(dialogName);
|
closeDialog(dialogName);
|
||||||
};
|
};
|
||||||
// Retrieves the default excess account id.
|
|
||||||
const defaultAccountId = useDefaultExcessPaymentDeposit();
|
|
||||||
|
|
||||||
const excessAmount = usePaymentMadeExcessAmount();
|
const excessAmount = usePaymentMadeExcessAmount();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Formik
|
<Formik initialValues={{}} onSubmit={handleSubmit}>
|
||||||
initialValues={{
|
|
||||||
...initialValues,
|
|
||||||
accountId: defaultAccountId,
|
|
||||||
}}
|
|
||||||
validationSchema={Schema}
|
|
||||||
onSubmit={handleSubmit}
|
|
||||||
>
|
|
||||||
<Form>
|
<Form>
|
||||||
<ExcessPaymentDialogContentForm
|
<ExcessPaymentDialogContentForm
|
||||||
excessAmount={
|
excessAmount={
|
||||||
@@ -90,7 +63,6 @@ function ExcessPaymentDialogContentForm({
|
|||||||
onClose,
|
onClose,
|
||||||
}: ExcessPaymentDialogContentFormProps) {
|
}: ExcessPaymentDialogContentFormProps) {
|
||||||
const { submitForm, isSubmitting } = useFormikContext();
|
const { submitForm, isSubmitting } = useFormikContext();
|
||||||
const { accounts } = usePaymentMadeFormContext();
|
|
||||||
|
|
||||||
const handleCloseBtn = () => {
|
const handleCloseBtn = () => {
|
||||||
onClose && onClose();
|
onClose && onClose();
|
||||||
@@ -100,26 +72,8 @@ function ExcessPaymentDialogContentForm({
|
|||||||
<div className={Classes.DIALOG_BODY}>
|
<div className={Classes.DIALOG_BODY}>
|
||||||
<p style={{ marginBottom: 20 }}>
|
<p style={{ marginBottom: 20 }}>
|
||||||
Would you like to record the excess amount of{' '}
|
Would you like to record the excess amount of{' '}
|
||||||
<strong>{excessAmount}</strong> as advanced payment from the customer.
|
<strong>{excessAmount}</strong> as credit payment from the vendor.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<FFormGroup
|
|
||||||
name={'accountId'}
|
|
||||||
label={'The excessed amount will be deposited in the'}
|
|
||||||
helperText={
|
|
||||||
'Only other current asset and non current asset accounts will show.'
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<AccountsSelect
|
|
||||||
name={'accountId'}
|
|
||||||
items={accounts}
|
|
||||||
buttonProps={{ small: true }}
|
|
||||||
filterByTypes={[
|
|
||||||
ACCOUNT_TYPE.OTHER_CURRENT_ASSET,
|
|
||||||
ACCOUNT_TYPE.NON_CURRENT_ASSET,
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
</FFormGroup>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className={Classes.DIALOG_FOOTER}>
|
<div className={Classes.DIALOG_FOOTER}>
|
||||||
@@ -129,7 +83,7 @@ function ExcessPaymentDialogContentForm({
|
|||||||
loading={isSubmitting}
|
loading={isSubmitting}
|
||||||
onClick={() => submitForm()}
|
onClick={() => submitForm()}
|
||||||
>
|
>
|
||||||
Save Payment
|
Save Payment as Credit
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={handleCloseBtn}>Cancel</Button>
|
<Button onClick={handleCloseBtn}>Cancel</Button>
|
||||||
</div>
|
</div>
|
||||||
@@ -137,11 +91,3 @@ function ExcessPaymentDialogContentForm({
|
|||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const useDefaultExcessPaymentDeposit = () => {
|
|
||||||
const { accounts } = usePaymentMadeFormContext();
|
|
||||||
return useMemo(() => {
|
|
||||||
return accounts?.find((a) => a.slug === DEFAULT_ACCOUNT_SLUG)?.id;
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, []);
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import { DashboardPageContent } from '@/components';
|
|||||||
import { PaymentMadesListProvider } from './PaymentMadesListProvider';
|
import { PaymentMadesListProvider } from './PaymentMadesListProvider';
|
||||||
import PaymentMadeActionsBar from './PaymentMadeActionsBar';
|
import PaymentMadeActionsBar from './PaymentMadeActionsBar';
|
||||||
import PaymentMadesTable from './PaymentMadesTable';
|
import PaymentMadesTable from './PaymentMadesTable';
|
||||||
import PaymentMadeViewTabs from './PaymentMadeViewTabs';
|
|
||||||
|
|
||||||
import withPaymentMades from './withPaymentMade';
|
import withPaymentMades from './withPaymentMade';
|
||||||
import withPaymentMadeActions from './withPaymentMadeActions';
|
import withPaymentMadeActions from './withPaymentMadeActions';
|
||||||
@@ -41,7 +40,6 @@ function PaymentMadeList({
|
|||||||
<PaymentMadeActionsBar />
|
<PaymentMadeActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<PaymentMadeViewTabs />
|
|
||||||
<PaymentMadesTable />
|
<PaymentMadesTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</PaymentMadesListProvider>
|
</PaymentMadesListProvider>
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import '@/style/pages/CreditNote/List.scss';
|
|||||||
|
|
||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
import CreditNotesActionsBar from './CreditNotesActionsBar';
|
import CreditNotesActionsBar from './CreditNotesActionsBar';
|
||||||
import CreditNotesViewTabs from './CreditNotesViewTabs';
|
|
||||||
import CreditNotesDataTable from './CreditNotesDataTable';
|
import CreditNotesDataTable from './CreditNotesDataTable';
|
||||||
|
|
||||||
import withCreditNotes from './withCreditNotes';
|
import withCreditNotes from './withCreditNotes';
|
||||||
@@ -36,8 +35,8 @@ function CreditNotesList({
|
|||||||
tableStateChanged={creditNoteTableStateChanged}
|
tableStateChanged={creditNoteTableStateChanged}
|
||||||
>
|
>
|
||||||
<CreditNotesActionsBar />
|
<CreditNotesActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<CreditNotesViewTabs />
|
|
||||||
<CreditNotesDataTable />
|
<CreditNotesDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</CreditNotesListProvider>
|
</CreditNotesListProvider>
|
||||||
|
|||||||
@@ -1,11 +1,10 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { DashboardContentTable, DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
|
|
||||||
import '@/style/pages/SaleEstimate/List.scss';
|
import '@/style/pages/SaleEstimate/List.scss';
|
||||||
|
|
||||||
import EstimatesActionsBar from './EstimatesActionsBar';
|
import EstimatesActionsBar from './EstimatesActionsBar';
|
||||||
import EstimatesViewTabs from './EstimatesViewTabs';
|
|
||||||
import EstimatesDataTable from './EstimatesDataTable';
|
import EstimatesDataTable from './EstimatesDataTable';
|
||||||
|
|
||||||
import withEstimates from './withEstimates';
|
import withEstimates from './withEstimates';
|
||||||
@@ -41,7 +40,6 @@ function EstimatesList({
|
|||||||
<EstimatesActionsBar />
|
<EstimatesActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<EstimatesViewTabs />
|
|
||||||
<EstimatesDataTable />
|
<EstimatesDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</EstimatesListProvider>
|
</EstimatesListProvider>
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import '@/style/pages/SaleInvoice/List.scss';
|
|||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
import { InvoicesListProvider } from './InvoicesListProvider';
|
import { InvoicesListProvider } from './InvoicesListProvider';
|
||||||
|
|
||||||
import InvoiceViewTabs from './InvoiceViewTabs';
|
|
||||||
import InvoicesDataTable from './InvoicesDataTable';
|
import InvoicesDataTable from './InvoicesDataTable';
|
||||||
import InvoicesActionsBar from './InvoicesActionsBar';
|
import InvoicesActionsBar from './InvoicesActionsBar';
|
||||||
|
|
||||||
@@ -43,7 +42,6 @@ function InvoicesList({
|
|||||||
<InvoicesActionsBar />
|
<InvoicesActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<InvoiceViewTabs />
|
|
||||||
<InvoicesDataTable />
|
<InvoicesDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</InvoicesListProvider>
|
</InvoicesListProvider>
|
||||||
|
|||||||
@@ -1,37 +1,21 @@
|
|||||||
// @ts-nocheck
|
// @ts-nocheck
|
||||||
import { useMemo } from 'react';
|
|
||||||
import * as Yup from 'yup';
|
import * as Yup from 'yup';
|
||||||
import * as R from 'ramda';
|
import * as R from 'ramda';
|
||||||
import { Button, Classes, Intent } from '@blueprintjs/core';
|
import { Button, Classes, Intent } from '@blueprintjs/core';
|
||||||
import { Form, Formik, FormikHelpers, useFormikContext } from 'formik';
|
import { Form, Formik, FormikHelpers, useFormikContext } from 'formik';
|
||||||
import { AccountsSelect, FFormGroup, FormatNumber } from '@/components';
|
import { FormatNumber } from '@/components';
|
||||||
import { usePaymentReceiveFormContext } from '../../PaymentReceiveFormProvider';
|
import { usePaymentReceiveFormContext } from '../../PaymentReceiveFormProvider';
|
||||||
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
import withDialogActions from '@/containers/Dialog/withDialogActions';
|
||||||
import { usePaymentReceivedTotalExceededAmount } from '../../utils';
|
import { usePaymentReceivedTotalExceededAmount } from '../../utils';
|
||||||
import { ACCOUNT_TYPE } from '@/constants';
|
|
||||||
|
|
||||||
interface ExcessPaymentValues {
|
interface ExcessPaymentValues {}
|
||||||
accountId: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialValues = {
|
|
||||||
accountId: '',
|
|
||||||
} as ExcessPaymentValues;
|
|
||||||
|
|
||||||
const Schema = Yup.object().shape({
|
|
||||||
accountId: Yup.number().required(),
|
|
||||||
});
|
|
||||||
|
|
||||||
const DEFAULT_ACCOUNT_SLUG = 'unearned-revenue';
|
|
||||||
|
|
||||||
export function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
export function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
||||||
const {
|
const {
|
||||||
setFieldValue,
|
|
||||||
submitForm,
|
submitForm,
|
||||||
values: { currency_code: currencyCode },
|
values: { currency_code: currencyCode },
|
||||||
} = useFormikContext();
|
} = useFormikContext();
|
||||||
const { setIsExcessConfirmed } = usePaymentReceiveFormContext();
|
const { setIsExcessConfirmed } = usePaymentReceiveFormContext();
|
||||||
const initialAccountId = useDefaultExcessPaymentDeposit();
|
|
||||||
const exceededAmount = usePaymentReceivedTotalExceededAmount();
|
const exceededAmount = usePaymentReceivedTotalExceededAmount();
|
||||||
|
|
||||||
const handleSubmit = (
|
const handleSubmit = (
|
||||||
@@ -40,7 +24,6 @@ export function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
|||||||
) => {
|
) => {
|
||||||
setSubmitting(true);
|
setSubmitting(true);
|
||||||
setIsExcessConfirmed(true);
|
setIsExcessConfirmed(true);
|
||||||
setFieldValue('unearned_revenue_account_id', values.accountId);
|
|
||||||
|
|
||||||
submitForm().then(() => {
|
submitForm().then(() => {
|
||||||
closeDialog(dialogName);
|
closeDialog(dialogName);
|
||||||
@@ -52,14 +35,7 @@ export function ExcessPaymentDialogContentRoot({ dialogName, closeDialog }) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Formik
|
<Formik initialValues={{}} onSubmit={handleSubmit}>
|
||||||
initialValues={{
|
|
||||||
...initialValues,
|
|
||||||
accountId: initialAccountId,
|
|
||||||
}}
|
|
||||||
validationSchema={Schema}
|
|
||||||
onSubmit={handleSubmit}
|
|
||||||
>
|
|
||||||
<Form>
|
<Form>
|
||||||
<ExcessPaymentDialogContentForm
|
<ExcessPaymentDialogContentForm
|
||||||
exceededAmount={
|
exceededAmount={
|
||||||
@@ -77,7 +53,6 @@ export const ExcessPaymentDialogContent = R.compose(withDialogActions)(
|
|||||||
);
|
);
|
||||||
|
|
||||||
function ExcessPaymentDialogContentForm({ onClose, exceededAmount }) {
|
function ExcessPaymentDialogContentForm({ onClose, exceededAmount }) {
|
||||||
const { accounts } = usePaymentReceiveFormContext();
|
|
||||||
const { submitForm, isSubmitting } = useFormikContext();
|
const { submitForm, isSubmitting } = useFormikContext();
|
||||||
|
|
||||||
const handleCloseBtn = () => {
|
const handleCloseBtn = () => {
|
||||||
@@ -89,27 +64,8 @@ function ExcessPaymentDialogContentForm({ onClose, exceededAmount }) {
|
|||||||
<div className={Classes.DIALOG_BODY}>
|
<div className={Classes.DIALOG_BODY}>
|
||||||
<p style={{ marginBottom: 20 }}>
|
<p style={{ marginBottom: 20 }}>
|
||||||
Would you like to record the excess amount of{' '}
|
Would you like to record the excess amount of{' '}
|
||||||
<strong>{exceededAmount}</strong> as advanced payment from the
|
<strong>{exceededAmount}</strong> as credit payment from the customer.
|
||||||
customer.
|
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<FFormGroup
|
|
||||||
name={'accountId'}
|
|
||||||
label={'The excessed amount will be deposited in the'}
|
|
||||||
helperText={
|
|
||||||
'Only other other current liability and non current liability accounts will show.'
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<AccountsSelect
|
|
||||||
name={'accountId'}
|
|
||||||
items={accounts}
|
|
||||||
buttonProps={{ small: true }}
|
|
||||||
filterByTypes={[
|
|
||||||
ACCOUNT_TYPE.OTHER_CURRENT_LIABILITY,
|
|
||||||
ACCOUNT_TYPE.NON_CURRENT_LIABILITY,
|
|
||||||
]}
|
|
||||||
/>
|
|
||||||
</FFormGroup>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className={Classes.DIALOG_FOOTER}>
|
<div className={Classes.DIALOG_FOOTER}>
|
||||||
@@ -120,7 +76,7 @@ function ExcessPaymentDialogContentForm({ onClose, exceededAmount }) {
|
|||||||
disabled={isSubmitting}
|
disabled={isSubmitting}
|
||||||
onClick={() => submitForm()}
|
onClick={() => submitForm()}
|
||||||
>
|
>
|
||||||
Save Payment
|
Save Payment as Credit
|
||||||
</Button>
|
</Button>
|
||||||
<Button onClick={handleCloseBtn}>Cancel</Button>
|
<Button onClick={handleCloseBtn}>Cancel</Button>
|
||||||
</div>
|
</div>
|
||||||
@@ -128,11 +84,3 @@ function ExcessPaymentDialogContentForm({ onClose, exceededAmount }) {
|
|||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
const useDefaultExcessPaymentDeposit = () => {
|
|
||||||
const { accounts } = usePaymentReceiveFormContext();
|
|
||||||
return useMemo(() => {
|
|
||||||
return accounts?.find((a) => a.slug === DEFAULT_ACCOUNT_SLUG)?.id;
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, []);
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -48,7 +48,6 @@ export const defaultPaymentReceive = {
|
|||||||
exchange_rate: 1,
|
exchange_rate: 1,
|
||||||
entries: [],
|
entries: [],
|
||||||
attachments: [],
|
attachments: [],
|
||||||
unearned_revenue_account_id: '',
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const defaultRequestPaymentEntry = {
|
export const defaultRequestPaymentEntry = {
|
||||||
@@ -299,10 +298,9 @@ export const resetFormState = ({ initialValues, values, resetForm }) => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
export const getExceededAmountFromValues = (values) => {
|
export const getExceededAmountFromValues = (values) => {
|
||||||
const totalApplied = sumBy(values.entries, 'payment_amount');
|
const totalApplied = sumBy(values.entries, 'payment_amount');
|
||||||
const totalAmount = values.amount;
|
const totalAmount = values.amount;
|
||||||
|
|
||||||
return totalAmount - totalApplied;
|
return totalAmount - totalApplied;
|
||||||
}
|
};
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import '@/style/pages/PaymentReceive/List.scss';
|
|||||||
|
|
||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
import { PaymentReceivesListProvider } from './PaymentReceiptsListProvider';
|
import { PaymentReceivesListProvider } from './PaymentReceiptsListProvider';
|
||||||
import PaymentReceiveViewTabs from './PaymentReceiveViewTabs';
|
|
||||||
import PaymentReceivesTable from './PaymentReceivesTable';
|
import PaymentReceivesTable from './PaymentReceivesTable';
|
||||||
import PaymentReceiveActionsBar from './PaymentReceiveActionsBar';
|
import PaymentReceiveActionsBar from './PaymentReceiveActionsBar';
|
||||||
|
|
||||||
@@ -41,7 +40,6 @@ function PaymentReceiveList({
|
|||||||
<PaymentReceiveActionsBar />
|
<PaymentReceiveActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<PaymentReceiveViewTabs />
|
|
||||||
<PaymentReceivesTable />
|
<PaymentReceivesTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</PaymentReceivesListProvider>
|
</PaymentReceivesListProvider>
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import { DashboardPageContent } from '@/components';
|
|||||||
|
|
||||||
import { VendorsListProvider } from './VendorsListProvider';
|
import { VendorsListProvider } from './VendorsListProvider';
|
||||||
import VendorActionsBar from './VendorActionsBar';
|
import VendorActionsBar from './VendorActionsBar';
|
||||||
import VendorViewsTabs from './VendorViewsTabs';
|
|
||||||
import VendorsTable from './VendorsTable';
|
import VendorsTable from './VendorsTable';
|
||||||
|
|
||||||
import withVendors from './withVendors';
|
import withVendors from './withVendors';
|
||||||
@@ -42,7 +41,6 @@ function VendorsList({
|
|||||||
<VendorActionsBar />
|
<VendorActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<VendorViewsTabs />
|
|
||||||
<VendorsTable />
|
<VendorsTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</VendorsListProvider>
|
</VendorsListProvider>
|
||||||
|
|||||||
@@ -183,13 +183,6 @@ export function useVendorsTableColumns() {
|
|||||||
width: 85,
|
width: 85,
|
||||||
clickable: true,
|
clickable: true,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
id: 'credit_balance',
|
|
||||||
Header: 'Credit Balance',
|
|
||||||
accessor: 'formatted_unused_credit',
|
|
||||||
width: 100,
|
|
||||||
align: 'right',
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
id: 'balance',
|
id: 'balance',
|
||||||
Header: intl.get('receivable_balance'),
|
Header: intl.get('receivable_balance'),
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import React from 'react';
|
|||||||
|
|
||||||
import { DashboardPageContent } from '@/components';
|
import { DashboardPageContent } from '@/components';
|
||||||
import WarehouseTransfersActionsBar from './WarehouseTransfersActionsBar';
|
import WarehouseTransfersActionsBar from './WarehouseTransfersActionsBar';
|
||||||
import WarehouseTransfersViewTabs from './WarehouseTransfersViewTabs';
|
|
||||||
import WarehouseTransfersDataTable from './WarehouseTransfersDataTable';
|
import WarehouseTransfersDataTable from './WarehouseTransfersDataTable';
|
||||||
import withWarehouseTransfers from './withWarehouseTransfers';
|
import withWarehouseTransfers from './withWarehouseTransfers';
|
||||||
import withWarehouseTransfersActions from './withWarehouseTransfersActions';
|
import withWarehouseTransfersActions from './withWarehouseTransfersActions';
|
||||||
@@ -33,8 +32,8 @@ function WarehouseTransfersList({
|
|||||||
tableStateChanged={warehouseTransferTableStateChanged}
|
tableStateChanged={warehouseTransferTableStateChanged}
|
||||||
>
|
>
|
||||||
<WarehouseTransfersActionsBar />
|
<WarehouseTransfersActionsBar />
|
||||||
|
|
||||||
<DashboardPageContent>
|
<DashboardPageContent>
|
||||||
<WarehouseTransfersViewTabs />
|
|
||||||
<WarehouseTransfersDataTable />
|
<WarehouseTransfersDataTable />
|
||||||
</DashboardPageContent>
|
</DashboardPageContent>
|
||||||
</WarehouseTransfersListProvider>
|
</WarehouseTransfersListProvider>
|
||||||
|
|||||||
Reference in New Issue
Block a user