123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222 |
- <?php
-
- namespace App\Models\Setting;
-
- use App\Casts\TrimLeadingZeroCast;
- use App\Concerns\Blamable;
- use App\Concerns\CompanyOwned;
- use App\Enums\Accounting\DocumentType;
- use App\Enums\Setting\Font;
- use App\Enums\Setting\PaymentTerms;
- use App\Enums\Setting\Template;
- use Database\Factories\Setting\DocumentDefaultFactory;
- use Illuminate\Database\Eloquent\Builder;
- use Illuminate\Database\Eloquent\Casts\AsArrayObject;
- use Illuminate\Database\Eloquent\Casts\Attribute;
- use Illuminate\Database\Eloquent\Factories\Factory;
- use Illuminate\Database\Eloquent\Factories\HasFactory;
- use Illuminate\Database\Eloquent\Model;
- use Illuminate\Support\Facades\Storage;
-
- class DocumentDefault extends Model
- {
- use Blamable;
- use CompanyOwned;
- use HasFactory;
-
- protected $table = 'document_defaults';
-
- protected $fillable = [
- 'company_id',
- 'type',
- 'logo',
- 'show_logo',
- 'number_prefix',
- 'number_digits',
- 'number_next',
- 'payment_terms',
- 'header',
- 'subheader',
- 'terms',
- 'footer',
- 'accent_color',
- 'font',
- 'template',
- 'item_name',
- 'unit_name',
- 'price_name',
- 'amount_name',
- 'created_by',
- 'updated_by',
- ];
-
- protected $casts = [
- 'type' => DocumentType::class,
- 'show_logo' => 'boolean',
- 'number_next' => TrimLeadingZeroCast::class,
- 'payment_terms' => PaymentTerms::class,
- 'font' => Font::class,
- 'template' => Template::class,
- 'item_name' => AsArrayObject::class,
- 'unit_name' => AsArrayObject::class,
- 'price_name' => AsArrayObject::class,
- 'amount_name' => AsArrayObject::class,
- ];
-
- protected $appends = [
- 'logo_url',
- ];
-
- protected function logoUrl(): Attribute
- {
- return Attribute::get(static function (mixed $value, array $attributes): ?string {
- return $attributes['logo'] ? Storage::disk('public')->url($attributes['logo']) : null;
- });
- }
-
- public function scopeType(Builder $query, string | DocumentType $type): Builder
- {
- return $query->where($this->qualifyColumn('type'), $type);
- }
-
- public function scopeInvoice(Builder $query): Builder
- {
- return $query->type(DocumentType::Invoice);
- }
-
- public function scopeRecurringInvoice(Builder $query): Builder
- {
- return $query->type(DocumentType::RecurringInvoice);
- }
-
- public function scopeBill(Builder $query): Builder
- {
- return $query->type(DocumentType::Bill);
- }
-
- public function scopeEstimate(Builder $query): Builder
- {
- return $query->type(DocumentType::Estimate);
- }
-
- public static function availableNumberDigits(): array
- {
- return array_combine(range(1, 20), range(1, 20));
- }
-
- public function getNumberNext(?bool $padded = null, ?bool $format = null, ?string $prefix = null, int | string | null $digits = null, int | string | null $next = null): string
- {
- [$number_prefix, $number_digits, $number_next] = $this->initializeAttributes($prefix, $digits, $next);
-
- return match (true) {
- $format && $padded => $number_prefix . $this->getPaddedNumberNext($number_next, $number_digits),
- $format => $number_prefix . $number_next,
- $padded => $this->getPaddedNumberNext($number_next, $number_digits),
- default => $number_next,
- };
- }
-
- public function initializeAttributes(?string $prefix, int | string | null $digits, int | string | null $next): array
- {
- $number_prefix = $prefix ?? $this->number_prefix;
- $number_digits = $digits ?? $this->number_digits;
- $number_next = $next ?? $this->number_next;
-
- return [$number_prefix, $number_digits, $number_next];
- }
-
- /**
- * Get the next number with padding for dynamic display purposes.
- * Even if number_next is a string, it will be cast to an integer.
- */
- public function getPaddedNumberNext(int | string | null $number_next, int | string | null $number_digits): string
- {
- return str_pad($number_next, $number_digits, '0', STR_PAD_LEFT);
- }
-
- public static function getAvailableItemNameOptions(): array
- {
- $options = [
- 'items' => 'Items',
- 'products' => 'Products',
- 'services' => 'Services',
- 'other' => 'Other',
- ];
-
- return array_map(translate(...), $options);
- }
-
- public static function getAvailableUnitNameOptions(): array
- {
- $options = [
- 'quantity' => 'Quantity',
- 'hours' => 'Hours',
- 'other' => 'Other',
- ];
-
- return array_map(translate(...), $options);
- }
-
- public static function getAvailablePriceNameOptions(): array
- {
- $options = [
- 'price' => 'Price',
- 'rate' => 'Rate',
- 'other' => 'Other',
- ];
-
- return array_map(translate(...), $options);
- }
-
- public static function getAvailableAmountNameOptions(): array
- {
- $options = [
- 'amount' => 'Amount',
- 'total' => 'Total',
- 'other' => 'Other',
- ];
-
- return array_map(translate(...), $options);
- }
-
- public function getLabelOptionFor(string $optionType, ?string $optionValue)
- {
- $optionValue = $optionValue ?? $this->{$optionType}['option'];
-
- if (! $optionValue) {
- return null;
- }
-
- $options = match ($optionType) {
- 'item_name' => static::getAvailableItemNameOptions(),
- 'unit_name' => static::getAvailableUnitNameOptions(),
- 'price_name' => static::getAvailablePriceNameOptions(),
- 'amount_name' => static::getAvailableAmountNameOptions(),
- default => [],
- };
-
- return $options[$optionValue] ?? null;
- }
-
- public function resolveColumnLabel(string $column, string $default, ?array $data = null): string
- {
- if ($data) {
- $custom = $data[$column]['custom'] ?? null;
- $option = $data[$column]['option'] ?? null;
- } else {
- $custom = $this->{$column}['custom'] ?? null;
- $option = $this->{$column}['option'] ?? null;
- }
-
- if ($custom) {
- return $custom;
- }
-
- return $this->getLabelOptionFor($column, $option) ?? $default;
- }
-
- protected static function newFactory(): Factory
- {
- return DocumentDefaultFactory::new();
- }
- }
|