| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215 | <?php
namespace App\Models\Setting;
use App\Casts\TrimLeadingZeroCast;
use App\Enums\DocumentType;
use App\Enums\Font;
use App\Enums\PaymentTerms;
use App\Enums\Template;
use App\Traits\Blamable;
use App\Traits\CompanyOwned;
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 = [
        '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->scopes(['type' => [DocumentType::Invoice]]);
    }
    public function scopeBill(Builder $query): Builder
    {
        return $query->scopes(['type' => [DocumentType::Bill]]);
    }
    public static function availableNumberDigits(): array
    {
        return array_combine(range(1, 20), range(1, 20));
    }
    public static function getNumberNext(?bool $padded = null, ?bool $format = null, ?string $prefix = null, int | string | null $digits = null, int | string | null $next = null, ?string $type = null): string
    {
        $initializeAttributes = new static;
        [$number_prefix, $number_digits, $number_next] = $initializeAttributes->initializeAttributes($prefix, $digits, $next, $type);
        if ($format) {
            return $number_prefix . static::getPaddedNumberNext($number_next, $number_digits);
        }
        if ($padded) {
            return static::getPaddedNumberNext($number_next, $number_digits);
        }
        return $number_next;
    }
    public function initializeAttributes(?string $prefix, int | string | null $digits, int | string | null $next, ?string $type): array
    {
        $number_prefix = $prefix ?? $this->getAttributeFromArray('number_prefix');
        $number_digits = $digits ?? $this->getAttributeFromArray('number_digits');
        $number_next = $next ?? $this->getAttributeFromArray('number_next');
        if ($type) {
            $attributes = static::getAttributesByType($type);
            $number_prefix = $attributes['number_prefix'] ?? $number_prefix;
            $number_digits = $attributes['number_digits'] ?? $number_digits;
            $number_next = $attributes['number_next'] ?? $number_next;
        }
        return [$number_prefix, $number_digits, $number_next];
    }
    public static function getAttributesByType(?string $type): array
    {
        $model = new static;
        $attributes = $model->newQuery()->type($type)->first();
        return $attributes ? $attributes->toArray() : [];
    }
    /**
     * 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 static 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;
    }
    protected static function newFactory(): Factory
    {
        return DocumentDefaultFactory::new();
    }
}
 |