2020-04-08 12:48:31 +02:00
|
|
|
<?php
|
|
|
|
/**
|
2020-09-06 11:38:10 +02:00
|
|
|
* Invoice Ninja (https://invoiceninja.com).
|
2020-04-08 12:48:31 +02:00
|
|
|
*
|
|
|
|
* @link https://github.com/invoiceninja/invoiceninja source repository
|
|
|
|
*
|
2021-01-03 22:54:54 +01:00
|
|
|
* @copyright Copyright (c) 2021. Invoice Ninja LLC (https://invoiceninja.com)
|
2020-04-08 12:48:31 +02:00
|
|
|
*
|
|
|
|
* @license https://opensource.org/licenses/AAL
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace App\Services\Invoice;
|
|
|
|
|
2020-04-09 01:35:40 +02:00
|
|
|
use App\Events\Invoice\InvoiceWasReversed;
|
2020-04-08 12:48:31 +02:00
|
|
|
use App\Factory\CreditFactory;
|
|
|
|
use App\Factory\InvoiceItemFactory;
|
|
|
|
use App\Helpers\Invoice\InvoiceSum;
|
|
|
|
use App\Models\Client;
|
2020-09-22 08:11:32 +02:00
|
|
|
use App\Models\Credit;
|
2020-04-08 12:48:31 +02:00
|
|
|
use App\Models\Invoice;
|
|
|
|
use App\Models\Payment;
|
|
|
|
use App\Models\Paymentable;
|
|
|
|
use App\Services\AbstractService;
|
2020-07-08 14:02:16 +02:00
|
|
|
use App\Utils\Ninja;
|
2020-04-08 12:48:31 +02:00
|
|
|
use App\Utils\Traits\GeneratesCounter;
|
|
|
|
|
|
|
|
class HandleReversal extends AbstractService
|
|
|
|
{
|
|
|
|
use GeneratesCounter;
|
|
|
|
|
|
|
|
private $invoice;
|
|
|
|
|
|
|
|
public function __construct(Invoice $invoice)
|
2020-04-15 02:30:52 +02:00
|
|
|
{
|
2020-04-08 12:48:31 +02:00
|
|
|
$this->invoice = $invoice;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function run()
|
|
|
|
{
|
|
|
|
/* Check again!! */
|
2020-09-06 11:38:10 +02:00
|
|
|
if (! $this->invoice->invoiceReversable($this->invoice)) {
|
2020-04-08 12:48:31 +02:00
|
|
|
return $this->invoice;
|
2020-04-15 02:30:52 +02:00
|
|
|
}
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-07-16 05:54:26 +02:00
|
|
|
/* If the invoice has been cancelled - we need to unwind the cancellation before reversing*/
|
2020-09-06 11:38:10 +02:00
|
|
|
if ($this->invoice->status_id == Invoice::STATUS_CANCELLED) {
|
2020-06-15 08:06:32 +02:00
|
|
|
$this->invoice = $this->invoice->service()->reverseCancellation()->save();
|
2020-09-06 11:38:10 +02:00
|
|
|
}
|
2020-06-15 01:34:18 +02:00
|
|
|
|
2020-04-08 12:48:31 +02:00
|
|
|
$balance_remaining = $this->invoice->balance;
|
|
|
|
|
|
|
|
$total_paid = $this->invoice->amount - $this->invoice->balance;
|
|
|
|
|
|
|
|
/*Adjust payment applied and the paymentables to the correct amount */
|
2020-09-22 05:09:42 +02:00
|
|
|
$paymentables = Paymentable::wherePaymentableType('invoices')
|
2020-04-08 12:48:31 +02:00
|
|
|
->wherePaymentableId($this->invoice->id)
|
|
|
|
->get();
|
|
|
|
|
2020-04-15 02:30:52 +02:00
|
|
|
$paymentables->each(function ($paymentable) use ($total_paid) {
|
2020-04-08 12:48:31 +02:00
|
|
|
$reversable_amount = $paymentable->amount - $paymentable->refunded;
|
|
|
|
|
|
|
|
$total_paid -= $reversable_amount;
|
|
|
|
|
|
|
|
$paymentable->amount = $paymentable->refunded;
|
|
|
|
$paymentable->save();
|
|
|
|
});
|
|
|
|
|
|
|
|
/* Generate a credit for the $total_paid amount */
|
2020-09-06 11:38:10 +02:00
|
|
|
$notes = 'Credit for reversal of '.$this->invoice->number;
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-04-15 02:30:52 +02:00
|
|
|
if ($total_paid > 0) {
|
2020-04-08 12:55:28 +02:00
|
|
|
$credit = CreditFactory::create($this->invoice->company_id, $this->invoice->user_id);
|
|
|
|
$credit->client_id = $this->invoice->client_id;
|
2020-06-18 01:43:28 +02:00
|
|
|
$credit->invoice_id = $this->invoice->id;
|
2020-09-06 11:38:10 +02:00
|
|
|
|
2020-04-15 02:30:52 +02:00
|
|
|
$item = InvoiceItemFactory::create();
|
|
|
|
$item->quantity = 1;
|
2020-09-06 11:38:10 +02:00
|
|
|
$item->cost = (float) $total_paid;
|
2020-04-15 02:30:52 +02:00
|
|
|
$item->notes = $notes;
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-04-15 02:30:52 +02:00
|
|
|
$line_items[] = $item;
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-04-08 12:55:28 +02:00
|
|
|
$credit->line_items = $line_items;
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-04-08 12:55:28 +02:00
|
|
|
$credit->save();
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-04-08 12:55:28 +02:00
|
|
|
$credit_calc = new InvoiceSum($credit);
|
|
|
|
$credit_calc->build();
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-10-15 11:41:59 +02:00
|
|
|
$credit = $credit_calc->purgeTaxes()->getCredit();
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-04-08 12:55:28 +02:00
|
|
|
$credit->service()->markSent()->save();
|
|
|
|
}
|
2020-06-15 01:34:18 +02:00
|
|
|
|
2020-09-22 05:09:42 +02:00
|
|
|
/*If there is a payment linked, then the credit needs to be linked back to that payment in case of refund*/
|
2020-11-25 15:19:52 +01:00
|
|
|
if ($paymentables->count() > 0) {
|
2020-09-22 05:09:42 +02:00
|
|
|
$payment = $paymentables->first()->payment;
|
|
|
|
$payment->credits()->save($credit);
|
|
|
|
|
|
|
|
$paymentable_credit = $payment->credits()
|
2020-09-22 08:11:32 +02:00
|
|
|
->wherePaymentableType(Credit::class)
|
2020-09-22 05:09:42 +02:00
|
|
|
->wherePaymentableId($credit->id)
|
|
|
|
->first();
|
|
|
|
|
|
|
|
//harvest the credit record and add in the amount for the credit.
|
|
|
|
$paymentable_credit->pivot->amount = $total_paid;
|
|
|
|
$paymentable_credit->pivot->save();
|
|
|
|
}
|
|
|
|
|
2020-04-08 12:48:31 +02:00
|
|
|
/* Set invoice balance to 0 */
|
2020-09-06 11:38:10 +02:00
|
|
|
if ($this->invoice->balance != 0) {
|
|
|
|
$this->invoice->ledger()->updateInvoiceBalance($balance_remaining * -1, $notes)->save();
|
|
|
|
}
|
2020-04-08 12:48:31 +02:00
|
|
|
|
2020-09-06 11:38:10 +02:00
|
|
|
$this->invoice->balance = 0;
|
2020-04-08 12:48:31 +02:00
|
|
|
|
|
|
|
/* Set invoice status to reversed... somehow*/
|
|
|
|
$this->invoice->service()->setStatus(Invoice::STATUS_REVERSED)->save();
|
|
|
|
|
|
|
|
/* Reduce client.paid_to_date by $total_paid amount */
|
|
|
|
/* Reduce the client balance by $balance_remaining */
|
|
|
|
|
|
|
|
$this->invoice->client->service()
|
2020-09-06 11:38:10 +02:00
|
|
|
->updateBalance($balance_remaining * -1)
|
|
|
|
->updatePaidToDate($total_paid * -1)
|
2020-04-08 12:48:31 +02:00
|
|
|
->save();
|
|
|
|
|
2020-07-08 14:02:16 +02:00
|
|
|
event(new InvoiceWasReversed($this->invoice, $this->invoice->company, Ninja::eventVars()));
|
2020-09-06 11:38:10 +02:00
|
|
|
|
2020-04-08 12:48:31 +02:00
|
|
|
return $this->invoice;
|
|
|
|
//create a ledger row for this with the resulting Credit ( also include an explanation in the notes section )
|
|
|
|
}
|
|
|
|
}
|