mirror of
https://github.com/InvoiceShelf/InvoiceShelf.git
synced 2026-04-15 09:14:08 +00:00
238 lines
9.2 KiB
PHP
238 lines
9.2 KiB
PHP
<?php
|
|
|
|
namespace App\Services;
|
|
|
|
use App\Facades\Hashids;
|
|
use App\Http\Requests\RecurringInvoiceRequest;
|
|
use App\Models\Company;
|
|
use App\Models\CompanySetting;
|
|
use App\Models\Customer;
|
|
use App\Models\ExchangeRateLog;
|
|
use App\Models\Invoice;
|
|
use App\Models\RecurringInvoice;
|
|
use Carbon\Carbon;
|
|
use Illuminate\Support\Collection;
|
|
|
|
class RecurringInvoiceService
|
|
{
|
|
public function __construct(
|
|
private readonly DocumentItemService $documentItemService,
|
|
private readonly InvoiceService $invoiceService,
|
|
) {}
|
|
|
|
public function create(RecurringInvoiceRequest $request): RecurringInvoice
|
|
{
|
|
$recurringInvoice = RecurringInvoice::create($request->getRecurringInvoicePayload());
|
|
|
|
$companyCurrency = CompanySetting::getSetting('currency', $request->header('company'));
|
|
|
|
if ((string) $recurringInvoice['currency_id'] !== $companyCurrency) {
|
|
ExchangeRateLog::addExchangeRateLog($recurringInvoice);
|
|
}
|
|
|
|
$this->createItems($recurringInvoice, $request->items);
|
|
|
|
if ($request->has('taxes') && (! empty($request->taxes))) {
|
|
$this->createTaxes($recurringInvoice, $request->taxes);
|
|
}
|
|
|
|
if ($request->customFields) {
|
|
$recurringInvoice->addCustomFields($request->customFields);
|
|
}
|
|
|
|
return $recurringInvoice;
|
|
}
|
|
|
|
public function update(RecurringInvoice $recurringInvoice, RecurringInvoiceRequest $request): RecurringInvoice
|
|
{
|
|
$data = $request->getRecurringInvoicePayload();
|
|
|
|
$recurringInvoice->update($data);
|
|
|
|
$companyCurrency = CompanySetting::getSetting('currency', $request->header('company'));
|
|
|
|
if ((string) $data['currency_id'] !== $companyCurrency) {
|
|
ExchangeRateLog::addExchangeRateLog($recurringInvoice);
|
|
}
|
|
|
|
$recurringInvoice->items()->delete();
|
|
$this->createItems($recurringInvoice, $request->items);
|
|
|
|
$recurringInvoice->taxes()->delete();
|
|
if ($request->has('taxes') && (! empty($request->taxes))) {
|
|
$this->createTaxes($recurringInvoice, $request->taxes);
|
|
}
|
|
|
|
if ($request->customFields) {
|
|
$recurringInvoice->updateCustomFields($request->customFields);
|
|
}
|
|
|
|
return $recurringInvoice;
|
|
}
|
|
|
|
public function delete(Collection $ids): bool
|
|
{
|
|
foreach ($ids as $id) {
|
|
$recurringInvoice = RecurringInvoice::find($id);
|
|
|
|
if ($recurringInvoice->invoices()->exists()) {
|
|
$recurringInvoice->invoices()->update(['recurring_invoice_id' => null]);
|
|
}
|
|
|
|
if ($recurringInvoice->items()->exists()) {
|
|
$recurringInvoice->items()->delete();
|
|
}
|
|
|
|
if ($recurringInvoice->taxes()->exists()) {
|
|
$recurringInvoice->taxes()->delete();
|
|
}
|
|
|
|
$recurringInvoice->delete();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public function generateInvoice(RecurringInvoice $recurringInvoice): void
|
|
{
|
|
if (Carbon::now()->lessThan($recurringInvoice->starts_at)) {
|
|
return;
|
|
}
|
|
|
|
if ($recurringInvoice->limit_by == 'DATE') {
|
|
$startDate = Carbon::today()->format('Y-m-d');
|
|
$endDate = $recurringInvoice->limit_date;
|
|
|
|
if ($endDate >= $startDate) {
|
|
$this->createInvoiceFromRecurring($recurringInvoice);
|
|
$recurringInvoice->updateNextInvoiceDate();
|
|
} else {
|
|
$recurringInvoice->markStatusAsCompleted();
|
|
}
|
|
} elseif ($recurringInvoice->limit_by == 'COUNT') {
|
|
$invoiceCount = Invoice::where('recurring_invoice_id', $recurringInvoice->id)->count();
|
|
|
|
if ($invoiceCount < $recurringInvoice->limit_count) {
|
|
$this->createInvoiceFromRecurring($recurringInvoice);
|
|
$recurringInvoice->updateNextInvoiceDate();
|
|
} else {
|
|
$recurringInvoice->markStatusAsCompleted();
|
|
}
|
|
} else {
|
|
$this->createInvoiceFromRecurring($recurringInvoice);
|
|
$recurringInvoice->updateNextInvoiceDate();
|
|
}
|
|
}
|
|
|
|
private function createInvoiceFromRecurring(RecurringInvoice $recurringInvoice): void
|
|
{
|
|
$serial = (new SerialNumberService)
|
|
->setModel(new Invoice)
|
|
->setCompany($recurringInvoice->company_id)
|
|
->setCustomer($recurringInvoice->customer_id)
|
|
->setNextNumbers();
|
|
|
|
$days = intval(CompanySetting::getSetting('invoice_due_date_days', $recurringInvoice->company_id));
|
|
|
|
if (! $days || $days == 'null') {
|
|
$days = 7;
|
|
}
|
|
|
|
$newInvoice['creator_id'] = $recurringInvoice->creator_id;
|
|
$newInvoice['invoice_date'] = Carbon::today()->format('Y-m-d');
|
|
$newInvoice['due_date'] = Carbon::today()->addDays($days)->format('Y-m-d');
|
|
$newInvoice['status'] = Invoice::STATUS_DRAFT;
|
|
$newInvoice['company_id'] = $recurringInvoice->company_id;
|
|
$newInvoice['paid_status'] = Invoice::STATUS_UNPAID;
|
|
$newInvoice['sub_total'] = $recurringInvoice->sub_total;
|
|
$newInvoice['tax_per_item'] = $recurringInvoice->tax_per_item;
|
|
$newInvoice['discount_per_item'] = $recurringInvoice->discount_per_item;
|
|
$newInvoice['tax'] = $recurringInvoice->tax;
|
|
$newInvoice['total'] = $recurringInvoice->total;
|
|
$newInvoice['customer_id'] = $recurringInvoice->customer_id;
|
|
$newInvoice['currency_id'] = Customer::find($recurringInvoice->customer_id)->currency_id;
|
|
$newInvoice['template_name'] = $recurringInvoice->template_name;
|
|
$newInvoice['due_amount'] = $recurringInvoice->total;
|
|
$newInvoice['recurring_invoice_id'] = $recurringInvoice->id;
|
|
$newInvoice['discount_val'] = $recurringInvoice->discount_val;
|
|
$newInvoice['discount'] = $recurringInvoice->discount;
|
|
$newInvoice['discount_type'] = $recurringInvoice->discount_type;
|
|
$newInvoice['notes'] = $recurringInvoice->notes;
|
|
$newInvoice['exchange_rate'] = $recurringInvoice->exchange_rate;
|
|
$newInvoice['sales_tax_type'] = $recurringInvoice->sales_tax_type;
|
|
$newInvoice['sales_tax_address_type'] = $recurringInvoice->sales_tax_address_type;
|
|
$newInvoice['invoice_number'] = $serial->getNextNumber();
|
|
$newInvoice['sequence_number'] = $serial->nextSequenceNumber;
|
|
$newInvoice['customer_sequence_number'] = $serial->nextCustomerSequenceNumber;
|
|
$newInvoice['base_due_amount'] = $recurringInvoice->exchange_rate * $recurringInvoice->due_amount;
|
|
$newInvoice['base_discount_val'] = $recurringInvoice->exchange_rate * $recurringInvoice->discount_val;
|
|
$newInvoice['base_sub_total'] = $recurringInvoice->exchange_rate * $recurringInvoice->sub_total;
|
|
$newInvoice['base_tax'] = $recurringInvoice->exchange_rate * $recurringInvoice->tax;
|
|
$newInvoice['base_total'] = $recurringInvoice->exchange_rate * $recurringInvoice->total;
|
|
$invoice = Invoice::create($newInvoice);
|
|
$invoice->unique_hash = Hashids::connection(Invoice::class)->encode($invoice->id);
|
|
$invoice->save();
|
|
|
|
$recurringInvoice->load('items.taxes');
|
|
$this->documentItemService->createItems($invoice, $recurringInvoice->items->toArray());
|
|
|
|
if ($recurringInvoice->taxes()->exists()) {
|
|
$this->documentItemService->createTaxes($invoice, $recurringInvoice->taxes->toArray());
|
|
}
|
|
|
|
if ($recurringInvoice->fields()->exists()) {
|
|
$customField = [];
|
|
|
|
foreach ($recurringInvoice->fields as $data) {
|
|
$customField[] = [
|
|
'id' => $data->custom_field_id,
|
|
'value' => $data->defaultAnswer,
|
|
];
|
|
}
|
|
|
|
$invoice->addCustomFields($customField);
|
|
}
|
|
|
|
if ($recurringInvoice->send_automatically == true) {
|
|
$data = [
|
|
'body' => CompanySetting::getSetting('invoice_mail_body', $recurringInvoice->company_id),
|
|
'from' => config('mail.from.address'),
|
|
'to' => $recurringInvoice->customer->email,
|
|
'subject' => trans('invoices')['new_invoice'],
|
|
'invoice' => $invoice->toArray(),
|
|
'customer' => $invoice->customer->toArray(),
|
|
'company' => Company::find($invoice->company_id),
|
|
];
|
|
|
|
$this->invoiceService->send($invoice, $data);
|
|
}
|
|
}
|
|
|
|
private function createItems(RecurringInvoice $recurringInvoice, array $items): void
|
|
{
|
|
foreach ($items as $item) {
|
|
$item['company_id'] = $recurringInvoice->company_id;
|
|
$createdItem = $recurringInvoice->items()->create($item);
|
|
if (array_key_exists('taxes', $item) && $item['taxes']) {
|
|
foreach ($item['taxes'] as $tax) {
|
|
$tax['company_id'] = $recurringInvoice->company_id;
|
|
if (gettype($tax['amount']) !== 'NULL') {
|
|
$createdItem->taxes()->create($tax);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private function createTaxes(RecurringInvoice $recurringInvoice, array $taxes): void
|
|
{
|
|
foreach ($taxes as $tax) {
|
|
$tax['company_id'] = $recurringInvoice->company_id;
|
|
|
|
if (gettype($tax['amount']) !== 'NULL') {
|
|
$recurringInvoice->taxes()->create($tax);
|
|
}
|
|
}
|
|
}
|
|
}
|