210 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			210 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| namespace App\Jobs\Sale;
 | |
| 
 | |
| use App\Abstracts\Job;
 | |
| use App\Events\Sale\InvoiceUpdated;
 | |
| use App\Events\Sale\InvoiceUpdating;
 | |
| use App\Models\Sale\Invoice;
 | |
| use App\Models\Sale\InvoiceTotal;
 | |
| use App\Traits\Currencies;
 | |
| use App\Traits\DateTime;
 | |
| use App\Traits\Relationships;
 | |
| 
 | |
| class UpdateInvoice extends Job
 | |
| {
 | |
|     use Currencies, DateTime, Relationships;
 | |
| 
 | |
|     protected $invoice;
 | |
| 
 | |
|     protected $request;
 | |
| 
 | |
|     /**
 | |
|      * Create a new job instance.
 | |
|      *
 | |
|      * @param  $request
 | |
|      */
 | |
|     public function __construct($invoice, $request)
 | |
|     {
 | |
|         $this->invoice = $invoice;
 | |
|         $this->request = $this->getRequestInstance($request);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Execute the job.
 | |
|      *
 | |
|      * @return Invoice
 | |
|      */
 | |
|     public function handle()
 | |
|     {
 | |
|         if (empty($this->request['amount'])) {
 | |
|             $this->request['amount'] = 0;
 | |
|         }
 | |
| 
 | |
|         event(new InvoiceUpdating($this->invoice, $this->request));
 | |
| 
 | |
|         // Upload attachment
 | |
|         if ($this->request->file('attachment')) {
 | |
|             $media = $this->getMedia($this->request->file('attachment'), 'invoices');
 | |
| 
 | |
|             $this->invoice->attachMedia($media, 'attachment');
 | |
|         }
 | |
| 
 | |
|         $this->createItemsAndTotals();
 | |
| 
 | |
|         $invoice_paid = $this->invoice->paid;
 | |
| 
 | |
|         unset($this->invoice->reconciled);
 | |
| 
 | |
|         if (($invoice_paid) && $this->request['amount'] > $invoice_paid) {
 | |
|             $this->request['status'] = 'partial';
 | |
|         }
 | |
| 
 | |
|         $this->invoice->update($this->request->all());
 | |
| 
 | |
|         $this->invoice->updateRecurring();
 | |
| 
 | |
|         event(new InvoiceUpdated($this->invoice, $this->request));
 | |
| 
 | |
|         return $this->invoice;
 | |
|     }
 | |
| 
 | |
|     protected function createItemsAndTotals()
 | |
|     {
 | |
|         // Create items
 | |
|         list($sub_total, $taxes) = $this->createItems();
 | |
| 
 | |
|         // Delete current totals
 | |
|         $this->deleteRelationships($this->invoice, 'totals');
 | |
| 
 | |
|         $sort_order = 1;
 | |
| 
 | |
|         // Add sub total
 | |
|         InvoiceTotal::create([
 | |
|             'company_id' => $this->invoice->company_id,
 | |
|             'invoice_id' => $this->invoice->id,
 | |
|             'code' => 'sub_total',
 | |
|             'name' => 'invoices.sub_total',
 | |
|             'amount' => $sub_total,
 | |
|             'sort_order' => $sort_order,
 | |
|         ]);
 | |
| 
 | |
|         $this->request['amount'] += $sub_total;
 | |
| 
 | |
|         $sort_order++;
 | |
| 
 | |
|         // Add discount
 | |
|         if (!empty($this->request['discount'])) {
 | |
|             $discount_total = $sub_total * ($this->request['discount'] / 100);
 | |
| 
 | |
|             InvoiceTotal::create([
 | |
|                 'company_id' => $this->invoice->company_id,
 | |
|                 'invoice_id' => $this->invoice->id,
 | |
|                 'code' => 'discount',
 | |
|                 'name' => 'invoices.discount',
 | |
|                 'amount' => $discount_total,
 | |
|                 'sort_order' => $sort_order,
 | |
|             ]);
 | |
| 
 | |
|             $this->request['amount'] -= $discount_total;
 | |
| 
 | |
|             $sort_order++;
 | |
|         }
 | |
| 
 | |
|         // Add taxes
 | |
|         if (!empty($taxes)) {
 | |
|             foreach ($taxes as $tax) {
 | |
|                 InvoiceTotal::create([
 | |
|                     'company_id' => $this->invoice->company_id,
 | |
|                     'invoice_id' => $this->invoice->id,
 | |
|                     'code' => 'tax',
 | |
|                     'name' => $tax['name'],
 | |
|                     'amount' => $tax['amount'],
 | |
|                     'sort_order' => $sort_order,
 | |
|                 ]);
 | |
| 
 | |
|                 $this->request['amount'] += $tax['amount'];
 | |
| 
 | |
|                 $sort_order++;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Add extra totals, i.e. shipping fee
 | |
|         if (!empty($this->request['totals'])) {
 | |
|             foreach ($this->request['totals'] as $total) {
 | |
|                 $total['company_id'] = $this->invoice->company_id;
 | |
|                 $total['invoice_id'] = $this->invoice->id;
 | |
|                 $total['sort_order'] = $sort_order;
 | |
| 
 | |
|                 if (empty($total['code'])) {
 | |
|                     $total['code'] = 'extra';
 | |
|                 }
 | |
| 
 | |
|                 InvoiceTotal::create($total);
 | |
| 
 | |
|                 if (empty($total['operator']) || ($total['operator'] == 'addition')) {
 | |
|                     $this->request['amount'] += $total['amount'];
 | |
|                 } else {
 | |
|                     // subtraction
 | |
|                     $this->request['amount'] -= $total['amount'];
 | |
|                 }
 | |
| 
 | |
|                 $sort_order++;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Add total
 | |
|         InvoiceTotal::create([
 | |
|             'company_id' => $this->invoice->company_id,
 | |
|             'invoice_id' => $this->invoice->id,
 | |
|             'code' => 'total',
 | |
|             'name' => 'invoices.total',
 | |
|             'amount' => $this->request['amount'],
 | |
|             'sort_order' => $sort_order,
 | |
|         ]);
 | |
|     }
 | |
| 
 | |
|     protected function createItems()
 | |
|     {
 | |
|         $sub_total = 0;
 | |
| 
 | |
|         $taxes = [];
 | |
| 
 | |
|         if (empty($this->request['items'])) {
 | |
|             return [$sub_total, $taxes];
 | |
|         }
 | |
| 
 | |
|         // Delete current items
 | |
|         $this->deleteRelationships($this->invoice, ['items', 'item_taxes']);
 | |
| 
 | |
|         foreach ((array) $this->request['items'] as $item) {
 | |
|             if (empty($item['discount'])) {
 | |
|                 $item['discount'] = !empty($this->request['discount']) ? !empty($this->request['discount']) : 0;
 | |
|             }
 | |
| 
 | |
|             $invoice_item = $this->dispatch(new CreateInvoiceItem($item, $this->invoice));
 | |
| 
 | |
|             // Calculate totals
 | |
|             $sub_total += $invoice_item->total;
 | |
| 
 | |
|             if (!$invoice_item->item_taxes) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             // Set taxes
 | |
|             foreach ((array) $invoice_item->item_taxes as $item_tax) {
 | |
|                 if (array_key_exists($item_tax['tax_id'], $taxes)) {
 | |
|                     $taxes[$item_tax['tax_id']]['amount'] += $item_tax['amount'];
 | |
|                 } else {
 | |
|                     $taxes[$item_tax['tax_id']] = [
 | |
|                         'name' => $item_tax['name'],
 | |
|                         'amount' => $item_tax['amount']
 | |
|                     ];
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return [$sub_total, $taxes];
 | |
|     }
 | |
| }
 |