195 lines
5.0 KiB
PHP
195 lines
5.0 KiB
PHP
<?php
|
||
|
||
namespace App\Models;
|
||
|
||
use App\Classes\PDF;
|
||
use App\Classes\Price;
|
||
use App\Classes\Settings;
|
||
use App\Models\Traits\HasProperties;
|
||
use App\Observers\InvoiceObserver;
|
||
use Illuminate\Database\Eloquent\Attributes\ObservedBy;
|
||
use Illuminate\Database\Eloquent\Casts\Attribute;
|
||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||
use OwenIt\Auditing\Contracts\Auditable;
|
||
|
||
#[ObservedBy([InvoiceObserver::class])]
|
||
class Invoice extends Model implements Auditable
|
||
{
|
||
use \App\Models\Traits\Auditable, HasFactory, HasProperties;
|
||
|
||
public const STATUS_PENDING = 'pending';
|
||
|
||
public const STATUS_PAID = 'paid';
|
||
|
||
public const STATUS_CANCELLED = 'cancelled';
|
||
|
||
protected $fillable = ['number', 'user_id', 'currency_code', 'due_at', 'status'];
|
||
|
||
protected $casts = [
|
||
'due_at' => 'date',
|
||
];
|
||
|
||
public bool $send_create_email = true;
|
||
|
||
/**
|
||
* Total of the invoice.
|
||
*
|
||
* @return string
|
||
*/
|
||
public function total(): Attribute
|
||
{
|
||
return Attribute::make(
|
||
get: fn () => $this->items->sum(fn ($item) => $item->price * $item->quantity)
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Total of the invoice.
|
||
*
|
||
* @return string
|
||
*/
|
||
public function formattedTotal(): Attribute
|
||
{
|
||
return Attribute::make(
|
||
get: fn () => new Price(['price' => $this->total, 'currency' => $this->currency, 'tax' => $this->tax])
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Formatted remaining amount of the invoice.
|
||
*/
|
||
public function formattedRemaining(): Attribute
|
||
{
|
||
return Attribute::make(
|
||
get: fn () => new Price(['price' => $this->remaining, 'currency' => $this->currency, 'tax' => $this->tax])
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Remaining amount of the invoice.
|
||
*/
|
||
public function remaining(): Attribute
|
||
{
|
||
return Attribute::make(
|
||
get: fn () => $this->total - $this->transactions->where('status', \App\Enums\InvoiceTransactionStatus::Succeeded)->sum('amount')
|
||
);
|
||
}
|
||
|
||
/**
|
||
* Get tax model of the invoice.
|
||
* Either via locked invoice or via users country.
|
||
*/
|
||
public function tax(): Attribute
|
||
{
|
||
if (config('settings.invoice_snapshot', true) && $this?->snapshot?->tax_name) {
|
||
return Attribute::make(
|
||
get: fn () => new TaxRate([
|
||
'name' => $this->snapshot->tax_name,
|
||
'rate' => $this->snapshot->tax_rate,
|
||
'country' => $this->snapshot->tax_country,
|
||
])
|
||
);
|
||
}
|
||
|
||
return Attribute::make(
|
||
get: fn () => Settings::tax($this->user)
|
||
);
|
||
}
|
||
|
||
public function userProperties(): Attribute
|
||
{
|
||
if (config('settings.invoice_snapshot', true) && $this?->snapshot?->properties) {
|
||
return Attribute::make(
|
||
get: fn () => $this->snapshot->properties
|
||
);
|
||
}
|
||
|
||
return Attribute::make(
|
||
get: fn () => $this->user->properties()->with('parent_property')->whereHas('parent_property', function ($query) {
|
||
$query->where('show_on_invoice', true);
|
||
})->pluck('value', 'key')->toArray()
|
||
);
|
||
}
|
||
|
||
public function userName(): Attribute
|
||
{
|
||
if (config('settings.invoice_snapshot', true) && $this?->snapshot?->name) {
|
||
return Attribute::make(
|
||
get: fn () => $this->snapshot->name
|
||
);
|
||
}
|
||
|
||
return Attribute::make(
|
||
get: fn () => $this->user->name
|
||
);
|
||
}
|
||
|
||
public function billTo(): Attribute
|
||
{
|
||
if (config('settings.invoice_snapshot', true) && $this?->snapshot?->bill_to) {
|
||
return Attribute::make(
|
||
get: fn () => $this->snapshot->bill_to
|
||
);
|
||
}
|
||
|
||
return Attribute::make(
|
||
get: fn () => config('settings.bill_to_text', config('settings.company_name'))
|
||
);
|
||
}
|
||
|
||
public function currency()
|
||
{
|
||
return $this->belongsTo(Currency::class, 'currency_code');
|
||
}
|
||
|
||
public function user()
|
||
{
|
||
return $this->belongsTo(User::class);
|
||
}
|
||
|
||
public function items()
|
||
{
|
||
return $this->hasMany(InvoiceItem::class);
|
||
}
|
||
|
||
public function transactions()
|
||
{
|
||
return $this->hasMany(InvoiceTransaction::class);
|
||
}
|
||
|
||
public function snapshot()
|
||
{
|
||
return $this->hasOne(InvoiceSnapshot::class);
|
||
}
|
||
|
||
public function pdf(): Attribute
|
||
{
|
||
return Attribute::make(
|
||
get: fn () => PDF::generateInvoice($this)
|
||
);
|
||
}
|
||
|
||
public function getRouteKey()
|
||
{
|
||
// Prefer using number if it’s set, otherwise fallback to id
|
||
return $this->number ?: $this->id;
|
||
}
|
||
|
||
public function resolveRouteBinding($value, $field = null)
|
||
{
|
||
if ($field) {
|
||
return $this->where($field, $value)->firstOrFail();
|
||
}
|
||
|
||
// Try to find by number first
|
||
$query = $this->where('number', $value);
|
||
|
||
// Only try to match by ID if value is numeric
|
||
if (is_numeric($value)) {
|
||
$query->orWhere('id', $value);
|
||
}
|
||
|
||
return $query->firstOrFail();
|
||
}
|
||
}
|