'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(); } }