mirror of
https://github.com/invoiceninja/invoiceninja.git
synced 2025-07-09 03:14:30 -04:00
Fixes for over payments
This commit is contained in:
parent
6d366febd8
commit
3ee3f67c8c
@ -83,8 +83,6 @@ class PaymentController extends Controller
|
|||||||
|
|
||||||
$gateway = CompanyGateway::find($request->input('company_gateway_id'));
|
$gateway = CompanyGateway::find($request->input('company_gateway_id'));
|
||||||
|
|
||||||
//refactor from here!
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* find invoices
|
* find invoices
|
||||||
*
|
*
|
||||||
@ -95,11 +93,13 @@ class PaymentController extends Controller
|
|||||||
$invoices = Invoice::whereIn('id', $this->transformKeys($payable_invoices->pluck('invoice_id')->toArray()))->get();
|
$invoices = Invoice::whereIn('id', $this->transformKeys($payable_invoices->pluck('invoice_id')->toArray()))->get();
|
||||||
|
|
||||||
/* pop non payable invoice from the $payable_invoices array */
|
/* pop non payable invoice from the $payable_invoices array */
|
||||||
|
|
||||||
$payable_invoices = $payable_invoices->filter(function ($payable_invoice) use ($invoices) {
|
$payable_invoices = $payable_invoices->filter(function ($payable_invoice) use ($invoices) {
|
||||||
return $invoices->where('hashed_id', $payable_invoice['invoice_id'])->first()->isPayable();
|
return $invoices->where('hashed_id', $payable_invoice['invoice_id'])->first()->isPayable();
|
||||||
});
|
});
|
||||||
|
|
||||||
/*return early if no invoices*/
|
/*return early if no invoices*/
|
||||||
|
|
||||||
if ($payable_invoices->count() == 0) {
|
if ($payable_invoices->count() == 0) {
|
||||||
return redirect()
|
return redirect()
|
||||||
->route('client.invoices.index')
|
->route('client.invoices.index')
|
||||||
@ -108,23 +108,32 @@ class PaymentController extends Controller
|
|||||||
|
|
||||||
$settings = auth()->user()->client->getMergedSettings();
|
$settings = auth()->user()->client->getMergedSettings();
|
||||||
|
|
||||||
/*iterate through invoices and add gateway fees and other payment metadata*/
|
/* This loop checks for under / over payments and returns the user if a check fails */
|
||||||
$payable_invoices = $payable_invoices->map(function ($payable_invoice) use ($invoices, $settings) {
|
|
||||||
$payable_invoice['amount'] = Number::parseFloat($payable_invoice['amount']);
|
foreach($payable_invoices as $payable_invoice)
|
||||||
|
{
|
||||||
|
|
||||||
|
/*Match the payable invoice to the Model Invoice*/
|
||||||
|
|
||||||
$invoice = $invoices->first(function ($inv) use ($payable_invoice) {
|
$invoice = $invoices->first(function ($inv) use ($payable_invoice) {
|
||||||
return $payable_invoice['invoice_id'] == $inv->hashed_id;
|
return $payable_invoice['invoice_id'] == $inv->hashed_id;
|
||||||
});
|
});
|
||||||
|
|
||||||
// Check if company supports over & under payments.
|
/*
|
||||||
// In case it doesn't this is where process should stop.
|
* Check if company supports over & under payments.
|
||||||
|
* Determine the payable amount and the max payable. ie either partial or invoice balance
|
||||||
|
*/
|
||||||
|
|
||||||
$payable_amount = Number::roundValue(Number::parseFloat($payable_invoice['amount']), auth()->user()->client->currency()->precision);
|
$payable_amount = Number::roundValue(Number::parseFloat($payable_invoice['amount']), auth()->user()->client->currency()->precision);
|
||||||
$invoice_balance = Number::roundValue($invoice->balance, auth()->user()->client->currency()->precision);
|
$invoice_balance = Number::roundValue(($invoice->partial > 0 ? $invoice->partial : $invoice->balance), auth()->user()->client->currency()->precision);
|
||||||
|
|
||||||
|
/*If we don't allow under/over payments force the payable amount - prevents inspect element adjustments in JS*/
|
||||||
|
|
||||||
if ($settings->client_portal_allow_under_payment == false && $settings->client_portal_allow_over_payment == false) {
|
if ($settings->client_portal_allow_under_payment == false && $settings->client_portal_allow_over_payment == false) {
|
||||||
$payable_invoice['amount'] = Number::roundValue(($invoice->partial > 0 ? $invoice->partial : $invoice->balance), auth()->user()->client->currency()->precision);
|
$payable_invoice['amount'] = Number::roundValue(($invoice->partial > 0 ? $invoice->partial : $invoice->balance), auth()->user()->client->currency()->precision);
|
||||||
} // We don't allow either of these, reset the amount to default invoice (to prevent inspect element payments).
|
}
|
||||||
|
|
||||||
|
/* If we DO allow under payments check the minimum amount is present else return */
|
||||||
|
|
||||||
if ($settings->client_portal_allow_under_payment) {
|
if ($settings->client_portal_allow_under_payment) {
|
||||||
if ($payable_invoice['amount'] < $settings->client_portal_under_payment_minimum) {
|
if ($payable_invoice['amount'] < $settings->client_portal_under_payment_minimum) {
|
||||||
@ -133,23 +142,38 @@ class PaymentController extends Controller
|
|||||||
->with('message', ctrans('texts.minimum_required_payment', ['amount' => $settings->client_portal_under_payment_minimum]));
|
->with('message', ctrans('texts.minimum_required_payment', ['amount' => $settings->client_portal_under_payment_minimum]));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
$payable_amount = Number::roundValue(Number::parseFloat($payable_invoice['amount']), auth()->user()->client->currency()->precision);
|
|
||||||
$invoice_balance = Number::roundValue($invoice->balance, auth()->user()->client->currency()->precision);
|
|
||||||
|
|
||||||
|
/*Double check!!*/
|
||||||
if ($payable_amount < $invoice_balance) {
|
if ($payable_amount < $invoice_balance) {
|
||||||
return redirect()
|
return redirect()
|
||||||
->route('client.invoices.index')
|
->route('client.invoices.index')
|
||||||
->with('message', ctrans('texts.under_payments_disabled'));
|
->with('message', ctrans('texts.under_payments_disabled'));
|
||||||
}
|
}
|
||||||
} // Make sure 'amount' from form is not lower than 'amount' from invoice.
|
}
|
||||||
|
|
||||||
if ($settings->client_portal_allow_over_payment == false) {
|
/* If we don't allow over payments and the amount exceeds the balance */
|
||||||
|
|
||||||
|
if (!$settings->client_portal_allow_over_payment) {
|
||||||
if ($payable_amount > $invoice_balance) {
|
if ($payable_amount > $invoice_balance) {
|
||||||
return redirect()
|
return redirect()
|
||||||
->route('client.invoices.index')
|
->route('client.invoices.index')
|
||||||
->with('message', ctrans('texts.over_payments_disabled'));
|
->with('message', ctrans('texts.over_payments_disabled'));
|
||||||
}
|
}
|
||||||
} // Make sure 'amount' from form is not higher than 'amount' from invoice.
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/*Iterate through invoices and add gateway fees and other payment metadata*/
|
||||||
|
|
||||||
|
$payable_invoices = $payable_invoices->map(function ($payable_invoice) use ($invoices, $settings) {
|
||||||
|
$payable_invoice['amount'] = Number::parseFloat($payable_invoice['amount']);
|
||||||
|
|
||||||
|
$invoice = $invoices->first(function ($inv) use ($payable_invoice) {
|
||||||
|
return $payable_invoice['invoice_id'] == $inv->hashed_id;
|
||||||
|
});
|
||||||
|
|
||||||
|
$payable_amount = Number::roundValue(Number::parseFloat($payable_invoice['amount']), auth()->user()->client->currency()->precision);
|
||||||
|
$invoice_balance = Number::roundValue($invoice->balance, auth()->user()->client->currency()->precision);
|
||||||
|
|
||||||
$payable_invoice['due_date'] = $this->formatDate($invoice->due_date, $invoice->client->date_format());
|
$payable_invoice['due_date'] = $this->formatDate($invoice->due_date, $invoice->client->date_format());
|
||||||
$payable_invoice['invoice_number'] = $invoice->number;
|
$payable_invoice['invoice_number'] = $invoice->number;
|
||||||
|
@ -192,7 +192,7 @@ class PayPalExpressPaymentDriver extends BaseDriver
|
|||||||
'cancelUrl' => $this->client->company->domain() . '/client/invoices',
|
'cancelUrl' => $this->client->company->domain() . '/client/invoices',
|
||||||
'description' => implode(',', collect($this->payment_hash->data->invoices)
|
'description' => implode(',', collect($this->payment_hash->data->invoices)
|
||||||
->map(function ($invoice) {
|
->map(function ($invoice) {
|
||||||
return sprintf('%s: %s', ctrans('texts.invoice_number'), $invoice->number);
|
return sprintf('%s: %s', ctrans('texts.invoice_number'), $invoice->invoice_number);
|
||||||
})->toArray()),
|
})->toArray()),
|
||||||
'transactionId' => $this->payment_hash->hash . '-' . time(),
|
'transactionId' => $this->payment_hash->hash . '-' . time(),
|
||||||
'ButtonSource' => 'InvoiceNinja_SP',
|
'ButtonSource' => 'InvoiceNinja_SP',
|
||||||
|
@ -39,6 +39,7 @@ class UpdateInvoicePayment
|
|||||||
$invoices = Invoice::whereIn('id', $this->transformKeys(array_column($paid_invoices, 'invoice_id')))->get();
|
$invoices = Invoice::whereIn('id', $this->transformKeys(array_column($paid_invoices, 'invoice_id')))->get();
|
||||||
|
|
||||||
collect($paid_invoices)->each(function ($paid_invoice) use ($invoices) {
|
collect($paid_invoices)->each(function ($paid_invoice) use ($invoices) {
|
||||||
|
|
||||||
$invoice = $invoices->first(function ($inv) use ($paid_invoice) {
|
$invoice = $invoices->first(function ($inv) use ($paid_invoice) {
|
||||||
return $paid_invoice->invoice_id == $inv->hashed_id;
|
return $paid_invoice->invoice_id == $inv->hashed_id;
|
||||||
});
|
});
|
||||||
@ -49,6 +50,11 @@ class UpdateInvoicePayment
|
|||||||
$paid_amount = $paid_invoice->amount;
|
$paid_amount = $paid_invoice->amount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Need to determine here is we have an OVER payment - if YES only apply the max invoice amount */
|
||||||
|
if($paid_amount > $invoice->partial && $paid_amount > $invoice->balance)
|
||||||
|
$paid_amount = $invoice->balance;
|
||||||
|
|
||||||
|
/* Updates the company ledger */
|
||||||
$this->payment
|
$this->payment
|
||||||
->ledger()
|
->ledger()
|
||||||
->updatePaymentBalance($paid_amount * -1);
|
->updatePaymentBalance($paid_amount * -1);
|
||||||
|
@ -3362,4 +3362,5 @@ return [
|
|||||||
'currency_albanian_lek' => 'Albanian Lek',
|
'currency_albanian_lek' => 'Albanian Lek',
|
||||||
|
|
||||||
'endless' => 'Endless',
|
'endless' => 'Endless',
|
||||||
|
'minimum_payment' => 'Minimum Payment',
|
||||||
];
|
];
|
||||||
|
Loading…
x
Reference in New Issue
Block a user